20.Eclipse下Ndk开发(pthread开启线程调用Java方法)

简介: 本项目最终的目的是在pthread线程中,调用Java一个工具类得到多个uuid,然后调用类中另一个方法弹出toast,实现在c中获取安卓上下文对象Context编译native方法,生成头文件的一系列过程不再赘述,直接上代码,都在注释中PosixUtils:package com.

本项目最终的目的是在pthread线程中,调用Java一个工具类得到多个uuid,然后调用类中另一个方法弹出toast,实现在c中获取安卓上下文对象Context

编译native方法,生成头文件的一系列过程不再赘述,直接上代码,都在注释中

PosixUtils:

package com.example.ndk_pthread;

public class PosixUtils {
    
    static{
        System.loadLibrary("ndk_pthread");
    }
    /**
     * pthread开启子线程前的一些初始化操作,比如获取本类的jclass对象,生成需要的
     * 全局引用等等,在子线程中无法获取到类的jclass对象,就是这行代码,获取class必须要在主线程中
     * jclass uuidutils_class_tmp = (*env)->FindClass(env,"com/example/ndk_pthread/UUIDUtils");
     */
    public native void init();
    /**
     * 执行一些善后的操作,比如init方法中生成的全局引用的销毁等等
     */
    public native void destroy();
    /**
     * 子线程操作
     */
    public native void pthread(); 
}

UUIDUtils:

package com.example.ndk_pthread;

import java.util.UUID;

import android.content.Context;
import android.widget.Toast;

public class UUIDUtils {

    public static String get(){
        return UUID.randomUUID().toString();
    }
    
    public static void showToast(Context context){
        Toast.makeText(context, "c端弹出了toast", 0).show();
    }
    
}

编译生成的头文件com_example_ndk_pthread_UUIDUtils.h:

package com.example.ndk_pthread;

import java.util.UUID;

import android.content.Context;
import android.widget.Toast;

public class UUIDUtils {

    public static String get(){
        return UUID.randomUUID().toString();
    }
    
    public static void showToast(Context context){
        Toast.makeText(context, "c端弹出了toast", 0).show();
    }
    
}

MainActivity:

package com.example.ndk_pthread;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;

public class MainActivity extends Activity {

    private PosixUtils p;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        p = new PosixUtils();
        p.init();
    }

    public void start(View btn){
        p.pthread();
    }
    
    @Override
    protected void onDestroy() {
        super.onDestroy();
        //页面销毁时同时销毁c端的一些东西
        p.destroy();
    }

}

ndk_pthread.c

#include "com_example_ndk_pthread_PosixUtils.h"
#include <stdio.h>
#include <pthread.h>
#include <android/log.h>
#include <unistd.h>
#define LOGI(FORMAT,...) __android_log_print(ANDROID_LOG_INFO,"renzhenming",FORMAT,##__VA_ARGS__);
#define LOGE(FORMAT,...) __android_log_print(ANDROID_LOG_ERROR,"renzhenming",FORMAT,##__VA_ARGS__);

JavaVM *javaVM;
jobject uuidutils_jcls;
jmethodID get_mid;
jmethodID toastId;
jobject jcontext;
//动态库加载时会被调用执行,不需要我们手动调用
//兼容Android SDK 2.2之后,2.2没有这个函数
JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved){
    LOGI("%s","JNI_OnLoad");
    javaVM = vm;
    return JNI_VERSION_1_4;
}

/**
 * 目标:调用UUIDUtils类中的方法得到一个uuid
 */
void thread_fun(void* arg){
    //得到这个类UUIDUtils,需要用到JNIEnv,但是这是一个子线程,每个线程都有独立的JNIEnv,所以我们需要获取到
    //这个线程的JNIEnv,通过JavaVM关联当前线程,获取当前线程的JNIEnv,(*javaVM)->AttachCurrentThread(javaVM,&env,NULL);
    //那么就需要先获取到JavaVM
    //如何获取JavaVM?
    //1.在JNI_OnLoad函数中获取
    //2.(*env)->GetJavaVM(env,&javaVM);

    //的到JavaVM后,会得到env并赋值给变量
    JNIEnv* env = NULL;
    (*javaVM)->AttachCurrentThread(javaVM,&env,NULL);

    char* no = (char*)arg;
    int i;
    for (i = 0; i < 5; ++i) {
        LOGI("thread %s, i:%d",no,i);
        jobject uuid = (*env)->CallStaticObjectMethod(env,uuidutils_jcls,get_mid);
        char* uuid_char = (*env)->GetStringUTFChars(env,uuid,NULL);
        LOGI("%s",uuid_char);
        if(i == 4){
            goto end;
        }
        (*env)->ReleaseStringUTFChars(env,uuid,uuid_char);
        sleep(1);
    }
end:
    //如果直接将这两行写在循环之后是有问题的,当i=4的时候,线程直接退出,导致没有执行DetachCurrentThread,报异常
    (*javaVM)->DetachCurrentThread(javaVM);
    pthread_exit((void*)0);
}
/**
 * 当在JNI调用Android自带的类时,经常需要传入Context参数,那怎么在JNI层获取Context呢?
 * 我们知道Application和Activity是Context的子类,由于每个Activity对应的Context是不一样的,
 * 所以一般情况下我们使用Application的Context,它在整个程序中只有一个实例。所以现在问题就变成了
 * 怎么在JNI中获取Application呢?
 * Android APP在启动时会创建一个Activity Thread作为主线程,只要程序存活,这个线程就一直存在,
 * 所以我们可以考虑从Activity Thread中获取Application,查看Activity Thread的源码发现,
 * 它提供了一个方法可以获取Application,如下:
 *
 * public Application getApplication() {
 *    return mInitialApplication;
 * }
 *
 * 也就是说我们只需要获取到Activity Thread的对象即可,Activity Thread提供了一个静态方法用于获取其实例,如下:
 *
 * public static ActivityThread currentActivityThread() {
 *    return sCurrentActivityThread;
 * }
 *
 * 至此获取Context的步骤已经很清晰了
 */
