Android Studio上NDK编程步骤与演示

简介: Android Studio上NDK编程步骤与演示在AndroidStudio(AS)上搞NDK编程首先要下载与安装NDK,搞好了这步之后。只需要以下几步配置与操作就可以轻松开始NDK编程与运行。

Android Studio上NDK编程步骤与演示

在AndroidStudio(AS)上搞NDK编程首先要下载与安装NDK,搞好了这步之后。只需要以下几步配置与操作就可以轻松开始NDK编程与运行。

新建一个纯Android项目(不包含C++支持)

在新项目中创建一个新Java文件为BitmapProcessor.java, 定义两个本地方法,代码实现如下:

package com.gloomyfish.ndkdemo;

import android.graphics.Bitmap;

/**
 * Created by jia20003 on 2017/5/18.
 */

public class BitmapProcessor {

    static {
        System.loadLibrary("BitmapProcessor");
    }

    public native void gray(Bitmap bm);

    public native void inverse(Bitmap bm);

}

编译含有本地方法的Java文件和产生.h的C++文件

首先新建一个bat脚本文件javahrun.bat然后写入如下内容

set JAVA_HOME="C:\Program Files\Java\jdk1.8.0_92"
set path=%JAVA_HOME%\bin;%path%
set classpath=.;%classpath%;%JAVA_HOME%\lib;C:\Users\Administrator\AppData\Local\Android\sdk\platforms\android-24\android.jar
javah com.gloomyfish.ndkdemo.BitmapProcessor

首先用javac BitmapProcessor.java这句替换脚本文件中的最后一句,然后保存该文件到项目所在的目录下app\src\main\java\com\gloomyfish\ndkdemo位置之后,通过cmd命令行执行javahrun.bat即可。

成功执行的话会生成一个.class文件在相同目录下面
这里写图片描述

然后退到目录app\src\main\java\下面,同时把bat文件修改为之前的脚本,直接执行就会生成.h的头文件,在app\src\main目录下新建jni与jniLIB两个目录文件夹之后,把.h头文件copy到jni文件夹中,这个时候在AS看到的目录结构应该如下:
这里写图片描述

新建一个C++源文件命名为BitmapProcessor.cpp, 实现头文件中定义与声明的两个本地方法即可,注意这个可以在VS2015中完成。设置一下头文件即可,如果你不知道该include多少个或者哪个,直接把目录%disk_dir%Android\android-ndk-r13b\platforms\android-24\arch-arm\usr\include下的全部添加到VS2015的include配置中即可。这样写完程序,确保没有C++的语法错误,然后copy到AS中。

编译含有C++的NDK项目

首先要设置一下NDK的路径, 右键项目打开Module Settings
这里写图片描述
然后设置好NDK路径即可。

其次,在app对应的build.gradle文件添加如下一段脚本
这里写图片描述

最后在gradle.properties文件中添加上一句话
android.useDeprecatedNdk=true
如果不加的话,编译就会又错误出现。

然后【build】->【clean build】, 【rebuild】
就编译好啦!

使用so库与编写java相关代码

首先把编译生成的app\build\intermediates\ndk\debug\lib下面的文件全部copy到之前建好的jniLIB里面即可。然后完成Java部分代码编写,实现用户交互与UI响应操作。
这里我通过JNI实现了在C++层对Bitmap对象的灰度化操作与颜色取反操作然后返回结果。运行效果如下:
这里写图片描述
这里写图片描述

相关Java与C++代码实现

MainActivity.java

package com.gloomyfish.ndkdemo;

import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;

