转载自:http://www.cnblogs.com/lknlfy/archive/2012/03/16/2400786.html

一、概述

      JNI编程和Linux上的C/C++编程还是挺相似的,每次java调用JNI中的函数时都会传入有关JVM的一些参数(如JNIEnv,jobject),每次JNI回调java中的方法时都要通过JVM的有关参数来实现,当在JNI中涉及到多线程的话还是有一些不一样的地方,就是要在子线程函数里使用AttachCurrentThread()和DetachCurrentThread()这两个函数,在这两个函数之间加入回调java方法所需要的代码。

 

二、要求

     掌握JNI多线程编程的方法。

 

三、实现

     新建工程MyThread,修改main.xml文件,在里面只有一个Button,如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <Button 
        android:id="@+id/button"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="启动JNI线程"
        />

</LinearLayout>

修改MyThreadActivity.java文件,实现按钮的监听,在里面调用JNI中的函数来启动JNI中的线程,比较简单,如下:

package com.nan.thread;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;

public class MyThreadActivity extends Activity 
{
    private Button mButton = null; 
    
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        mButton = (Button)this.findViewById(R.id.button);
        //按钮监听
        mButton.setOnClickListener(new View.OnClickListener() 
        {
            
            @Override
            public void onClick(View v) 
            {
                // TODO Auto-generated method stub
                //调用JNI中的函数来启动JNI中的线程
                mainThread();
                
            }
        });
        //初始化JNI环境
        setJNIEnv();
    }
         
    //由JNI中的线程回调
    private static void fromJNI(int i)
    {
        Log.v("Java------>", ""+i);
    }
    
    //本地方法
    private native void mainThread();
    private native void setJNIEnv(); 
    
    static
    {
        //加载动态库
        System.loadLibrary("JNIThreads");
    }
    
}

编写JNI_Thread.c文件,在mainThread()函数里启动5个子线程,在子线程函数里回调java中的静态方法fromJNI()来输出当前子线程是第几个被启动的线程。完整的内容如下:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<pthread.h>

#include<jni.h>
#include<android/log.h>

#define LOGI(...) ((void)__android_log_print(ANDROID_LOG_INFO, "native-activity", __VA_ARGS__))
#define LOGW(...) ((void)__android_log_print(ANDROID_LOG_WARN, "native-activity", __VA_ARGS__))
#define LOGE(...) ((void)__android_log_print(ANDROID_LOG_ERROR, "native-activity", __VA_ARGS__))

//线程数
#define NUMTHREADS 5

//全局变量
JavaVM *g_jvm = NULL;
jobject g_obj = NULL;


void *thread_fun(void* arg)
{
    JNIEnv *env;
    jclass cls;
    jmethodID mid;

    //Attach主线程
    if((*g_jvm)->AttachCurrentThread(g_jvm, &env, NULL) != JNI_OK)
    {
        LOGE("%s: AttachCurrentThread() failed", __FUNCTION__);
        return NULL;
    }
    //找到对应的类
    cls = (*env)->GetObjectClass(env,g_obj);
    if(cls == NULL)
    {
        LOGE("FindClass() Error.....");
        goto error; 
    }
    //再获得类中的方法
    mid = (*env)->GetStaticMethodID(env, cls, "fromJNI", "(I)V");
    if (mid == NULL) 
    {
        LOGE("GetMethodID() Error.....");
        goto error;  
    }
    //最后调用java中的静态方法
        (*env)->CallStaticVoidMethod(env, cls, mid ,(int)arg);
    

error:    
    //Detach主线程
    if((*g_jvm)->DetachCurrentThread(g_jvm) != JNI_OK)
    {
        LOGE("%s: DetachCurrentThread() failed", __FUNCTION__);
    }
    

    pthread_exit(0);
}

//由java调用以创建子线程
JNIEXPORT void Java_com_nan_thread_MyThreadActivity_mainThread( JNIEnv* env, jobject obj)
{
    int i;
    pthread_t pt[NUMTHREADS];
    
    for (i = 0; i < NUMTHREADS; i++)
        //创建子线程
        pthread_create(&pt[i], NULL, &thread_fun, (void *)i);
}


//由java调用来建立JNI环境
JNIEXPORT void Java_com_nan_thread_MyThreadActivity_setJNIEnv( JNIEnv* env, jobject obj)
{
    //保存全局JVM以便在子线程中使用
    (*env)->GetJavaVM(env,&g_jvm);
    //不能直接赋值(g_obj = obj)
    g_obj = (*env)->NewGlobalRef(env,obj);
}


//当动态库被加载时这个函数被系统调用
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved)
{
    JNIEnv* env = NULL;
    jint result = -1;    

    //获取JNI版本
    if ((*vm)->GetEnv(vm, (void**)&env, JNI_VERSION_1_4) != JNI_OK) 
    {
        LOGE("GetEnv failed!");
            return result;
    }

    return JNI_VERSION_1_4;
}

最后,编写Android.mk文件:

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := JNIThreads
LOCAL_SRC_FILES := JNI_Threads.c

LOCAL_LDLIBS    := -llog

include $(BUILD_SHARED_LIBRARY)

注意:当在一个线程里面调用AttachCurrentThread后,如果不需要用的时候一定要DetachCurrentThread,否则线程无法正常退出。

posted on 2014-12-31 13:02  ip小子  阅读(2481)  评论(0编辑  收藏  举报