Android - JNI静态(static)加载OpenCV

JNI静态(static)加载OpenCV

本文地址: http://blog.csdn.net/caroline_wendy

步骤:

1. 准备OpenCV-Android库

复制OpenCV的sdk至工作文件夹.

下载OpenCV的最新android-sdk: OpenCV-2.4.9-android-sdk;

把其中的sdk文件夹, 复制到工作目录; sdk文件夹里包含native, java, etc. 三个文件夹.

2. 加载OpenCV-Android库

导入(import)工作目录的java文件夹, 设置为当前项目的库.

[项目名]右键点击->Android->Add Library->选择添加的库. 即可. 不需要选择Is Library.

3. 创建JniClient类.

包含需要使用的方法(method), 可以多个.

/**
 *
 */
package com.example.hellomyjni;

/**
 * @author Administrator
 *
 */
public class JniClient {
	static public native String sayName();
	static public native int[] grayImage(int[] pixels, int w, int h);
}

使用命令行, 生成头文件.

进入项目文件夹, 生成JNI的头文件, 使用命令:

"javah -classpath bin/classes -d jni com.example.hellomyjni.JniClient"

命令解析:

javah 生成头文件;

-classpath 使用类的位置(bin/classes), 都是.class文件;

-d jni 需要生成JNI的类(com.example.hellomyjni.JniClient), 包括[package].[classname].

其余参考: http://blog.csdn.net/caroline_wendy/article/details/39032551

4. 修改JNI配置文件

需要在jni文件夹内创建Application.mk, 主要负责C++标准的使用(APP_STL), 和需要编译的库.

APP_STL := gnustl_static
APP_CPPFLAGS := -frtti -fexceptions
APP_ABI := armeabi-v7a

修改Android.mk, 不需要使用OPENCV_LIB_TYPE:=STATIC

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

#OPENCV_LIB_TYPE:=STATIC
OPENCV_INSTALL_MODULES:=on

include C:/Users/Administrator/android-workspace/opencv-sdk/native/jni/OpenCV.mk

LOCAL_MODULE    := HelloMyJni
LOCAL_SRC_FILES := HelloMyJni.cpp
LOCAL_LDLIBS +=  -llog -ldl
#LOCAL_C_INCLUDES += ./
LOCAL_SRC_FILES += ./sayname.cpp

include $(BUILD_SHARED_LIBRARY)

5. JNI函数

#include <jni.h>
#include <opencv2/opencv.hpp>

using namespace cv;

#include "com_example_hellomyjni_JniClient.h"

#include "sayname.h"

JNIEXPORT jstring JNICALL Java_com_example_hellomyjni_JniClient_sayName
  (JNIEnv *env, jclass) {
	string str = sayname();
	return env->NewStringUTF(str.c_str());
}

JNIEXPORT jintArray JNICALL Java_com_example_hellomyjni_JniClient_grayImage
  (JNIEnv *env, jclass, jintArray buf, jint w, jint h ) {

	jint *cbuf;
	cbuf = env->GetIntArrayElements(buf, false);
	if (cbuf == NULL) {
		return 0;
	}

	Mat imgData(h, w, CV_8UC4, (unsigned char*) cbuf);

	uchar* ptr = imgData.ptr(0);
	for (int i = 0; i < w * h; i++) {
		int grayScale = (int) (ptr[4 * i + 2] * 0.299 + ptr[4 * i + 1] * 0.587
				+ ptr[4 * i + 0] * 0.114);
		ptr[4 * i + 1] = grayScale;
		ptr[4 * i + 2] = grayScale;
		ptr[4 * i + 0] = grayScale;
	}

	int size = w * h;
	jintArray result = env->NewIntArray(size);
	env->SetIntArrayRegion(result, 0, size, cbuf);
	env->ReleaseIntArrayElements(buf, cbuf, 0);
	return result;
}

在项目中添加OpenCV头文件的检查, 可以避免文件报错.

6. 修改MainActivity

调用JNI的函数.

package com.example.hellomyjni;

import org.opencv.android.OpenCVLoader;

import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Bitmap.Config;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.ImageView;
import android.widget.TextView;

public class MainActivity extends Activity{

	private ImageView iv;
	private Bitmap bmp;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		TextView tv = (TextView) findViewById(R.id.text_view);

		tv.setText(JniClient.sayName()); //recall JNI

		iv = (ImageView) findViewById(R.id.image_view);
		bmp = BitmapFactory.decodeResource(getResources(), R.drawable.shoes);
		int w = bmp.getWidth();
		int h = bmp.getHeight();
		int[] pixels = new int[w * h];
		bmp.getPixels(pixels, 0, w, 0, 0, w, h);

		int[] resultInt = JniClient.grayImage(pixels, w, h); //recall JNI

		Bitmap resultImg = Bitmap.createBitmap(w, h, Config.ARGB_8888);
		resultImg.setPixels(resultInt, 0, w, 0, 0, w, h);
		iv.setImageBitmap(resultImg);

	}

	@Override
	public boolean onCreateOptionsMenu(Menu menu) {
		// Inflate the menu; this adds items to the action bar if it is present.
		getMenuInflater().inflate(R.menu.main, menu);
		return true;
	}

	@Override
	public boolean onOptionsItemSelected(MenuItem item) {
		// Handle action bar item clicks here. The action bar will
		// automatically handle clicks on the Home/Up button, so long
		// as you specify a parent activity in AndroidManifest.xml.
		int id = item.getItemId();
		if (id == R.id.action_settings) {
			return true;
		}
		return super.onOptionsItemSelected(item);
	}

	static {
		if (!OpenCVLoader.initDebug()) {
	    } else {
	        System.loadLibrary("HelloMyJni");
	    }
	}
}

