编程基础知识——Java JNI开发流程(2)

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)。由于发现非常多人都没搞懂而喊它加密算法。




当中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机的硬件配置高得令人抓狂。

posted on 2017-05-23 12:19  yjbjingcha  阅读(130)  评论(0编辑  收藏  举报

导航