jni调用 java和c是同个线程吗

Python016

jni调用 java和c是同个线程吗,第1张

是的,jni调用时, java和c是同个线程。

检测方法:在java和c中分别把当前线程的id输出出来查看。

Java获取线程Id:

Thread.currentThread().getId()

C中获取线程Id:

GetCurrentThreadId()

java 与 C++ 两种编程语言,它们之间的相互调用:

1、java 调用C++编写的dll,可使用JNI 或 Jawin 开源项目(推荐第二种方法)。

2、C++ 调用java 的变量、方法,通过JNI (Java Native Interface)与java类交互。

----操作步骤(只总结第二个)-----

(1) vc6.0编译C++程序,开发环境设置:工具--》选项--》工具,工具标签下:选择“include files”,加入头文件目录:C:\Program Files\Java\jdk1.5.0\include 和 C:\Program Files\Java\jdk1.5.0\include\win32 ;选择“Libary files"下,加入LIB目录:C:\Program Files\Java\jdk1.5.0\lib 。会编译成exe文件。

执行程序环境设置: Path环境变量加入:C:\Program Files\Java\jdk1.5.0\jre\bin\client (jvm.dll所在目录),若不加入path会提示,执行时找不到jvm.dll.

(2)GetStaticMethodID(cls,"main","([Ljava/lang/String)V")

//([Ljava/lang/String)V 是main()签名

在java程序目录下执行:javap -s -p ClassDemo (注:ClassDemo.java 已经编译)

取main 下面的语句 :: Signature: ([Ljava/lang/String)V

(3)附代码示例:

java程序

import java.io.*

public class DemoMain{

public static void main(String[] args) throws java.io.IOException, java.lang.NullPointerException

{

System.out.println("This is a test.")

}

}

C++程序:

#ifndef __cplusplus

#define __cplusplus

#endif

#include "jni.h"

#include <stdio.h>

#include <stdlib.h>

#include <windows.h>

#pragma comment (lib,"C:\\Program Files\\Java\\jdk1.5.0\\lib\\jvm.lib") // 动态调用lib

#pragma warning(disable: 4129) // 关闭 warning, 4129

void main() {

LoadLibrary("C:\\Program Files\\Java\jre1.5.0\\bin\\client\\jvm.dll")// 动态调用dll

JavaVM *jvm

JNIEnv *env

JavaVMInitArgs vm_args

JavaVMOption options[3]

options[0].optionString = "-Djava.compiler=NONE"

options[1].optionString = "-Djava.classpath=."

options[2].optionString = "-verbose:jni"

vm_args.version = JNI_VERSION_1_4

vm_args.nOptions = 3

vm_args.options = options

vm_args.ignoreUnrecognized = JNI_TRUE

jint res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args)// 创建虚拟机

if (res <0) {

fprintf(stderr, "Can't create Java VM\n")

exit(1)

}

jclass cls = env->FindClass("DemoMain")

if (cls == 0) printf("Sorry, I can't find the class")

fprintf(stdout, "This is invokeSimplified4.\n")

jmethodID get_main_id

if(cls != NULL)

{

get_main_id =env->GetStaticMethodID(cls,"main","([Ljava/lang/String)V")

fprintf(stdout, "This is invokeSimplified5.\n")

if(get_main_id != NULL )

{

jclass string = env->FindClass("java/lang/String")

jobjectArray args = env->NewObjectArray(0,string, NULL)

fprintf(stdout, "This is invokeSimplified6.\n")

int i = env->CallIntMethod(cls, get_main_id, args)

fprintf(stdout, i+ "This is invokeSimplified7.\n")

}

}

jvm->DestroyJavaVM()

fprintf(stdout, "Java VM destory\n")

}

JAVA以其跨平台的特性深受人们喜爱,而又正由于它的跨平台的目的,使得它和本地机器的各种内部联系变得很少,约束了它的功能。解决JAVA对本地操作的一种方法就是JNI。

JAVA通过JNI调用本地方法,而本地方法是以库文件的形式存放的(在WINDOWS平台上是DLL文件形式,在UNIX机器上是SO文件形式)。通过调用本地的库文件的内部方法,使JAVA可以实现和本地机器的紧密联系,调用系统级的各接口方法。

简单介绍及应用如下:

一、JAVA中所需要做的工作

在JAVA程序中,首先需要在类中声明所调用的库名称,如下:

static {

System.loadLibrary(“goodluck”)

}

在这里,库的扩展名字可以不用写出来,究竟是DLL还是SO,由系统自己判断。

还需要对将要调用的方法做本地声明,关键字为native。并且只需要声明,而不需要具 体实现。如下:

public native static void set(int i)

public native static int get()

然后编译该JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就会生成C/C++的头文件。

例如程序testdll.java,内容为:

public class testdll

{

static

{

System.loadLibrary("goodluck")

}

public native static int get()

public native static void set(int i)

public static void main(String[] args)

{

testdll test = new testdll()

test.set(10)

System.out.println(test.get())

}

}

用javac testdll.java编译它,会生成testdll.class。

再用javah testdll,则会在当前目录下生成testdll.h文件,这个文件需要被C/C++程序调用来生成所需的库文件。

二、C/C++中所需要做的工作

对于已生成的.h头文件,C/C++所需要做的,就是把它的各个方法具体的实现。然后编译连接成库文件即可。再把库文件拷贝到JAVA程序的路径下面,就可以用JAVA调用C/C++所实现的功能了。

接上例子。我们先看一下testdll.h文件的内容:

/* DO NOT EDIT THIS FILE - it is machine generated */

#include <jni.h>

/* Header for class testdll */

#ifndef _Included_testdll

#define _Included_testdll

#ifdef __cplusplus

extern "C" {

#endif

/*

* Class: testdll

* Method: get

* Signature: ()I

*/

JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass)

/*

* Class: testdll

* Method: set

* Signature: (I)V

*/

JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint)

#ifdef __cplusplus

}

#endif

#endif

在具体实现的时候,我们只关心两个函数原型

JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass)和

JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint)

这里JNIEXPORT和JNICALL都是JNI的关键字,表示此函数是要被JNI调用的。而jint是以JNI为中介使JAVA的int类型与本地的int沟通的一种类型,我们可以视而不见,就当做int使用。函数的名称是JAVA_再加上java程序的package路径再加函数名组成的。参数中,我们也只需要关心在JAVA程序中存在的参数,至于JNIEnv*和jclass我们一般没有必要去碰它。

好,下面我们用testdll.cpp文件具体实现这两个函数:

#include "testdll.h"

int i = 0

JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass)

{

return i

}

JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint j)

{

i = j

}

编译连接成库文件,本例是在WINDOWS下做的,生成的是DLL文件。并且名称要与JAVA中需要调用的一致,这里就是goodluck.dll 。把goodluck.dll拷贝到testdll.class的目录下,java testdll运行它,就可以观察到结果了。