7. 输出

时间: 2024-09-29 16:22:35

Android - JNI静态(static)加载OpenCV的相关文章

Android运行时ART加载类和方法的过程分析

在前一篇文章中,我们通过分析OAT文件的加载过程,认识了OAT文件的格式,其中包含了原始的DEX文件.既然ART运行时执行的都是翻译DEX字节码后得到的本地机器指令了,为什么还需要在OAT文件中包含DEX文件,并且将它加载到内存去呢?这是因为ART运行时提供了Java虚拟机接口,而要实现Java虚拟机接口不得不依赖于DEX文件.本文就通过分析ART运行时加载类及其方法的过程来理解DEX文件的作用. 老罗的新浪微博:http://weibo.com/shengyangluo,欢迎关注! 在前面An

[原]零基础学习SDL开发之在Android使用SDL2.0加载字体

在上一篇文章我们知道了如何在android使用SDL2.0来渲染显示一张png图,而且在上上一篇我们知道如何使用sdl来渲染输出bmp图,那么sdl是否可以渲染输出自己喜爱的字体库的字体呢?答案是当然可以. 我们需要移植SDL_ttf字体库来支持相应的字体的渲染输出. 一.移植SDL_ttf库: 使用如下命令,从SDL Mercurial获取SDL_image的源码: hg clone https://hg.libsdl.org/SDL_ttf/ 将SDL_ttf拷贝到在上一篇文章中的andro

Android中的动态加载机制--薛彦顺

在目前的软硬件环境下,Native App与Web App在用户体验上有着明显的优势,但在实际项目中有些会因为业务的频繁变更而频繁的升级客户端,造成较差的用户体验,而这也恰恰是Web App的优势.本文对网上Android动态加载jar的资料进行梳理和实践在这里与大家一起分享,试图改善频繁升级这一弊病. Android应用开发在一般情况下,常规的开发方式和代码架构就能满足我们的普通需求.但是有些特殊问题,常常引发我们进一步的沉思.我们从沉思中产生顿悟,从而产生新的技术形式. 如何开发一个可以自定

Xilinx SDDoc 加载opencv库

Xilinx SDSoc 加载opencv库需要下载两个文件 xfopencv 和 Revision Platform 1.xfopencv : github地址 https://github.com/Xilinx/xfopencv 2.revision platform : https://www.xilinx.com/member/forms/download/design-license-xef.html?akdm=1&filename=zcu102-rv-ss-2017-4.zip p

Android中的缩略图加载-不浪费一点多余的内存

本文链接    http://blog.csdn.net/xiaodongrush/article/details/29355651 1. Why,为什么要加载缩略图? 有的时候不需要展示原图,只需展示图片的缩略图,可以节省内存.比如:网易新闻中的图片浏览,左边展示的小狮子图片就是一个缩略图,点击这个图片,才会展示原图.      2. How,怎么做呢? http://developer.android.com/training/displaying-bitmaps/load-bitmap.h

Android中的动态加载机制

在目前的软硬件环境下,Native App与Web App在用户体验上有着明显的优势,但在实际项目中有些会因为业务的频繁变更而频繁的升级客户端,造成较差的用户体验,而这也恰恰是Web App的优势.本文对网上Android动态加载jar的资料进行梳理和实践在这里与大家一起分享,试图改善频繁升级这一弊病. Android应用开发在一般情况下,常规的开发方式和代码架构就能满足我们的普通需求.但是有些特殊问题,常常引发我们进一步的沉思.我们从沉思中产生顿悟,从而产生新的技术形式.如何开发一个可以自定义

Android之ListView异步加载网络图片(优化缓存机制)【转】

网上关于这个方面的文章也不少,基本的思路是线程+缓存来解决.下面提出一些优化: 1.采用线程池 2.内存缓存+文件缓存 3.内存缓存中网上很多是采用SoftReference来防止堆溢出,这儿严格限制只能使用最大JVM内存的1/4 4.对下载的图片进行按比例缩放,以减少内存的消耗 具体的代码里面说明.先放上内存缓存类的代码MemoryCache.java: public class MemoryCache { private static final String TAG = "MemoryCa

Android之ListView异步加载图片且仅显示可见子项中的图片

折腾了好多天,遇到 N 多让人崩溃无语的问题,不过今天终于有些收获了,这是实验的第一版,有些混乱,下一步进行改造细分,先把代码记录在这儿吧. 网上查了很多资料,发现都千篇一律,抄来抄去,很多细节和完整实例都没看到,只有自己一点点研究了,总体感觉 android 下面要显示个图片真不容易啊. 项目主要实现的功能: 异步加载图片图片内存缓存.异步磁盘文件缓存解决使用 viewHolder 后出现的图片错位问题优化列表滚动性能,仅显示可见子项中的图片无需固定图片显示高度,对高度进行缓存使列表滚动时不会

Android中使用WebView加载大图

在Android系统中,加载图片是一个充满挑战的工作,特别是超出屏幕很大的图片,要考虑适应屏幕,滚动等问题. 而采用网页的方式可以很好的屏蔽这些问题,Webview+JS可以很好的控制屏幕的适配,也不会出现内存溢出的情况,来看看怎么实现的吧 package com.msn.support.gallery; import android.annotation.TargetApi; import android.os.Bundle; import android.support.annotation