最近做一个app,需要调用同事提供的jni接口,封装过程中出现了一次崩溃,解决的时候顺便熟悉了下jni一些知识,在此记录下。
activity不是直接调用的jni,而是通过aidl接口和service通信,然后service来调用jni。有个接口要传入一个interface对象,下面是jni方法声明:
public native int selfTest(MyCallback callback);
MyCallback 是一个interface对象:
public interface MyCallback
{
public void onSelfTest(String info,int res);
};
在AIDL文件中申明接口:
interface MyAidlInterface {
int selfTest(MyCallback callback);
}
直接使用interface编译的时候会报错,提示接口未定义,解决办法是删除interface的定义文件(.java),新建一个aidl文件,申明方法:
interface MyCallback {
void onSelfTest(int progress,String info);
}
内容和.java文件是一样的,就是后缀变成.aidl。
jni方法都在JniManager.java文件中申明。
然后集成好了,编译通过,运行的时候通过aidl接口调用:
private MyCallback myCallback = new MyCallback () {
@Override
public void onProduce(int progress) throws RemoteException {
}
@Override
public void onSelfTest(int progress,String info) throws RemoteException {
}
@Override
public IBinder asBinder() {
return null;
}
};
jniManager.selfTest(myCallback );
直接崩溃掉,查看原因:
java.lang.NoSuchMethodError: no non-static method “Lxxx/xxx/xxx/xxx/xxx/ParameterActivity$1;.onSelfTest(Ljava/lang/String;I)V”
提示找不到onSelfTest方法,很茫然,activity中定义的Mycallback实现的方法都是androidstudio自动生成的,居然说找不到方法。后来又注意到括号里面的参数,上网一搜,这是jni特有的字段描述符(更详细的说明参考这里),意思是传入两个参数,一个是String,一个是int,函数类型是void,回到调用代码中,是这样的:
public void onSelfTest(int progress,String info) throws RemoteException {
}
参数位置搞反了!就是MyCallback .aidl中就把onSelfTest参数定义错了,修改aidl文件,重新implement接口,编译,运行,就OK了。查看jni程序,的确是申明了参数的:
xxxxxxxx = env->GetMethodID(env->GetObjectClass(callback),"onSelfTest","(Ljava/lang/String;I)V");
主要是记录一下解决的办法并加深印象,(Ljava/lang/String;I)V,()里面的内容是传入的参数说明,L是类说明,如果是String要在前面加Ljava/lang/,有些基本变量比如Boolean,int就不需要,“;”后面是第二个参数,“I”代表int,最后的V是void的缩写,代表返回类型为void。
2017.8.16补充
从jni申明java方法的时候,传的参数之间一般不需要用“;”来分割。
比如:
void onInform(byte cmd, int param);
从jni调用的时候是这样的:
env->GetMethodID(env->GetObjectClass(callback),"onInform","(BI)V");
“B”代表byte,”I”代表int,之间不需要“;”,只有String类型是自带了“;”,比如:
void onInform(String cmd, int param);
在jni中:
env->GetMethodID(env->GetObjectClass(callback),"onInform","(Ljava/lang/String;I)V");
从”L”到“;”都是String的一部分。