轉載:http://zzqrj.iteye.com/blog/1285262
相關鏈接可供參考:
http://blog.csdn.net/mr_raptor/article/details/7401178/
http://www.cnblogs.com/shaweng/p/4013320.html
http://blog.csdn.net/heyabo/article/details/7927713
1. JNIEnv對象
- 對於本地函數
JNIEXPORT void JNICALL Java_video1_TestNative_sayHello(JNIEnv * env, jobject obj){
cout<<"Hello Native Test !"<<endl;
}
JNIEnv類型代表Java環境。通過這個JNIEnv*指針,就可以對Java端的代碼進行操作。如,創建Java類得對象,調用Java對象的方法,獲取Java對象的屬性等。
JNIEnv的指針會被JNI傳送到本地方法的實現函數中來對Java端的代碼進行操作
JNIEnv類中的函數:
NewObject/NewString/New<TYPE>Array
Get/Set<TYPE>Field
Get/SetStatic<TYPE>Field
Call<TYPE>Method/CallStatic<TYPE>Method
2. Java數據類型與C/C++數據類型的對應關係
Java類型 別名 本地類型 字節(bit)
boolean jboolean unsigned char 8, unsigned
byte jbyte signed char 8
char jchar unsigned short 16, unsigned
short jshort short 16
int jint long 32
long jlong __int64 64
float jfloat float 32
double jdouble double 64
void void n/a
Object _jobject *jobject
3. 獲取jclass
爲了能夠在C/C++使用Java類,jni.h頭文件中專門定義了jclass類型來表示Java中的Class類
jclass的取得:
JNIEnv類中有如下幾個簡單的函數可以取得jclass
jclass FindClass(const char* clsName)
jclass GetObjectClass(jobject obj)
jclass GetSuperClass(jclass obj)
FindClass 會在classpath系統環境變量下尋找類,需要傳入完整的類名,注意包與包之間是用"/"而不是"."來分割
如:jclass cls_string= env->FindClass(“java/lang/String”);
4. 本地代碼訪問Java類中的屬性與方法
JNI在jni.h頭文件中定義了jfieldID,jmethodID類表示Java端的屬性和方法
在訪問或設置Java屬性的時候,首先就要現在本地代碼中取得代表Java屬性的jfieldID,然後才能在本地代碼中進行Java屬性操作。
同樣,在需要調用Java端的方法時,也需要取得代表方法的jmethodID才能進行Java方法調用
JNIEnv獲取相應的fieldID和jmethodID的方法:
GetFieldID/GetMethodID
GetStaticFieldID/GetStaticMethodID
GetMethodID也可以取得構造函數的jmethodID。創建Java對象時調用指定的構造函數。
如:env->GetMethodID(data_Clazz,””,”()V”)
5. sign簽名
對於 jfieldID GetFieldID(jclass clazz, const char *name, const char *sign)
clazz代表該屬性所在的類,name表示方法名稱,sign是簽名
例如TestNative類中有兩個重載方法:
package video1;
public class TestNative{
public void methodTest(int i){
System.out.println(i);
}
public void methodTest(double d){
System.out.println(d);
}
}
/*
在C/C++代碼中調用其中一個methodTest方法:
首先取得要調用方法所在的類
jclass clazz_TestNative = env->FindClass("video1/TestNative");
//取得jmethodID
jmethodID id_func = env->GetMethodID(clazz_TestNative,"methodTest","");
sign用於指定取得的屬性/方法的類型
如果sign指定爲(I)V,則取回void methodTest(int)的methodID
如果sign指定爲(D)V,則取回void methodTest(double)的methodID
*/
簽名sign
用來表示要取得的屬性/方法的類型
類型 相應的簽名
boolean Z
byte B
char C
short S
int I
long L
float F
double D
void V
object L用/分隔包的完整類名: Ljava/lang/String;
Array [簽名 [I [Ljava/lang/Object;
Method (參數1類型簽名 參數2類型簽名···)返回值類型簽名
使用javap命令來產生簽名
javap -s -p [full class Name]
-s 表示輸出簽名信息
-p 同-private,輸出包括private訪問權限的成員信息
例子:
C:\E\java\workspaces\myeclipseblue\JNITest\bin>javap -s -private video1.TestNative
Compiled from "TestNative.java"
public class video1.TestNative extends java.lang.Object{
public java.lang.String name;
Signature: Ljava/lang/String;
public video1.TestNative();
Signature: ()V
public int signTest(int, java.util.Date, int[]);
Signature: (ILjava/util/Date;[I)I
public native void sayHello();
Signature: ()V
public static void main(java.lang.String[]);
Signature: ([Ljava/lang/String;)V
}
6. 本地方法調用Java方法的完整示例:
package video1;
import java.util.Date;
public class TestNative {
public String name="Test";
public int number =100;
public int signTest(int i,Date date,int[] arr){
System.out.println("Sign Test");
return 0;
}
//native關鍵字修飾的方法,其內容是C/C++編寫的,java中不必爲它編寫具體的實現
public native void sayHello();
public static void main(String[] args) {
System.loadLibrary("NativeCode");
TestNative tn = new TestNative();
tn.sayHello();
}
}
C/C++代碼
#include "video1_TestNative.h"
#include <iostream>
using namespace std;
JNIEXPORT void JNICALL Java_video1_TestNative_sayHello(JNIEnv * env, jobject obj){
cout<<"Hello Native Test !"<<endl;
//因爲test不是靜態函數,所以傳進來的就是調用這個函數的對象
//否則就傳入一個jclass對象表示native()方法所在的類
jclass native_clazz = env->GetObjectClass(obj);
//得到jfieldID
jfieldID fieldID_prop = env->GetFieldID(native_clazz,"name","Ljava/lang/String;");
jfieldID fieldID_num = env->GetFieldID(native_clazz,"number","I");
//得到jmethodID
jmethodID methodID_func=env->GetMethodID(native_clazz,"signTest","(ILjava/util/Date;[I)I");
//調用signTest方法
env->CallIntMethod(obj,methodID_func,1L,NULL,NULL);
//得到name屬性
jobject name = env->GetObjectField(obj,fieldID_name);
//得到number屬性
jint number= env->GetIntField(obj,fieldID_num);
cout<<number<<endl;//100
//修改number屬性的值
env->SetIntField(obj,fieldID_num,18880L);
number= env->GetIntField(obj,fieldID_num);
cout<<number<<endl;//18880
}
編譯source.cpp,執行TestNative.java類。