1. 报错:

  1. -20 ::): threadid=: thread exiting with uncaught exception (group=0x4001d400)
  2. -20 ::): FATAL EXCEPTION: GLThread
  3. -20 ::: E/AndroidRuntime(): java.lang.CloneNotSupportedException: Class doesn't implement Cloneable
  4. -20 ::: E/AndroidRuntime():    at java.lang.Object.clone(Object.java:)
  5. -20 ::: E/AndroidRuntime():    at com.jnitest.nativetest(Native Method)

2. 现场说明:

在进行android项目开发时, 想用java代码调用jni的test函数,, 而且再通过c的callCallBack函数, 回调到java的onCallback函数., 发生以上错误.

3. 代码:(这里仅仅列出重要的,关键的代码.)

 java代码:

public class NativeClass {

private static String TAG= "NativeClass";

static {

        System.loadLibrary("test-jni");

    }

public static int onCallBack(int event, int type,String str) {  //回调回来的回调函数

  Log.i(TAG, "onCallBack======= " + event);

  Log.i(TAG, "onCallBack======= " + type);

Log.i(TAG, "onCallBack======= " + str);

return event;

}

public static native int test();  //native函数

c代码:

jobject my_obj;

JNIEnv* my_env;

extern "C" int callCallBack(int event, int type, const char *str) ;

extern "C"

{

//用于调用java的callback

int  callCallBack(int event, int type, const char *str)

{

    int err = 1000;

 jclass objClass = (my_env)->FindClass("com/jnitest/native/NativeClass");

 if(!objClass) {

  return -1;

 }

 //获取并调用java层的onCallBack函数

 jmethodID methodId = (my_env)->GetStaticMethodID(objClass, "onCallBack", "(IILjava/lang/String;)I");

 if (methodId == 0) {

  LOGDV("here can not find method %s\n", "onCallBack");

 } else {

  jstring data = (update_env)->NewStringUTF(str));

  

  err = (my_env)->CallStaticIntMethod(objClass, methodId, event, type, data);

  if(data)  my_env->DeleteLocalRef(data);

 }

my_env->DeleteLocalRef(objClass);

 

 

 return err;

}

//jni native函数.

JNIEXPORT jstring JNICALL

Java_com_jnitest_native_test( JNIEnv* env, jobject thiz,jstring testStr )

{

    my_env = env; //必须保存,用于在回调的时候获取回调函数所在的class.

 my_obj = thiz;  //在本程序中,没实用到这个变量.

    const char *test_char = (env)->GetStringUTFChars(testStr, NULL);

 

 //调用callCallBack

 int i = callCallBack(0,1,test_char);

return (env)->NewStringUTF(test_char);

}

}

測试代码:

maintest.java:

void testFunc() {

String s = NativeClass.test("ssss");

Log.i("jniTest", " testFunc  s = " + s);

}

測试结果:

onCallBack=======0

onCallBack=======1

onCallBack=======ssss

testFunc  s =ssss

4. 分析与 解决:

分析,:

依据错误提示:

1).  有clone相关错误;

2)   就发生在NativeClass的test函数中.

3). 那么, 是哪个类没有进行clone?

或者, 根本就不是clone的原因?

4). NativeClass中,没有数据成员, 理论上讲,就不须要进行克隆.所以,应该不是NativeClass克隆的原因.

5). 在jni的test函数中,有对于my_obj的赋值,会不会是这个变量的赋值导致的呢?

解决:  因为my_obj没实用到(本来赋值后想用的), 并且可能会引起错误,所以,将这个变量的声明和赋值语句都去掉,

再測试,程序正常执行.

5. 总结:

1)  由java调用c时,一定要注意參数的正确传递.

2)  本例展示了怎样用java调用c,再由c调用到java的过程.

3) 本例来源于android程序开发.