public class MainActivity extends AppCompatActivity implements View.OnClickListener{

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button grayBtn = (Button)this.findViewById(R.id.gray_btn);
        Button invertBtn = (Button)this.findViewById(R.id.invert_btn);
        grayBtn.setOnClickListener(this);
        invertBtn.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        int id = v.getId();
        switch (id) {
            case R.id.gray_btn:
                convert2Gray();
                break;
            case R.id.invert_btn:
                invertImage();
                break;
            default:
                break;
        }
    }

    private void invertImage() {
        Bitmap bm = BitmapFactory.decodeResource(this.getResources(), R.drawable.test);
        BitmapProcessor processor = new BitmapProcessor();
        processor.inverse(bm);
        ImageView iv = (ImageView)this.findViewById(R.id.imageView_001);
        iv.setImageBitmap(bm);
    }

    private void convert2Gray() {
        Bitmap bm = BitmapFactory.decodeResource(this.getResources(), R.drawable.test);
        BitmapProcessor processor = new BitmapProcessor();
        processor.gray(bm);
        ImageView iv = (ImageView)this.findViewById(R.id.imageView_001);
        iv.setImageBitmap(bm);
    }
}

BitmapProcessor.cpp

//
// Created by gloomy fish on 2017/5/18.
//
#include<android/bitmap.h>
#include<android/log.h>
#include<com_gloomyfish_ndkdemo_BitmapProcessor.h>

#ifndef eprintf
#define eprintf(...) __android_log_print(ANDROID_LOG_ERROR,"@",__VA_ARGS__)
#endif

#define RGBA_A(p) (((p) & 0xFF000000) >> 24)
#define RGBA_R(p) (((p) & 0x00FF0000) >> 16)
#define RGBA_G(p) (((p) & 0x0000FF00) >>  8)
#define RGBA_B(p)  ((p) & 0x000000FF)
#define MAKE_RGBA(r,g,b,a) (((a) << 24) | ((r) << 16) | ((g) << 8) | (b))

JNIEXPORT void JNICALL Java_com_gloomyfish_ndkdemo_BitmapProcessor_gray
  (JNIEnv *env, jobject clazz, jobject bmpObj) {
    AndroidBitmapInfo bmpInfo={0};
    if(AndroidBitmap_getInfo(env,bmpObj,&bmpInfo)<0)
    {
        eprintf("invalid bitmap\n");
        return;
    }

    void * dataFromBmp = NULL;
    int res = AndroidBitmap_lockPixels(env, bmpObj, &dataFromBmp);
    if(dataFromBmp == NULL)
    {
        eprintf("could not retrieve pixels from bitmap\n");
        return;
    }
    eprintf("Effect: %dx%d, %d\n", bmpInfo.width, bmpInfo.height, bmpInfo.format);
    int x = 0;
    int y = 0;
    int width = bmpInfo.width;
    int height = bmpInfo.height;
    for(y=0; y<height; y++) {
        for(x=0; x<width; x++) {
            int a = 0, r = 0, g = 0, b = 0;
            void *pixel = NULL;
            pixel = ((uint32_t *)dataFromBmp) + y * width + x;
            uint32_t v = *(uint32_t *)pixel;
            a = RGBA_A(v);
            r = RGBA_R(v);
            g = RGBA_G(v);
            b = RGBA_B(v);

            // Grayscale
            int gray = (r * 38 + g * 75 + b * 15) >> 7;

            // Write the pixel back
            *((uint32_t *)pixel) = MAKE_RGBA(gray, gray, gray, a);
        }
    }

    AndroidBitmap_unlockPixels(env, bmpObj);
}

JNIEXPORT void JNICALL Java_com_gloomyfish_ndkdemo_BitmapProcessor_inverse
    (JNIEnv *env, jobject clazz, jobject bmpObj) {
AndroidBitmapInfo bmpInfo={0};
    if(AndroidBitmap_getInfo(env,bmpObj,&bmpInfo)<0)
    {
        eprintf("invalid bitmap\n");
        return;
    }

    void * dataFromBmp = NULL;
    int res = AndroidBitmap_lockPixels(env, bmpObj, &dataFromBmp);
    if(dataFromBmp == NULL)
    {
        eprintf("could not retrieve pixels from bitmap\n");
        return;
    }
    eprintf("Effect: %dx%d, %d\n", bmpInfo.width, bmpInfo.height, bmpInfo.format);
    int x = 0;
    int y = 0;
    int width = bmpInfo.width;
    int height = bmpInfo.height;
    for(y=0; y<height; y++) {
        for(x=0; x<width; x++) {
            int a = 0, r = 0, g = 0, b = 0;
            void *pixel = NULL;
            pixel = ((uint32_t *)dataFromBmp) + y * width + x;
            uint32_t v = *(uint32_t *)pixel;
            a = RGBA_A(v);
            r = RGBA_R(v);
            g = RGBA_G(v);
            b = RGBA_B(v);

            // 取反操作
            r = 255 - r;
            g = 255 - g;
            b = 255 - b;

            // Write the pixel back
            *((uint32_t *)pixel) = MAKE_RGBA(r, g, b, a);
        }
    }

    AndroidBitmap_unlockPixels(env, bmpObj);
}

