天天看点

Android NDK基础样例

 <b>Android NDK</b><b>基础样例</b>

         NDK(Native Development Kit),用C/C++封装一些东西?好像就这么理解好了==

<b>一、环境准备</b>

         这个好讨厌==!因为我环境都已经搭了很久了。

         已经搭建好,或者不一样环境的可以忽视。(当然看下也无妨撒^^)

<b>1</b><b>)先说下我的环境吧</b>

         1、系统环境:XP。(总感觉在这上做这些事情不协调T^T)

         2、运行环境:Eclispe+Cygwin。

<b>2</b><b>)再说下这环境下的建议咯(不知道过时了没==</b><b>)</b>

         1、Eclispe重新下个C/C++版本,不要在原Android的Eclispe上加CDT组件了。貌似会有冲突。(补充:CDT 8.0.2已不存在这个问题。我在配Linux下集成环境时一起装了没事)

         2、Cygwin根目录\home\[your name]\ .bash_profile文件内配置下你的NDK路径、工程路径等。先这两样吧==

         如果你的ndk路径是“E:\android-ndk-r6”,工作空间在“E:\studyspace”。

         export NDKROOT="/cygdrive/e/android-ndk-r6"

         export STUDYSPACE="/cygdrive/e/studyspace"

         这个ndk版本有点低了哈,我自己笔记本上也有配,是比较高的版本(好像是r9吧)。

<b>3</b><b>)怎么开始呢?</b>

         1、Eclispe+Cygwin先跑出个C工程的hello world!确定环境没什么问题。

         2、在你的Android工程下建一个jni的目录(这个网上搜搜ndk开发,很多的。简要讲讲了,关键的再截个图吧==)

         3、在你的C版Eclispe建立C++工程。注意取消Use default location,选择你工程下的jni目录。Project type选择Empty Project。Toolchains选择Other Toolchain。

<a target="_blank" href="http://blog.51cto.com/attachment/201202/163820197.png"></a>

图1.3.3 创建工程注意项

         4、右键工程属性,选中C/C++ Build,取消Use default build command,改为“bash --login -c "cd $STUDYSPACE/AndroidNDK &amp;&amp; $NDKROOT/ndk-build"”。(cygwin下配的东西,你懂的,可以偷点懒==)

         提示:ndk-build在很早版本时还没这个,要自己写什么什么的(不太清楚T^T)。总之,你应该不会很低版本的。

<a target="_blank" href="http://blog.51cto.com/attachment/201202/163820987.png"></a>

图1.3.4 Build属性修改

         5、继续选至C/C++ General的Paths and Symbols。右侧Includes的C++项目,增加include路径。继续假设你的ndk路径是“E:\android-ndk-r6”==,如果是Android 2.2工程。

         Include的路径则为“E:\android-ndk-r6\platforms\android-8\arch-arm\usr\include”。

<a target="_blank" href="http://blog.51cto.com/attachment/201202/163820647.png"></a>

图1.3.4 Include路径增加

         6、把样例工程下的Android.mk、Application.mk复制到你的工程下。写完了你的程序后,仿照着修修改改就成了。

         具体的你可以搜“Android.mk详解”什么的。至于Application.mk,你2.2以上要用到Bitmap.h呢,这个就必须加了(不加编译会有什么错误来着?不记得了==)

<b>二、基础样例</b>

         这部分才是重点啊!多么切题。直接先看下运行的东西咯^^

<a target="_blank" href="http://blog.51cto.com/attachment/201202/163820223.png"></a>

         是不是更有看的动力了?还是你拉到下边直接下载去了==。(没被鄙视简单吧T^T)

<b>1</b><b>)C</b><b>部分</b>

1.1)TestJni.h

/* 

 * TestJni.h 

 * 

 *  Created on: 2011-12-20 

 *      Author: Join 

 */ 

#ifndef TESTJNI_H_ 

#define TESTJNI_H_ 

#include &lt;jni.h&gt; 

#include &lt;stdio.h&gt; 

#include &lt;stdlib.h&gt; 

#include &lt;android/log.h&gt; 

#include &lt;android/bitmap.h&gt; 

// 测试回调Java 

#include "CallJava.h" 

#define LOG_TAG "JNI_DEBUG" 

#define  LOGI(...)  __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__) 

#define  LOGE(...)  __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__) 

// JNI调用函数必须要用C编译器编译 

// C++不加extern "C",调用会有异常 

#ifdef __cplusplus 