Class doesn't implement Cloneable之怪象的更多相关文章

  1. [FindBugs分析记录]Class defines clone() but doesn't implement Cloneable

    官网解释: This class defines a clone() method but the class doesn't implement Cloneable. There are some ...

  2. Java 深拷贝、浅拷贝及Cloneable接口

    Cloneable接口是一个空接口,仅用于标记对象,Cloneable接口里面是没有clone()方法,的clone()方法是Object类里面的方法!默认实现是一个Native方法 protecte ...

  3. 实现 Cloneable 需要注意

    产品Product里面包含BaseInfo对象:Product(productName,companyName,baseinfo)如果implement Cloneable  需要实现   注意强转类 ...

  4. Cloneable注解使用

    使用 clone()方法的类必须 implement Cloneable 如果没有继承,clone()方法会报错 java.lang.CloneNotSupportedException异常

  5. java开发——Cloneable接口、clone()方法和深浅拷贝

    1.实现Cloneable接口表明该类的对象是允许克隆的. 2.允许克隆的意思是:可以调用clone()方法. 3.深拷贝还是浅拷贝,取决于如何重写Object的clone()方法. 4.原对象和克隆 ...

  6. 深拷贝、浅拷贝与Cloneable接口

    深拷贝与浅拷贝 浅拷贝 public class Student implements Cloneable{ Integer a; Integer b; @Override protected Obj ...

  7. Java面试葵花宝典

    面向对象的特征有哪些方面  1. 抽象:抽象就是忽略一个主题中与当前目标2. 无关的那些方面,3. 以便更充分地注意与当前目标4. 有关的方面.抽象并不5. 打算了解全部问题,而6. 只是选择其中的一 ...

  8. java面试题小全

    面向对象的特征有哪些方面   1. 抽象:抽象就是忽略一个主题中与当前目标2. 无关的那些方面,3. 以便更充分地注意与当前目标4. 有关的方面.抽象并不5. 打算了解全部问题,而6. 只是选择其中的 ...

  9. .Net面试葵花宝典

    1.                面向对象的特征有哪些方面    抽象:抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问题,而只是选择其中 ...

随机推荐

  1. 我要好offer之 字符串相关大总结

    1. str*系列手写代码 a. 一定要注意末尾'\0'的处理,切记切记 b. 一定要对输入做有效性判断,多用断言就是了 int Strlen(const char* str) { assert(st ...

  2. POJ 3037 Skiing

    Skiing Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4810   Accepted: 1287   Special ...

  3. c#.net分类上升达人~~~呵呵。。。

    原文发布时间为:2008-11-11 -- 来源于本人的百度文章 [由搬家工具导入] 觉得自己蛮无聊的~~~~~~~~~(>_<)~~~~

  4. 标准C程序设计七---101

    Linux应用             编程深入            语言编程 标准C程序设计七---经典C11程序设计    以下内容为阅读:    <标准C程序设计>(第7版) 作者 ...

  5. MSB与LSB Big Endian Little Endian

    Most Significant Bit, Last(Least) Significant Bit 最高有效位(MSB) 指二进制中最高值的比特.在16比特的数字音频中,其第1个比特便对16bit的字 ...

  6. Python入门--19--else语句、with语句

    1.else与while连用: x=input('请输出一个整数:') while x>0 x=x-2 print(x) else: print('x已经小于等于零了!') 2.else与try ...

  7. Glide加载图片问题记录

    Glide加载图片相比于Picasso而言性能较好,又比Fresco轻巧,而且又支持加载gif动图,是Google 推荐.专注平滑的滚动.简单易用.可扩展的一款图片加载框架.但是使用时还是会遇到一些问 ...

  8. Network | DHCP

    动态主机设置协议(Dynamic Host Configuration Protocol, DHCP)是一个局域网的网络协议,使用UDP协议工作,主要有两个用途: 给内部网络或网络服务供应商自动分配I ...

  9. python生成器、迭代器、__call__、闭包简单说明

    1.生成器 这种一边循环一边计算的机制,称为生成器:generator,最简单的方法是把生成式的[]改为(). >>> l=(x * x for x in range(1, 11) ...

  10. JDK1.8和Spring 3.2.0 的坑

    上午 org.apache.catalina.core.StandardContext listenerStart严重: Exception sending context initialized e ...