PS:
我之所有把图像处理相关的都整到C++这一层来,是因为Android Studio 上用Java干这个事情真TMD太慢了,唯一C++才是唯一出路,想做AR或者图像处理与视频分析必须要学会此技能!

目录
相关文章
|
27天前
|
SQL 人工智能 Dart
Android Studio的插件生态非常丰富
Android Studio的插件生态非常丰富
41 1
|
27天前
|
Ubuntu Linux Android开发
Android Studio支持多种操作系统
Android Studio支持多种操作系统
57 1
|
27天前
|
前端开发 数据处理 Android开发
Flutter前端开发中的调试技巧与工具使用方法,涵盖调试的重要性、基本技巧如打印日志与断点调试、常用调试工具如Android Studio/VS Code调试器和Flutter Inspector的介绍
本文深入探讨了Flutter前端开发中的调试技巧与工具使用方法,涵盖调试的重要性、基本技巧如打印日志与断点调试、常用调试工具如Android Studio/VS Code调试器和Flutter Inspector的介绍,以及具体操作步骤、常见问题解决、高级调试技巧、团队协作中的调试应用和未来发展趋势,旨在帮助开发者提高调试效率,提升应用质量。
45 8
|
23天前
|
API Android开发 iOS开发
深入探索Android与iOS的多线程编程差异
在移动应用开发领域,多线程编程是提高应用性能和响应性的关键。本文将对比分析Android和iOS两大平台在多线程处理上的不同实现机制,探讨它们各自的优势与局限性,并通过实例展示如何在这两个平台上进行有效的多线程编程。通过深入了解这些差异,开发者可以更好地选择适合自己项目需求的技术和策略,从而优化应用的性能和用户体验。
|
27天前
|
数据可视化 开发工具 Android开发
Android Studio
Android Studio
81 1
|
2月前
|
编译器 Android开发
配置环境变量,使CMakeLists.txt可直接使用Android NDK工具链编译项目
配置环境变量,使CMakeLists.txt可直接使用Android NDK工具链编译项目
|
2月前
|
Java Unix Linux
Android Studio中Terminal运行./gradlew clean build提示错误信息
遇到 `./gradlew clean build`命令执行出错时,首先应检查错误信息的具体内容,这通常会指向问题的根源。从权限、环境配置、依赖下载、版本兼容性到项目配置本身,逐一排查并应用相应的解决措施。记住,保持耐心,逐步解决问题,往往复杂问题都是由简单原因引起的。
352 2
|
3月前
|
Dart 开发工具 Android开发
在 Android 系统上搭建 Flutter 环境的具体步骤是什么?
在 Android 系统上搭建 Flutter 环境的具体步骤是什么?
|
3月前
|
Java Android开发 C++
🚀Android NDK开发实战!Java与C++混合编程,打造极致性能体验!📊
在Android应用开发中,追求卓越性能是不变的主题。本文介绍如何利用Android NDK(Native Development Kit)结合Java与C++进行混合编程,提升应用性能。从环境搭建到JNI接口设计,再到实战示例,全面展示NDK的优势与应用技巧,助你打造高性能应用。通过具体案例,如计算斐波那契数列,详细讲解Java与C++的协作流程,帮助开发者掌握NDK开发精髓,实现高效计算与硬件交互。
165 1
|
14天前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
40 19