extern "C" { 

#endif 

 * 说明: 

 * 1、JNIEXPORT、JNICALL:jni的宏,在android的jni中不必须 

 * 2、jstring:返回值类型(对应java的jni基本类型) 

 * 3、C方法名:Java_pacakege_class_method 

 * 4、JNIEnv*、jobject:jni必要参数(分别表示jni环境、java对象) 

// 样例1:获取字符串 

JNIEXPORT jstring JNICALL Java_org_join_ndk_jni_TestJni_getStr(JNIEnv*, 

        jobject); 

// 样例2:C回调Java静态及非静态方法 

JNIEXPORT jstring JNICALL Java_org_join_ndk_jni_TestJni_callJava(JNIEnv*, 

        jobject, int, int, int, int); 

// 样例3:灰度化图像(Bitmap作为参数)(注:android-8才提供Bitmap.h) 

JNIEXPORT void JNICALL Java_org_join_ndk_jni_TestJni_convertToGray(JNIEnv*, 

        jobject, jobject); 

// 样例4:缺少JNIEXPORT、JNICALL测试 

jintArray Java_org_join_ndk_jni_TestJni_getIntArray(JNIEnv*, jobject); 

// 样例5:C方法不按规则命名尝试 

jint getInt(JNIEnv*, jobject); 

// C++调用C 

//extern "C" { 

//#include "ImageUtil.h" 

//} 

//标准头文件为何有如下结构? 

//#ifdef __cplusplus 

//#endif 

//... 

#endif /* TESTJNI_H_ */ 

1.2)TestJni.cpp

 * TestJni.cpp 

#include "TestJni.h" 

