android中使用jni调用本地C++库 

android平台上的本地库文件后缀 .so。类似windows上的dll文件。


要在android上使用jni。首先须要下载android ndk。


操作步骤,正常建立androidproject,然后在androidproject那里右键。属性,选择Android Tools -> Add Native Support。就能够为androidproject添加本地库支持。

加入支持后的androidproject,会添加jni文件夹,C++代码就写在这个文件夹里。


新建一个类。而且使用native修饰一个函数。这个函数就是须要本地库来实现的。本样例中的本地库函数作用是将字符串进行内置算法转换,server可依据这个转换识别client的合法性,使用jni的目的是添加代码反编译破解的复杂度。


package com.show.shownative.lib;

public class ShowNativeLib {
	public native String showConvertCode(String param);	
}


使用javah生成头文件:ShowNativeLib.h

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_show_shownative_lib_ShowNativeLib */

#ifndef _Included_com_show_shownative_lib_ShowNativeLib
#define _Included_com_show_shownative_lib_ShowNativeLib
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_show_shownative_lib_ShowNativeLib
 * Method:    showConvertCode
 * Signature: (Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_show_shownative_lib_ShowNativeLib_showConvertCode
  (JNIEnv *, jobject, jstring);

#ifdef __cplusplus
}
#endif
#endif


在android的nativeproject设置中须要 mk文件类进行编译配置。

本样例中须要用到三个MK ,系统会为你自己主动创建一个,照着路径新建两个就能够了

1、Application.mk 内容例如以下

APP_STL := stlport_static
APP_ABI := armeabi armeabi-v7a mips x86


以上两行配置说明:第一个是添加C++ STL模版支持。第二个是设置须要为哪几种CPU架构编译so文件。


2、Android.mk 内容例如以下

LOCAL_PATH := $(call my-dir)

include $(LOCAL_PATH)/ShowNativeLib.mk


以上的配置,第一行是死的。

第二行是一个多模块引入方式。假设你还有其它本地库模块,能够分模块。这里include即可了。


3、ShowNativeLib.mk 内容例如以下

include $(CLEAR_VARS)

LOCAL_MODULE    := ShowNativeLib
LOCAL_SRC_FILES := ShowMD5.cpp ShowNativeLib.cpp

include $(BUILD_SHARED_LIBRARY)


这个文件才是我们样例里面測试的这个模块的配置。

ShowNativeLib.cpp 就是我们的实现文件,ShowMD5.cpp是一个第三方MD5摘要算法开源库。(这里得吐槽一下,MD5。SHA1这样的都叫摘要算法(DIGEST)。由于发现非常多人都没搞懂而喊它加密算法。


Android中jni开发技术 安卓jni开发流程_#include


当中ShowNativeLib.cpp的代码实现例如以下:解释都在代码里。

#include <jni.h>
#include <stddef.h>
#include <stdlib.h>
#include <ShowMD5.h>
#include <ShowNativeLib.h>

#define CONVERT_KEY ("sobey_showfun_convert_key_47511441331441")

JNIEXPORT jstring JNICALL Java_com_show_shownative_lib_ShowNativeLib_showConvertCode(JNIEnv* pEnv, jobject obj, jstring param)
{
	if(!param)
		return NULL ;

	/*
	 *  JNI里面操作java对象的方式,全都是基于java反射机制来做的。
	 *  假设你全然没接触过java反射。那么你先应该去查查java反射的资料。
	 *  以下仅仅做基本解释。不做深入讨论。 
	 */
	
	//取得java.lang.String的class
	jclass  clzString 	   = pEnv->FindClass("java/lang/String");
	//取得String.getBytes的函数的Method对象
	jmethodID getMethod    = pEnv->GetMethodID(clzString,"getBytes","()[B");
	//运行这个Method。得到String.getBytes返回的字节数组。
	jobject byteObj        = pEnv->CallObjectMethod(param, getMethod);
	jbyteArray paramBytes  = (jbyteArray)byteObj ;
	//将java的byte[]转成C++的 unsigned char*。以进行后面的转换算法。
	unsigned char* pBufIn = (unsigned char*)pEnv->GetByteArrayElements(paramBytes,NULL);
	jsize paramByteLen = pEnv->GetArrayLength(paramBytes);

	//调用MD5库,进行摘要处理。
	std::string paramMd5  = GetMD5Code(pBufIn, (int)paramByteLen);
	int	  lengthMd5 = paramMd5.size();

	//将内置的转换key,进行MD5处理
	std::string paramKey  = GetMD5Code((unsigned char*)CONVERT_KEY,strlen(CONVERT_KEY));

	//进行转换算法。
	for(int idx=0; idx<lengthMd5; idx++)
	{
		paramMd5[idx] ^= paramKey[idx];
		paramMd5[idx] |= 0x31 ;
	}

	//将结果MD5二进制数据,转化成16进制表示的字符串
	std::string resultMD5  = GetMD5Code((unsigned char*)paramMd5.c_str(),paramMd5.size());

	//将C++字符串转化成java字符串。
	jstring result   =  pEnv->NewStringUTF(resultMD5.c_str());

	return result ;
}


以上就是android里面使用JNI的样例。

jni还是非常有帮助的,android的java毕竟在效率上有瓶颈(解释性语言的虚拟机,多了一个解释层),非常多须要高效率的数据处理。还是免不了要调用本地库来处理。

苹果ios和android机比起来。从技术角度来看,人家的objc语言和ios平台。就比android平台的java运行效率高,这是不争的事实。

所以相同的硬件配置,苹果机跑起来的流畅度是android机比不了的。但让实际情况是android机的硬件配置高得令人抓狂。