最近在公司做一个项目,想把之前Linux小组做了三四年的一个项目移植到Android平台上。如果让Android从头开始做,估计又是三四年时间去对接不同平台,而且还需要两个组同时维护,这样太不划算了。所以就打算把Linux上的程序移植到Android上,Android只需要提供一些特有的接口给C就好了,以后只维护一套代码,简单方便。
经过一天的尝试,最终成功完成了一个满足需求的Demo,整体结构如下:
整体来看也比较简单,Java调JNI,JNI启动Linux程序,Linux程序运行中需要操作界面时再用JNI的反射方法调用到Java层修改界面。
接下来看下具体实现:
代码结构
LinuxHandler.java 定义native方法和等待反射调用方法
Androidhandle.c Linux代码调用方法反射Java层
demo.c Linux小组同事提供的测试代码,里面有他们简化的业务逻辑
linuxhandle.c JNI方法
LinuxHandler.java
Java层启动
1 2 3 4 5 6 7 8 9 10 11 12
| public class LinuxHandle {
public native void startLinux();
public void speak(String msg) { Log.e("TEST", "speak C-> " + msg); }
public void showMsg(String msg) { Log.e("TEST", "showMsg C-> " + msg); } }
|
定义了三个方法,native方法startLinux()从Java层启动Linux程序,另外两个是模拟等待反射调用的操作界面方法。
linuxhandle.c
真正去启动Linux程序
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| #include <jni.h> #include "linuxhandle.h" #include "demo.h"
JavaVM *local_jvm = NULL; jobject local_object = NULL;
JNIEXPORT void JNICALL Java_com_***_linuxdemo_LinuxHandle_startLinux (JNIEnv *env, jobject object) { (*env)->GetJavaVM(env, &local_jvm); local_object = (*env)->NewGlobalRef(env, object); main(); }
|
Androidhandle.c
等待Linux程序调用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| #include <jni.h> #include "linuxhandle.h"
void showMessage_c(char *message) { extern JavaVM *local_jvm; JNIEnv *env = NULL; extern jobject local_object; (*local_jvm)->AttachCurrentThread(local_jvm, &env, NULL);
jclass jclass1 = (*env)->FindClass(env, "com/***/linuxdemo/LinuxHandle"); jmethodID methodID = (*env)->GetMethodID(env, jclass1, "showMsg", "(Ljava/lang/String;)V");
(*env)->CallVoidMethod(env, local_object, methodID, (*env)->NewStringUTF(env, message)); }
void speak_c(char *text) { LOGE("speak at C"); }
|
Demo.c
1 2 3 4 5 6 7
| #include "Androidhandle.h"
int main(void) { ...... speak_c("1234"); showMessage_c("adc"); }
|
我自己做完后也做了多次验证,并且导入第三方SO库测试也没有问题,但因为还只是个验证可行性的Demo,后面就要开始正式做移植的项目了,如果遇到了什么坑我会尽快补上。