JNIEXPORT jstring JNICALL Java_org_join_ndk_jni_TestJni_getStr(JNIEnv *env, 

        jobject obj) { 

    return env-&gt;NewStringUTF("I'm from C!"); 

int min(int x, int y) { 

    return (x &lt;= y) ? x : y; 

JNIEXPORT jstring JNICALL Java_org_join_ndk_jni_TestJni_callJava(JNIEnv *env, 

        jobject obj, int add_x, int add_y, int sub_x, int sub_y) { 

    char str[128]; // 字符数组 

    int result_add = add(env, add_x, add_y); // 回调Java静态方法求和 

    LOGE("==[%d+%d=%d]==", add_x, add_y, result_add); 

    int result_sub = sub(env, sub_x, sub_y); // 回调Java非静态方法求差 

    LOGE("==[%d-%d=%d]==", sub_x, sub_y, result_sub); 

    // 将比较得的整数转成字符串 

    sprintf(str, "Hello, I'm Join! min=%d", min(result_add, result_sub)); 

    return env-&gt;NewStringUTF(str); 

typedef struct { 

    uint8_t red; 

    uint8_t green; 

    uint8_t blue; 

    uint8_t alpha; 

} argb; 

/** 

 * bitmap:ARGB_8888,32位ARGB位图 

JNIEXPORT void JNICALL Java_org_join_ndk_jni_TestJni_convertToGray(JNIEnv * env, 

        jobject obj, jobject bitmap) { 

    AndroidBitmapInfo info; 

    void* pixels; 

    int ret; 

    LOGI("convertToGray"); 

    if ((ret = AndroidBitmap_getInfo(env, bitmap, &amp;info)) &lt; 0) { 

        LOGE("AndroidBitmap_getInfo() failed ! error=%d", ret); 

        return; 

    } 

    LOGI( 

            "color image :: width is %d; height is %d; stride is %d; format is %d;flags is %d", info.width, info.height, info.stride, info.format, info.flags); 

    if (info.format != ANDROID_BITMAP_FORMAT_RGBA_8888) { 

        LOGE("Bitmap format is not RGBA_8888 !"); 

    if ((ret = AndroidBitmap_lockPixels(env, bitmap, &amp;pixels)) &lt; 0) { 

        LOGE("AndroidBitmap_lockPixels() failed ! error=%d", ret); 

    // modify pixels with image processing algorithm 

    int x, y; 

    uint8_t gray; 

    for (y = 0; y &lt; info.height; y++) { 

        argb * line = (argb *) pixels; 

        for (x = 0; x &lt; info.width; x++) { 

            gray = 0.3 * line[x].red + 0.59 * line[x].green 

                    + 0.11 * line[x].blue; 

            line[x].red = line[x].green = line[x].blue = gray; 

            // line[x].alpha = 0xff; // 完全不透明 

        } 

        pixels = (char *) pixels + info.stride; 

    LOGI("unlocking pixels"); 

    AndroidBitmap_unlockPixels(env, bitmap); 

jintArray Java_org_join_ndk_jni_TestJni_getIntArray(JNIEnv *env, jobject obj) { 

    int array[] = { 12, 34 }; // 新建一个int[] 

    jintArray result = env-&gt;NewIntArray(2); // 新建一个jintArray 

    env-&gt;SetIntArrayRegion(result, 0, 2, array); // 将int[]存入result 

    return result; 

jint getInt(JNIEnv *env, jobject obj) { 

    return 256; 

1.3)回调相关

<b>2</b><b>)Java</b><b>部分</b>

2.1)TestJni<b></b>

public class TestJni { 

    /** TAG标识 */ 

    private static final String TAG = "TestJni"; 

    /** 

     * 载入动态库 

     */ 

    static { 

        System.loadLibrary("TestJni"); 

    /** 样例1:获取字符串 */ 

    public static native String getStr(); 

    /** C回调Java方法(静态) */ 

    public static int add(int x, int y) { 

        Log.e(TAG, "==Java静态add方法=="); 

        return x + y; 

    /** C回调Java方法(非静态) */ 

    public int sub(int x, int y) { 

        Log.e(TAG, "==Java非静态sub方法=="); 

        return x - y; 

    /** 样例2:C回调Java静态及非静态方法 */ 

    public static native String callJava(int add_x, int add_y, int sub_x, 

            int sub_y); 

    /** 样例3:灰度化图像(Bitmap作为参数)(注:android-8才提供Bitmap.h) */ 

    public static native void convertToGray(Bitmap bitmap); 

    /** 样例4:缺少JNIEXPORT、JNICALL测试 */ 

    public static native int[] getIntArray(); 

    /** 样例5:C方法不按规则命名尝试 */ 

    public static native int getInt(); 

2.2)AndroidNDKActivity<b></b>

public class AndroidNDKActivity extends Activity implements 

        View.OnClickListener { 

    /** 标签 */ 

    private TextView text; 

    /** 按钮 */ 

    private Button btn1, btn2, btn3, btn4, btn5; 

    /** 图像 */ 

    private ImageView imageView; 

    @Override 

    public void onCreate(Bundle savedInstanceState) { 

        super.onCreate(savedInstanceState); 

        setContentView(R.layout.main); 

        /* 初始化话各组件 */ 

        text = (TextView) findViewById(R.id.text); 

        btn1 = (Button) findViewById(R.id.btn1); 

        btn1.setOnClickListener(this); 

        btn2 = (Button) findViewById(R.id.btn2); 

        btn2.setOnClickListener(this); 

        btn3 = (Button) findViewById(R.id.btn3); 

        btn3.setOnClickListener(this); 

        btn4 = (Button) findViewById(R.id.btn4); 

        btn4.setOnClickListener(this); 

        btn5 = (Button) findViewById(R.id.btn5); 

        btn5.setOnClickListener(this); 

        imageView = (ImageView) findViewById(R.id.imageView); 

    public void onClick(View v) { 

        switch (v.getId()) { 

        case R.id.btn1: 

            // 样例1:获取字符串 

            text.setText(TestJni.getStr()); 

            break; 

        case R.id.btn2: 

            // 样例2:C回调Java静态及非静态方法 

            text.setText(TestJni.callJava(2, 5, 8, 3)); 

        case R.id.btn3: 

            // 获得Bitmap资源(32位) 

            // BitmapFactory.Options options = new BitmapFactory.Options(); 

            // options.inPreferredConfig = Bitmap.Config.ARGB_8888; 

            // Bitmap bitmap = BitmapFactory.decodeResource(getResources(), 

            // R.drawable.ic_launcher, options); 

            Bitmap bitmap = BitmapFactory.decodeResource(getResources(), 

                    R.drawable.ic_launcher); 

            // 样例3:灰度化图像(Bitmap作为参数) 

            TestJni.convertToGray(bitmap); 

            imageView.setImageBitmap(bitmap); 

        case R.id.btn4: 

            // 样例4:缺少JNIEXPORT、JNICALL测试 

            int[] array = TestJni.getIntArray(); 

            int len = array.length; 

            StringBuffer sb = new StringBuffer(); 

            for (int i = 0; i &lt; len - 1; i++) { 

                sb.append(array[i]); 

                sb.append(","); 

            } 

            sb.append(array[len - 1]); 

            text.setText(sb.toString()); 

        case R.id.btn5: 

            // 样例5:C方法不按规则命名尝试 

            try { 

                int value = TestJni.getInt(); 

                text.setText(String.valueOf(value)); 

            } catch (UnsatisfiedLinkError e) { 

                text.setText("UnsatisfiedLinkError!"); 

                e.printStackTrace(); 

<b>三、后记</b>

         也是整理的基础样例工程,嘿嘿!

         注意:

         1)需要2.2系统才可,用了Bitmap.h。

         2)样例5(不按命名规则的那个),用jni对方法进行注册,不按要求写方法名也是可以的。

         3)灰度化是用的标准公式:Gray = R*0.299 + G*0.587 + B*0.114(考虑效率的话,有移位公式)

         4)在android-ndk-r6\platforms\android-8\arch-arm\usr\lib目录下是NDK提供的可调用库。(有OpenGL ES^^)

<a href="http://down.51cto.com/data/2359862" target="_blank">附件:http://down.51cto.com/data/2359862</a>

     本文转自winorlose2000 51CTO博客,原文链接:http://blog.51cto.com/vaero/782787,如需转载请自行联系原作者