1、说明
a) android NDK的环境
i. android下使用opencv需要有NDK的环境
ii. 请先阅读我的博客《NDK环境搭建》进行环境搭建,http://blog.csdn.net/jthink_/article/details/9414213
b) JNI的使用
i. 本文档需要有JNI的初步知识
ii. 请阅读我的博客《JNI的使用》,http://blog.csdn.net/jthink_/article/details/9414155
2、下载OpenCV-2.3.1-android.zip
a) 说明:svn下载方式现在无法checkout所需的资源(现在需要密码),到CSDN上下
b) 下载路径:http://download.csdn.net/detail/c_qiang0_0/3617059
3、OpenCV库在android下的调用方法
a) 使用OpenCV Java API
i. 解压OpenCV-2.3.1-android.zip,解压之后又2个文件夹,如下图:
ii. 将文件夹“OpenCV-2.3.1”拷贝到Eclipse工作空间所在的目录(也就是跟你的工作空间目录处在同一级目录上)
iii. 将刚刚拷贝的“OpenCV-2.3.1”导入到工作空间中(在eclipse Package Explorer中右击,选择import)
iv. 在Package Explorer中选择你的项目(注:是自己要开发的项目,不是OpenCV-2.3.1),右击选择properties,出现如下对话框:
v. 在左边选择Android选项,点击右下方的Add,出现如下对话框:
vi. 选择“OpenCV-2.3.1”,点击ok
vii. 如果你的项目中多出了opencv-2.3..1.jar,则说明是正确的,否则就是刚刚放置“OpenCV-2.3.1”的目录路径不正确,如下图:
viii. 进行测试
1. 修改activity_main.xml,内容如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<Button
android:id="@+id/getDataBtn"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="getData" />
<TextView
android:id="@+id/dataView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="lookHere" />
</LinearLayout>
注:添加了一个按钮和一个TextView用于显示文字
2. 在MainActivity的onCreate()函数中添加如下代码:
this.setTitle("opencv java api test");
dataView = (TextView)findViewById(R.id.dataView);
getDataBtn = (Button)findViewById(R.id.getDataBtn);
getDataBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Mat mat = new Mat();
dataView.setText(mat.size() + "");
}
});
注:点击按钮会在TextView上显示这个Mat的行数和列数,由于没有 初始化,所以这个应该是显示0x0
3. 右击项目,run as->android application,点击按钮如下图(运行成功):
b) 利用JNI编写C++ OpenCV代码,通过Android NDK创建动态库(.so)
i. 新建一个android的项目,选择android2.2版本,假设命名为“HaveImgFun”,活动名改为HaveImgFun,Package name中填写com.jthink(注:名称随便定)
ii. 如同使用OpenCV Java API那样,将“OpenCV-2.3.1”文件夹拷贝到与工作空间同一级目录中,将samples下面的includeOpenCV.mk文件拷贝到和项目HaveImgFun同一级目录中,现在的文件结构如下所示:
注:上面这个各个文件夹和文件的放置很重要,因为OpenCV-2.3.1下的 OpenCV.mk中有很多相对路径的指定,如果不是这样放置,在NDK生成动态 库时可能会报文件或文件夹无法找到的错误
iii. 选择Package Explorer中你的项目,右键选择new->folder,新建一个名为jni的文件夹,用来存放你的c/c++代码
iv. 修改activity_main.xml,内容如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<Button
android:id="@+id/btnNDK"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="使用C++ OpenCV进行处理" />
<Button
android:id="@+id/btnRestore"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="还原" />
<ImageView
android:id="@+id/ImageView01"
android:layout_width="fill_parent"
android:layout_height="fill_parent" />
</LinearLayout>
注:上述代码布局是一个线性布局,里面包含了2个按钮和一个显示图像的 ImageBView
v. 在文件夹src的com.jthink包中新建一个类用于包装使用了opencv c++代码的动态库的导出函数,类名为LibImgFun,eclipse生成LibImgFun.java,修改它的内容如下:
package com.jthink;
public class LibImgFun {
static {
System.loadLibrary("ImgFun");
}
/**
*
* @param w : the current view width
* @param h : the current view height
*/
public static native int[] ImgFun(int[] buf, int w, int h);
}
注:从上述代码中可知,动态链接库的名称是“LibImgFun.so”,注意"public static native int[] ImgFun(int[] buf, int w, int h)"中的native关键字,表明这个函数来自native code(本地调用)。static表示这是一个静态函数,这样就可以直接用类名去调用。
vi. 在jni文件夹下面新建一个“ImgFun.cpp”文件,修改内容如下:
#include <jni.h>
#include <stdio.h>
#include <stdlib.h>
#include <opencv2/opencv.hpp>
using namespace cv;
extern "C" {
JNIEXPORT jintArray JNICALL Java_com_jthink_LibImgFun_ImgFun(
JNIEnv* env, jobject obj, jintArray buf, int w, int h);
JNIEXPORT jintArray JNICALL Java_com_jthink_LibImgFun_ImgFun (
JNIEnv* env, jobject obj, jintArray buf, int w, int h) {
jint *cbuf; cbuf = env->GetIntArrayElements(buf, false);
if(cbuf == NULL) {
return 0;
}
Mat myimg(h, w, CV_8UC4, (unsigned char*)cbuf);
for(int j=0;j<myimg.rows/2;j++) {
myimg.row(j).setTo(Scalar(0,0,0,0));
}
int size=w * h;
jintArray result = env->NewIntArray(size);
env->SetIntArrayRegion(result, 0, size, cbuf);
env->ReleaseIntArrayElements(buf, cbuf, 0);
return result;
}
}
注:这个函数的功能是将传进来的图像的上半部分涂成黑色;有关cpp 文件为何这么写请参考JNI的相关知识
vii. 在jni下新建两个文件"Android.mk"文件和"Application.mk"文件,这两个文件事实上就是简单的Makefile文件
1. Android.mk的内容修改为:
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include ../includeOpenCV.mk
ifeq ("$(wildcard $(OPENCV_MK_PATH))","")
#try to load OpenCV.mk from default install location
include $(TOOLCHAIN_PREBUILT_ROOT)/user/share/OpenCV/OpenCV.mk
else
include $(OPENCV_MK_PATH)
endif
LOCAL_MODULE := ImgFun
LOCAL_SRC_FILES := ImgFun.cpp
include $(BUILD_SHARED_LIBRARY)
2. Application.mk的内容修改为:
APP_STL:=gnustl_static
APP_CPPFLAGS:=-frtti -fexceptions
APP_ABI:=armeabi armeabi-v7a
注:APP_ABI指定的是目标平台的CPU架构(android2.2必须指定为 armeabi,android2.2以上的使用armeabi-v7a,如果没有设置对,很有可能安装到android虚拟机失败,当然你同时如上面写上也是可以的)
viii. 完成上述步骤后,现在就可以使用NDK生成动态链接库了,打开安装的Cygwin Terminal,cd到项目的目录下,指令如下图所示:
ix. 输入$NDK/ndk-build命令,开始创建动态库,成功的话如图所示(如果不成功可能就是上述的某个步骤出错了,得检查一遍再继续):
x. 刷新Eclipse的HaveImgFun项目会发现多了两个新的文件夹obj和libs,如图所示:
xi. 现在进行最后一步,进行测试
1. 下载一张图片,改名字为:lena.jpg,将它放置到项目res->drawable-hdip目录中
2. 修改MainActivity.java的内容为:
package com.jthink;
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.Bitmap.Config;
import android.graphics.drawable.BitmapDrawable;
import android.os.Bundle;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import com.testopencv.haveimgfun.R;
public class MainActivity extends Activity {
private ImageView imgView = null;
private Button btnNDK = null;
private Button btnRestore = null;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
this.setTitle("使用NDK转换灰度图");
btnRestore=(Button)this.findViewById(R.id.btnRestore);
btnRestore.setOnClickListener(new ClickEvent());
btnNDK=(Button)this.findViewById(R.id.btnNDK);
btnNDK.setOnClickListener(new ClickEvent());
imgView=(ImageView)this.findViewById(R.id.ImageView01);
Bitmap img=((BitmapDrawable) getResources().getDrawable(R.drawable.lena)).getBitmap();
imgView.setImageBitmap(img);
}
class ClickEvent implements View.OnClickListener {
public void onClick(View v){
if(v == btnNDK) {
long current=System.currentTimeMillis();
Bitmap img1=((BitmapDrawable) getResources().getDrawable(R.drawable.lena)).getBitmap();
int w=img1.getWidth(),h=img1.getHeight();
int[] pix = new int[w * h];
img1.getPixels(pix, 0, w, 0, 0, w, h);
int[] resultInt=LibImgFun.ImgFun(pix, w, h);
Bitmap resultImg=Bitmap.createBitmap(w, h, Config.RGB_565);
resultImg.setPixels(resultInt, 0, w, 0, 0,w, h);
long performance=System.currentTimeMillis()-current;
imgView.setImageBitmap(resultImg);
MainActivity.this.setTitle("w:"+String.valueOf(img1.getWidth())+",h:"+String.valueOf(img1.getHeight()) + "NDK耗时 " + String.valueOf(performance)+" 毫秒");
} else if(v == btnRestore) {
Bitmap img2=((BitmapDrawable) getResources().getDrawable(R.drawable.lena)).getBitmap();
imgView.setImageBitmap(img2);
MainActivity.this.setTitle("使用OpenCV进行图像处理");
}
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_main, menu);
return true;
}
}
3. 右击项目,run as->android application,点击第一个按钮,如下图(运行成功):
文章浏览阅读3.8k次,点赞9次,收藏28次。直接上一个工作中碰到的问题,另外一个系统开启多线程调用我这边的接口,然后我这边会开启多线程批量查询第三方接口并且返回给调用方。使用的是两三年前别人遗留下来的方法,放到线上后发现确实是可以正常取到结果,但是一旦调用,CPU占用就直接100%(部署环境是win server服务器)。因此查看了下相关的老代码并使用JProfiler查看发现是在某个while循环的时候有问题。具体项目代码就不贴了,类似于下面这段代码。while(flag) {//your code;}这里的flag._main函数使用while(1)循环cpu占用99
文章浏览阅读347次。idea shift f6 快捷键无效_idea shift +f6快捷键不生效
文章浏览阅读135次。Ecmacript 中没有DOM 和 BOM核心模块Node为JavaScript提供了很多服务器级别,这些API绝大多数都被包装到了一个具名和核心模块中了,例如文件操作的 fs 核心模块 ,http服务构建的http 模块 path 路径操作模块 os 操作系统信息模块// 用来获取机器信息的var os = require('os')// 用来操作路径的var path = require('path')// 获取当前机器的 CPU 信息console.log(os.cpus._node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是
文章浏览阅读10w+次,点赞435次,收藏3.4k次。SPSS 22 下载安装过程7.6 方差分析与回归分析的SPSS实现7.6.1 SPSS软件概述1 SPSS版本与安装2 SPSS界面3 SPSS特点4 SPSS数据7.6.2 SPSS与方差分析1 单因素方差分析2 双因素方差分析7.6.3 SPSS与回归分析SPSS回归分析过程牙膏价格问题的回归分析_化工数学模型数据回归软件
文章浏览阅读7.5k次。如何利用hutool工具包实现邮件发送功能呢?1、首先引入hutool依赖<dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.7.19</version></dependency>2、编写邮件发送工具类package com.pc.c..._hutool发送邮件
文章浏览阅读867次,点赞2次,收藏2次。docker安装elasticsearch,elasticsearch-head,kibana,ik分词器安装方式基本有两种,一种是pull的方式,一种是Dockerfile的方式,由于pull的方式pull下来后还需配置许多东西且不便于复用,个人比较喜欢使用Dockerfile的方式所有docker支持的镜像基本都在https://hub.docker.com/docker的官网上能找到合..._docker安装kibana连接elasticsearch并且elasticsearch有密码
文章浏览阅读1.3w次,点赞57次,收藏92次。整理 | 郑丽媛出品 | CSDN(ID:CSDNnews)近年来,随着机器学习的兴起,有一门编程语言逐渐变得火热——Python。得益于其针对机器学习提供了大量开源框架和第三方模块,内置..._beeware
文章浏览阅读7.9k次。//// ViewController.swift// Day_10_Timer//// Created by dongqiangfei on 2018/10/15.// Copyright 2018年 飞飞. All rights reserved.//import UIKitclass ViewController: UIViewController { ..._swift timer 暂停
文章浏览阅读986次,点赞2次,收藏2次。1.硬性等待让当前线程暂停执行,应用场景:代码执行速度太快了,但是UI元素没有立马加载出来,造成两者不同步,这时候就可以让代码等待一下,再去执行找元素的动作线程休眠,强制等待 Thread.sleep(long mills)package com.example.demo;import org.junit.jupiter.api.Test;import org.openqa.selenium.By;import org.openqa.selenium.firefox.Firefox.._元素三大等待
文章浏览阅读3k次,点赞4次,收藏14次。Java软件工程师职位分析_java岗位分析
文章浏览阅读2k次。Java:Unreachable code的解决方法_java unreachable code
文章浏览阅读1w次。1、html中设置标签data-*的值 标题 11111 222222、点击获取当前标签的data-url的值$('dd').on('click', function() { var urlVal = $(this).data('ur_如何根据data-*属性获取对应的标签对象