(建立于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檔案

37761343.png
如上點選debug出現下拉框,選擇配置管理器,将我們的活動解決方案設定為x64
37833718.png
右鍵項目名->屬性,設定正常下的項目預設值下的配置類型選擇動态庫(.dll),然後
37905203.png
點選生成,選擇生成解決方案,這時候我們的dll動态庫就生成在指定目錄中了
37963765.png
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
解決辦法:針對這個項目的編譯,不要讓兩者同時存在