(创建于2017/11/18)
JNI(Java Native Interface)
Java调用C/C++,C/C++调用Java的一套API
1.编写native方法
public class JniUtils {
public static native String getStringFromC();
public native String getStringFromC2();
}
2.javah命令,生成.h头文件
cd 进入到src目录下,使用命令生成头文件 javah 包名+类型(如 com.renzhenming.utils.JniUtils)
3.复制.h头文件到CPP工程中
右键->添加现有项->将头文件添加到vs中的头文件目录中,不要直接复制,直接复制无效
4.复制jni.h和jni_md.h文件到CPP工程中
发现在我们生成的头文件中有对jni.h 的导入,jni.h中又有对jni_md.h的导入,所以我们直接去jdk目录下搜索道这两个头文件,引入到工程中,同样是添加现有项
注意include的时候,<>与“”的灵活使用,假设生成的头文件是这样的(有时候Javah命令生成的头文件中没有方法名,只有一些预编译的东西,不知何故)
/* DO NOT EDIT THIS FILE - it is machine generated */
#include "jni.h"
/* Header for class com_renzhenming_bsdiff_JniUtils */
#ifndef _Included_com_renzhenming_bsdiff_JniUtils
#define _Included_com_renzhenming_bsdiff_JniUtils
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_renzhenming_bsdiff_JniUtils
* Method: getStringFromC
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC
(JNIEnv *, jclass);
/*
* Class: com_renzhenming_bsdiff_JniUtils
* Method: getStringFromC2
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC2
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
5.实现.h头文件中声明的函数
假设如上边一样,我们生成了头文件,那么我们需要在自己的c文件中定义实现这两个方法
#include "com_renzhenming_bsdiff_JniUtils.h" //引入头文件
JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC
(JNIEnv *env, jclass jcls) {
return (*env)->NewStringUTF(env, "aaaaaaa"); //得到字符串
}
JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC2
(JNIEnv *env, jobject jobj) {
return (*env)->NewStringUTF(env, "bbbbbb");//得到字符串
}
6.visual studio生成dll文件
如上点击debug出现下拉框,选择配置管理器,将我们的活动解决方案设置为x64
右键项目名->属性,设置常规下的项目默认值下的配置类型选择动态库(.dll),然后
点击生成,选择生成解决方案,这时候我们的dll动态库就生成在指定目录中了
7.vs生成dll动态库的时候容易出现的问题和解决方法
1.使用了过时的函数
严重性 代码 说明 项目 文件 行 禁止显示状态
错误 C4996 'open': The POSIX name for this item is deprecated. Instead, use the ISO C and C++ conformant name: _open. See online help for details. ndk_update c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 263
解决办法:给所在的类添加宏定义
define _CRT_NONSTDC_NO_DEPRECATE
或者给整个项目添加右键项目名->属性->配置属性->c/c++->命令行,在其他选项中添加 -D _CRT_NONSTDC_NO_DEPRECATE,确定即可,这样会给整个项目设置这个配置,不再需要在每个类中分别添加了
2.使用了不安全的函数
严重性 代码 说明 项目 文件 行 禁止显示状态
错误 C4996 'open': This function or variable may be unsafe. Consider using _sopen_s instead. To disable deprecation, use _CRT_SECURE_NO_WARNINGS. See online help for details. ndk_update c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 263
解决办法:同上的方式添加宏定义,_CRT_SECURE_NO_WARNINGS
3.安全性检查相关的问题
bsdiff是外国程序员写的,可能使用的工具不是vs,而vs对代码安全性检查比较严格,所以导致这个问题
在linux系统下不会报这个错误
严重性 代码 说明 项目 文件 行 禁止显示状态
错误 C4703 使用了可能未初始化的本地指针变量“old” ndk_update c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 266
错误 C4703 使用了可能未初始化的本地指针变量“V” ndk_update c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 273
错误 C4703 使用了可能未初始化的本地指针变量“_new” ndk_update c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 295
解决办法:右键项目名->属性->配置属性->c/c++->常规->SDL检查选择否
4.重复引用的问题
这个错误的原因是因为项目中既添加了bsdiff.cpp也添加了bspatch.cpp文件,导致出现定义的重复问题
也就是二者不可同时存在,移除一个
严重性 代码 说明 项目 文件 行 禁止显示状态
错误 LNK2005 "void __cdecl err(int,char const *)" (?err@@YAXHPEBD@Z) 已经在 bsdiff.obj 中定义 ndk_update c:\Users\renzhenming\Documents\Visual Studio 2015\Projects\ndk_update\ndk_update\bspatch.obj 1
错误 LNK2005 "void __cdecl errx(int,char const *)" (?errx@@YAXHPEBD@Z) 已经在 bsdiff.obj 中定义 ndk_update c:\Users\renzhenming\Documents\Visual Studio 2015\Projects\ndk_update\ndk_update\bspatch.obj 1
错误 LNK1169 找到一个或多个多重定义的符号 ndk_update c:\Users\renzhenming\Documents\Visual Studio 2015\Projects\ndk_update\x64\Debug\ndk_update.exe 1
解决办法:针对这个项目的编译,不要让两者同时存在