当前位置:   article > 正文

关于ndk开发使用jni回掉java方法更新UI的问题_android jni 调用ui现场

android jni 调用ui现场


参考自:http://blog.sina.com.cn/s/blog_4b650d650100moda.html

重新整理,

原理:

应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件, 进行事件分发。

耗时的操作,放在一个子线程中,如果子线程涉及到UI更新,那就要用到handler,Android主线程是线程不安全的, 也就是说,更新UI只能在主线程中更新,子线程中操作是危险的。

Handler运行在主线程中(UI线程中),  它与子线程可以通过Message对象来传递数据, 这个时候,Handler就承担着接受子线程传过来的(子线程用sedMessage()方法传弟)Message对象,(里面包含数据)  , 把这些消息放入主线程队列中,配合主线程进行更新UI。

重点关注这里

MyHandler myHandler; 
static MyHandler mHandler;

要用这个mHandler才能保证回调的方法还在当前环境下,否则调不到,会报错。

代码如下:

  1. Java代码:
  2. package com.example.hellojni;
  3. public class callbyc extends Activity{
  4. TextView textpin;
  5. static Context context;
  6. MyHandler myHandler;
  7. static MyHandler mHandler;
  8. @Override
  9. public void onCreate(Bundle savedInstanceState)
  10. {
  11. super.onCreate(savedInstanceState);
  12. setContentView(R.layout.callbyc);
  13. JniInterface.nativeInit();
  14. textpin = (TextView)findViewById(R.id.textinfo);
  15. myHandler = new MyHandler();
  16. mHandler = myHandler;//重要,保存全局静态handler句柄,以便回掉的时候能找到该上下文
  17. MyThread m = new MyThread();
  18. new Thread(m).start();
  19. }
  20. class MyHandler extends Handler {
  21. public MyHandler() {
  22. }
  23. public MyHandler(Looper L) {
  24. super(L);
  25. }
  26. // 子类必须重写此方法,接受数据
  27. @Override
  28. public void handleMessage(Message msg) {
  29. // TODO Auto-generated method stub
  30. System.out.println("handleMessage。。。。。。");
  31. super.handleMessage(msg);
  32. // 此处可以更新UI
  33. Bundle b = msg.getData();
  34. String color = b.getString("color");
  35. System.out.println("color:"+color);
  36. textpin.setText(color);
  37. }
  38. }
  39. class MyThread implements Runnable {
  40. public void run() {
  41. try {
  42. Thread.sleep(1000);
  43. } catch (InterruptedException e) {
  44. // TODO Auto-generated catch block
  45. e.printStackTrace();
  46. }
  47. System.out.println("mThread。。。。。。。。");
  48. JniInterface.pinTest();//jni里面方法回调inputPin
  49. }
  50. }
  51. public void myCallbackFunc(String nMsg)
  52. {
  53. System.out.println("myCallbackFunc。。。。。。。。");
  54. Message msg = new Message();
  55. Bundle b = new Bundle();// 存放数据
  56. b.putString("color","我的");
  57. msg.setData(b);
  58. System.out.println("。。。。。。。。");
  59. // callbyc.this.myHandler.sendMessage(msg); // 向Handler发送消息,更新UI
  60. }
  61. private static void DispInfo(int value)
  62. {
  63. System.out.println("this is called by c value:"+value);
  64. //Toast.makeText(mainContext, "this is from c:"+value,Toast.LENGTH_LONG).show();
  65. }
  66. private String inputPin()
  67. {
  68. System.out.println("pls input pin");
  69. Message msg = new Message();
  70. Bundle b = new Bundle();// 存放数据
  71. b.putString("color","123456");
  72. msg.setData(b);
  73. mHandler.sendMessage(msg);//这里用静态mHandler才能在回掉该方法的时候正确执行
  74. //callbyc.this.myHandler.sendMessage(msg); // 向Handler发送消息,更新UI
  75. return "123456";
  76. //textpin.setText("bbb");
  77. }
  78. }
  79. jni代码:
  80. #include <jni.h>
  81. jmethodID gOnNativeID;
  82. JNIEnv* genv;
  83. jobject mObject;
  84. jclass mClass;
  85. JavaVM* gs_jvm;
  86. //native初始化函数
  87. void Java_com_example_hellojni_JniInterface_nativeInit(JNIEnv* env,jobject thiz)
  88. {
  89. printf("CalledForJni_nativeInit................");
  90. //获取类
  91. jclass clazz = (*env)->FindClass(env,"com/example/hellojni/callbyc");
  92. if(clazz==NULL)
  93. {
  94. printf("clazz IS NULL................");
  95. return;
  96. }
  97. mObject = (jobject)(*env)->NewGlobalRef(env,thiz);//永久保存mClass
  98. //获取方法ID
  99. gOnNativeID = (*env)->GetMethodID(env,clazz,"inputPin","()Ljava/lang/String;");
  100. if(gOnNativeID==NULL)
  101. {
  102. printf("gOnNativeID IS NULL................");
  103. return;
  104. }
  105. //得到JAVA VM,为了在其他没有传env参数的函数获取emv调用java方法
  106. (*env)->GetJavaVM(env,&gs_jvm);
  107. printf("CalledForJni_nativeInit end................");
  108. }
  109. void Java_com_example_hellojni_JniInterface_pinTest(JNIEnv* env,jobject thiz)
  110. {
  111. UTIL_OnlinePIN();
  112. }
  113. uchar UTIL_OnlinePIN()
  114. {
  115. char message[]="PLS INPUT PIN";
  116. int nStatus;
  117. JNIEnv *env;
  118. jstring recv;
  119. char *precvbuf;
  120. int i=0;
  121. //JNI_VERSION_1_2
  122. //nStatus = gs_jvm->GetEnv((void**)&env,JNI_VERSION_1_2);
  123. //得到当前线程的env,为了在没有传env参数的函数获取env调用java方法
  124. nStatus = (*gs_jvm)->AttachCurrentThread(gs_jvm, (void**)&env, NULL);
  125. if(nStatus<0)
  126. {
  127. printf("getenv faild");
  128. }
  129. printf("PLS INPUT PIN %s-%d",__FUNCTION__,__LINE__);
  130. //(*env)->CallStaticVoidMethod(env,mObject,gOnNativeID,99);
  131. recv = (*env)->CallObjectMethod(env,mObject,gOnNativeID);
  132. precvbuf = (*env)->GetStringUTFChars(env,recv,0);
  133. dumpData("recvpin",precvbuf,6);
  134. (*env)->ReleaseStringUTFChars(env,recv,precvbuf);
  135. printf("%s-%d",__FUNCTION__,__LINE__);
  136. //退出当前线程要调用DetachCurrentThread()释放对应的资源
  137. }


声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/笔触狂放9/article/detail/133181
推荐阅读
相关标签
  

闽ICP备14008679号