Android NDK开发入门

来源:互联网 发布:百度知道刷赞软件 编辑:程序博客网 时间:2024/05/17 23:15

        
NDKADTJNI

网上一大堆的都是使用javah命令来生成头文件来完成JNI编写,但其实ADT集成NDK后,点点鼠标就可以了,懒人方法网上介绍很少,这边主要讲懒人JNI开发。

对于ADT配置NDK,请个人google或查看Android developer,这边不多做介绍。

1. 新建一个Android工程,我这边取名为JNI_Learn,一键生成后代码片段如下:

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. public class JNI extends ActionBarActivity {  
  2.       
  3.     static{  
  4.         System.loadLibrary("JNI_Learn");  
  5.     }  
  6.       
  7.     public native int plus(int x, int y);  
  8.   
  9.     @Override  
  10.     protected void onCreate(Bundle savedInstanceState) {  
  11.         super.onCreate(savedInstanceState);  
  12.         setContentView(R.layout.activity_jni);  
  13.           
  14.         if (savedInstanceState == null) {  
  15.             getSupportFragmentManager().beginTransaction()  
  16.                     .add(R.id.container, new PlaceholderFragment())  
  17.                     .commit();  
  18.         }  
  19.           
  20.         Log.d("JNITest""3+5=" + plus(35));  
  21.           
  22.     }  
高亮部分是我自己添加部分,简单解释一下。

loadLibrary的话,里面的名字稍后我会再标注一下,这边主要是native层生成的so库的名字,需要去掉lib前缀和.so后缀。

声明native方法要在访问权限之后,其他声明之前,加一个native标记此方法实现在native层。

使用plus方法与正常使用无异。

2. 之后就是懒人操作了,在工程上右击鼠标,选择Android Tools->Add Native Support...


之后查看工程里面有两个文件


第一个是JNI_Learn.cpp,这个稍后详述。第二个是Android.mk,在Android编译中,都会寻找目录下是否存在Android.mk,然后根据这个文件进行编译。里面编写了具体的编译规则,打开文件后,内容如下:

[plain] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. LOCAL_PATH := $(call my-dir)  
  2.   
  3. include $(CLEAR_VARS)  
  4.   
  5. LOCAL_LDLIBS := -llog  
  6.   
  7. LOCAL_MODULE    := JNI_Learn  
  8. LOCAL_SRC_FILES := JNI_Learn.cpp  
  9.   
  10. include $(BUILD_SHARED_LIBRARY)  
除加红部分外,其他都为自动生成,加红部分是我添加为了使用native的log方法
其中LOCAL_PATH表示编译源文件的路径,LOCAL_LDLIBS表示编译模块时需要附加的连接器选项,LOCAL_MODULE表示最终编译出来模块的名称,LOCAL_SRC_FILES表示需要编译的源文件,include $(BUILD_SHARED_LIBRARY)表示最终编译成一个共享库文件。


3. 下面具体添加native层对plus方法的实现代码

[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. #include <jni.h>  
  2. #include <android/log.h>  
  3.   
  4. #define LOG_TAG "JNITest"  
  5. //log func  
  6. //int __android_log_print(int prio, const char *tag,  const char *fmt, ...)  
  7.   
  8. #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)  
  9.   
  10. extern "C"  
  11. JNIEXPORT int Java_com_example_jni_1</span>learn_JNI_plus");  
  12.     return x + y;  
  13. }  

使用了Android log将log打印到了logcat一样,想了解的可以自行查阅相关知识,这边没太大关系,这个也是在Android.mk中为什么需要增加LOCAL_LDLIBS原因。

extern “C”是必须要加的,因为C++和C编译后的导出函数原型不同,java只能call C类型的,所以需要将C++转化为C类型的,增加extern "C"声明。

JNIEXPORT声明他是一个导出函数。

方法名字必须以Java开头,然后包名和类名及方法名之间下划线隔开,如果不巧刚好你的包名中有下划线,那如何处理JNI中java包名含下划线情况呢?在前面加一个“1”。


这么简单程序,遇到几个错误:

第一个是说JNI Load Library失败,因为apk中可以看到so是不是被压缩进去了(更改apk后缀为rar,解压rar可以看到有个lib文件夹内有这个so),我这边看到压缩进去了,结果还是加载失败。原因在于loadLibrary的参数必须要不带前缀lib不加后缀.so才可以。

爆出undefined reference to,原因是由于我没有声明extern "C",java找不到C++声明方式的。

还有就是包名中恰好出现下划线,比如JNI_Learn,需要更改为_1才可以规避。

还有就是使用LOG,必须要在Android.mk中增加LOCAL_LDLIBS。

0 0
原创粉丝点击