Android Jni

初学Jni总是遇到java.lang.UnsatisfiedLinkError: Native method not found,而程序又实在看不出有什么问题,以下记录导致这个问题的过程与原因。

1.使用eclipse ->Android Tools ->add native support,自动生成jni所需要的目录与make文件与一个源文件,但是默认生成的是cpp文件。而按照jni规则,使用静态注册方式写好程序后编译没问题,运行时却总报错,网上一查,有人说是使用c规则编译的,所以找不到方法。把jni方法用extern "C"{}包起来,异常就没了。

extern "C"
{
void Java_com_example_androidjni_MainActivity_sayHello()
{
	LOG("say----------- Hello!");
	say();
}
}


2.使用动态注册的方式发现总是找不到方法,最后发现JNI_OnLoad写成了JNI_Onload,所以找不到。JNI_OnLoad会在java LoadLibrary时被调用,在此方法里打log会输出。下面给出动态注册的必须方法,默认一切都是正常的。

void sayHello(JNIEnv *env, jobject thiz) {
	LOG("say Hello!>>");
}
const char *name = "com/example/androidjni/MainActivity";
JNINativeMethod methods[] = {{"sayHello", "()V", (void*)sayHello}};
JNIEXPORT jint JNI_OnLoad(JavaVM* vm,void* reserved)
{
	JNIEnv *env = 0;
	vm->GetEnv((void**)&env,JNI_VERSION_1_4);
	LOG("INIT11111111111111>>");
	LOG("%d,%d",sizeof(methods),sizeof(methods[0]));
	jclass clazz = env->FindClass(name);
	env->RegisterNatives(clazz,methods,sizeof(methods)/sizeof(methods[0]));

	return JNI_VERSION_1_4;
}

上面sayHello中的第一第二个参数可以省略。如果java中的方法是static方法则第二个参数为jclass,否则jobject。代表调用此函数的java对象或者类。


上面的例子不够典型,再举一例,如:

public native String readString(int type);
对应的methods应该
{"readString", "(I)Ljava/lang/String;", (void*)readString}
注意,最后的一个readString是对应c/c++里的函数,不是死的,可自定义名字。第2个参数中的Ljava/lang/String后面有个“;”符号。

如果不是String,是其他类对象,写法差不多,如BItmap对象就是Landroid/Graphics/Bitmap;。

类型签名

Java类型

Z

boolean

B

byte

C

char

S

short

I

int

J

long

F

float

D

double

Lfully-qualified-class;

全限定的类

[type

type[]

(arg-types) ret-type

方法类型 




3、上面讲了如何java调用C++,下面讲如何C++调用java。

在java调用C++方法时,第2个参数jobject代表的是调用此方法的java对象,如下是java调用了c++中的一个方法,那个方法调用了如下代码

	jstring str = mEnv->NewStringUTF("java hello world");
	jclass clazz = mEnv->FindClass(className);
	jmethodID mid = mEnv->GetMethodID(clazz,"say","(Ljava/lang/String;Ljava/lang/Object;)V");
	mEnv->CallVoidMethod(mJobject,mid,str,mJobject);
	mEnv->DeleteLocalRef(str);
上面的代码执行结果就是调用了java的如下方法
	public void say(String str,Object en){
		System.out.println(str);
		System.out.println(en.getClass().getName());
	}
say方法中的第2个参数与C++中的代码必须对应,Object可以代表任何类,此处也可以写实际的类,不过C++的java/lang/Object必须也要变成实际的包类。


4、如何获取java中的参数与改变参数

接着3来,age是java类中的一个全局对象

	jfieldID fid = mEnv->GetFieldID(clazz,"age","I");
	jint a = mEnv->GetIntField(mJobject,fid); //获取age 的值
	mEnv->SetIntField(mJobject,fid,12);		 //设置age的值为12.


本博客只教人能顺利使用,知道关键的方法,不考虑稳定性、异常情况。


应该注意的地方:

1.java对象不要直接引用,通过jobject mObject = mEnv->NewGlobalRef(myobject);得到全局引用,这样就算java层这个对象已经没有引用了,内存也不会被释放。

2.要及时释放内存,mEnv->DeleteLocalRef(str); 为立即释放str所占用的内存,如果不调用此方法,函数执行完成后str的内存一样会被释放。但是如果函数内有大循环,循环中进行了new string的操作,不进行主动释放最后所占用的内存就比较大了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值