JNIEXPORT void JNICALL Java_com_example_ndk_1pthread_PosixUtils_init
(JNIEnv *env, jobject jobj){
    /**
     * 打印uuid
     */
    //获取class必须要在主线程中
    jclass uuidutils_class_tmp = (*env)->FindClass(env,"com/example/ndk_pthread/UUIDUtils");
    //创建全局引用
    uuidutils_jcls = (*env)->NewGlobalRef(env,uuidutils_class_tmp);
    //获取jmethodId也可以在子线程中
    get_mid = (*env)->GetStaticMethodID(env,uuidutils_jcls,"get","()Ljava/lang/String;");
    /**
     * show toast
     */
    //获取jmethodId也可以在子线程中
    toastId = (*env)->GetStaticMethodID(env,uuidutils_jcls,"showToast","(Landroid/content/Context;)V");

    //获取Activity Thread的实例对象
    jclass activityThread = (*env)->FindClass(env,"android/app/ActivityThread");
    jmethodID currentActivityThread = (*env)->GetStaticMethodID(env,activityThread, "currentActivityThread", "()Landroid/app/ActivityThread;");
    jobject at = (*env)->CallStaticObjectMethod(env,activityThread, currentActivityThread);
    //获取Application,也就是全局的Context
    jmethodID getApplication = (*env)->GetMethodID(env,activityThread, "getApplication", "()Landroid/app/Application;");
    jobject context = (*env)->CallObjectMethod(env,at, getApplication);

    jcontext = (*env)->NewGlobalRef(env,context);
}


JNIEXPORT void JNICALL Java_com_example_ndk_1pthread_PosixUtils_destroy
(JNIEnv *env, jobject jobj){
    //释放全局引用
    (*env)->DeleteGlobalRef(env,uuidutils_jcls);
}

JNIEXPORT void JNICALL Java_com_example_ndk_1pthread_PosixUtils_pthread
  (JNIEnv *env, jobject jobj){
    LOGI("%s","begin");
    //获取JavaVM的第一种方式,本项目中我们采用在JNI_OnLoad中获取的方式
    //(*env)->GetJavaVM(env,&javaVM);
    pthread_t tid;
    pthread_create(&tid,NULL,thread_fun,(void*)"NO1");

    //调用Java函数show toast
    (*env)->CallStaticVoidMethod(env,uuidutils_jcls,toastId,jcontext);
}

Android.mk

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := ndk_pthread
LOCAL_SRC_FILES := ndk_pthread.c
LOCAL_LDLIBS := -llog
include $(BUILD_SHARED_LIBRARY)
相关文章
|
2天前
|
缓存 监控 Java
如何运用JAVA开发API接口?
本文详细介绍了如何使用Java开发API接口,涵盖创建、实现、测试和部署接口的关键步骤。同时,讨论了接口的安全性设计和设计原则,帮助开发者构建高效、安全、易于维护的API接口。
16 4
|
8天前
|
SQL Java 程序员
倍增 Java 程序员的开发效率
应用计算困境:Java 作为主流开发语言,在数据处理方面存在复杂度高的问题,而 SQL 虽然简洁但受限于数据库架构。SPL(Structured Process Language)是一种纯 Java 开发的数据处理语言,结合了 Java 的架构灵活性和 SQL 的简洁性。SPL 提供简洁的语法、完善的计算能力、高效的 IDE、大数据支持、与 Java 应用无缝集成以及开放性和热切换特性,能够大幅提升开发效率和性能。
|
8天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
22 2
|
8天前
|
监控 Java 数据库连接
在Java开发中,数据库连接管理是关键问题之一
在Java开发中,数据库连接管理是关键问题之一。本文介绍了连接池技术如何通过预创建和管理数据库连接,提高数据库操作的性能和稳定性,减少资源消耗,并简化连接管理。通过示例代码展示了HikariCP连接池的实际应用。
13 1
|
2天前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
6 0
|
2天前
|
Java API Android开发
kotlin和java开发优缺点
kotlin和java开发优缺点
10 0
WK
|
8天前
|
开发框架 移动开发 Java
C++和Java哪个更适合开发移动应用
本文对比了C++和Java在移动应用开发中的优劣,从市场需求、学习难度、开发效率、跨平台性和应用领域等方面进行了详细分析。Java在Android开发中占据优势,而C++则适合对性能要求较高的场景。选择应根据具体需求和个人偏好综合考虑。
WK
17 0
|
30天前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
38 1
C++ 多线程之初识多线程
|
15天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
13 3
|
15天前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
12 2

推荐镜像

更多