[JNI] Eclipse直接完成JAVA調用C/C++ (Eclipse上使用CDT結合MinGW) 收藏
網 上都有很多在windows平臺下JAVA調用C/C++方法的介紹說明。但是,在此過程中,絕大多數都是利用VC等其他工具來進行DLL的生成。試想一 下,在eclipse平臺開發了JAVA,突然又要打開VC的工作環境,多不爽!何況如果內存不夠的話,還得先把其中一個退出。到需要再重新打開,多費勁 啊!!現在,我們讓這一系列的工作都在eclipse上完成!
首先、安裝eclipse3.2。
這個安裝過程我就不在這裏介紹了,大家都懂。
第二、安裝MinGW。
1.用戶可以直接登陸以下地址下載MinGW http://prdownloads.sourceforge.net/mingw/MinGW-5.0.3.exe
2. 鼠標雙擊執行安裝文件,選擇“Download and Install”,下一步
3.選擇軟件安裝的版本,這裏我們選擇“Candidate”,下一步
4.選擇安裝的編譯器
1) g++ compiler
2) g77 compiler
3) Objective C Compiler
4) MinGW Make
下一步
5. 選擇安裝路徑,下一步
這裏用默認路徑C:\MinGW
6. 等待下載軟件版本及安裝。安裝會自動完成,但是你要保證你的網絡是可用的。
第三、MinGW的環境變量設置
1. 系統變量裏面的Path添加“MinGW安裝目錄\bin;”(例如;C:\MinGW\bin;)
2. 系統變量裏添加 C_INCLUDE_PATH,值爲:C:\MinGW\include
3. 系統變量裏添加 CPLUS_INCLUDE_PATH,值爲:C:\MinGW\include\c++\3.4.2;C:\MinGW\include\c++ \3.4.2\mingw32;C:\MinGW\include\c++\3.4.2\backward;C:\MinGW\include (這裏要注意你的版本號哦)
第四、做一個小技巧修改
先將MinGW安裝目錄\bin底下的 mingw32-make.exe複製一份副本,將副本更名爲make.exe (因爲eclipse默認用了make,不是mingw32-make,當然,你也可以修改eclipse的設定,自己喜歡吧)。
第五、安裝CDT插件。
1、 到Eclipse的網站上面找到CDT插件的壓縮包,然後下載至電腦硬盤上;
下載地址:(http://www.eclipse.org/cdt/downloads.php )
2、就像你安裝其他安裝插件一下的方式安裝CDT插件吧。怎麼安裝不是關鍵問題,按你喜歡,能用就可以了。
第六、簡單介紹CDT的使用吧。
1. 新建一個C++項目 點擊菜單欄上的 File / New / Standard Make C++ Project
如果想建立一個C項目則選擇Standard Make C Project;
2. 輸入New Project名字,直接按Finish完成Project添加
3. 給項目新建一個的源文件,點擊菜單欄上的 File / New / File
4. 接着出現文件添加的窗口,在File Name欄裏面寫上文件名
C++的源文件後綴名爲cpp 如hello.cpp
而C的源文件後綴名爲c 如hello.c
5.
編輯剛剛添加的hello.cpp 文件,添加一段C++的代碼,內容如下:
using namespace std;
int main()
{
cout<<"hello world!!\n";
return 1;
}
若你新建的項目爲C的話,則需添加相應的C代碼,內容如下:
int main(){
printf("Hello world!\n");
return 1;
}
添加一個編譯命令
Target Name:MAKE FILE
Builder Command:g++ hello.cpp -g -o run
這裏如果是c環境就改成gcc hello.c -g -o run
雙擊MAKE FILE,此時Eclipse在項目文件的根目錄創建一個run.exe的文件,此文件根據hello.cpp代碼編譯生成。
雙擊run.exe可以預覽效果,控制檯輸出結果:Hello world!
第七、讓我們開始進入真正的工作吧!
The Java side
1.建立Java工程JavaHello,編寫java類
// Hello.class ,這裏主要包名,因爲後面的函數命名同包名也有聯繫的。
class Hello
{
public native void sayHello();
static
{
System.loadLibrary("hello");
}
public static void main(String[] args)
{
Hello h = new Hello();
h.sayHello();
}
}
2.用命令生成頭文件
在bin目錄下建立批處理文件,名稱隨意,用bat作爲擴展名就行了。這裏命名爲run.bat
內容爲:
set path=%java_home%\bin
set class_path=%java_home%\lib;%java_home%\lib\dt.jar;%java_home%\lib\tools.jar
javah -jni test.Hello
@pause
保存,執行(可以雙擊執行,也可在eclipse的“Run As”調用bat批處理文件來執行),則在bin目錄下生成test_Hello.h 文件。內容爲:
#include <jni.h>
/* Header for class test_Hello */
#ifndef _Included_test_Hello
#define _Included_test_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: test_Hello
* Method: sayHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_test_Hello_sayHello
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
The C side-Compiling the Library
1.建立標準C工程Cpro,並生成dll文件
1) 將生成的頭文件test_Hello.h拷貝到C工程Cpro下
2) 編寫C類Hello.c,內容爲:
#include "test_Hello.h"
#include <stdio.h>
JNIEXPORT void JNICALL Java_test_Hello_sayHello
(JNIEnv *env, jobject obj)
{
printf("Hello world !\n");
return;
}
3) 在C工程Cpro下建立hello.def文件(用於定義導出的函數),內容爲:
Java_test_Hello_sayHello
4)在C工程Cpro下建立makefile文件,內容爲:
gcc -c -I"F:\dev-software\dev-soft\jdk\jdk1.5.0\include" -I"F:\dev-software\dev-soft\jdk\jdk1.5.0\include\win32" -o hello.o Hello.c
step2:
gcc -shared -o hello.dll hello.o hello.def
5) 再Make Targets視圖下,爲Cpro工程添加兩個Make Target(對應makefile文件的內容),如下圖:
6) Make Targets視圖下雙擊step1,在C工程Cpro下生成hello.o 文件。
7) Make Targets視圖下雙擊step1,在C工程Cpro下生成hello.dll 文件。
JAVA調用DLL
1. 將hello.dll拷貝到Java工程JavaHello下。
2. 運行Hello.java,則可以看到輸出結果:
Hello world!
注:如果上面不是C工程而是是C++工程
1) 將Hello.c改爲Hello.cpp
2) 將makefile內容改爲:
g++ -c -I"F:\dev-software\dev-soft\jdk\jdk1.5.0\include" -I"F:\dev-software\dev-soft\jdk\jdk1.5.0\include\win32" -o hello.o Hello.cpp
step2:
g++ -shared -o hello.dll hello.o hello.def
< type="text/javascript"> //
評論:
=================================
Java類型 本地類型 說明
boolean jboolean 無符號,8 位
byte jbyte 無符號,8 位
char jchar 無符號,16 位
short jshort 有符號,16 位
int jint 有符號,32 位
long jlong 有符號,64 位
float jfloat 32 位
double jdouble 64 位
void void N/A
爲了使用方便,特提供以下定義。
#define JNI_FALSE 0
#define JNI_TRUE 1
jsize 整數類型用於描述主要指數和大小:
typedef jint jsize;
==================================
舉例子說說吧。
對於返回字符串類型:
JNIEXPORT jstring JNICALL Java_nativeGetSomeString
(JNIEnv *env, jobject obj,jstring s){
/* 取得 Java 字符串的 C++ 版本 */
const char *str = env->GetStringUTFChars(s, 0);//str可以在c++上用了,這是參數s轉換而來的
/* 處理該字符串 */
//===================================== 在這下面實現你的業務 =============================
char *retStr = "hello, this is test!";
jstring retJstr=env->NewStringUTF(retStr);
//===================================== 在這上面實現你的業務 =============================
/* 至此完成對 str 的處理,這個指針在調用ReleaseStringUTFChar()函數之前一直有效 */
env->ReleaseStringUTFChars(s, str);
return retJstr;
}
對於返回整數型的:
JNIEXPORT jint JNICALL Java_nativeGetOneInt
(JNIEnv *env, jobject obj){
jint retJint = 0;
int temp = 1001;
retJint = (jint)temp;
return retJint;
}
對於返回布爾類型的:
JNIEXPORT jboolean JNICALL Java_nativeGetOneBoolean
(JNIEnv *env, jobject obj,jstring s){
/* 取得 Java 字符串的 C++ 版本 */
const char *str = env->GetStringUTFChars(s, 0);//str可以在c++上用了,這是參數s轉換而來的
/* 處理該字符串 */
jboolean retBoolean=0; //直接可用於C/C++ ,按照業務來修改此值,此值有待修改候返回;
//===================================== 在這下面實現你的業務 =============================
//===================================== 在這上面實現你的業務 =============================
/* 至此完成對 str 的處理,這個指針在調用ReleaseStringUTFChar()函數之前一直有效 */
env->ReleaseStringUTFChars(s, str);
return retBoolean;
}
其他的還有數組類型的,等下次具體做些總結再貼出來了 回覆 更多評論
1、如果java直接調用的庫(暫稱爲libA.so)有用到了 你另外一些非系統庫(暫稱libB.so),那麼在生成libA.so前,應該先 把libB.so生成了,然後在生成libA.so的命令中加入 -lB 參數,以告訴libA.so要連接libB.so (具體命令如下):
g++ -Wall -rdynamic -shared -o libA.so ANative.o -lB
2、爲了你的C++庫能給C調用,在定義頭文件時最好使用上extern "C" (例如):
#ifdef __cplusplus
extern "C"
{
#endif
函數聲明
#ifdef __cplusplus
}
#endif 回覆 更多評論
< type="text/javascript"> function pageLoad() { Sys.WebForms.PageRequestManager.getInstance().add_initializeRequest(handleInitializeRequest); //Sys.WebForms.PageRequestManager.getInstance().add_endRequest(handleEndRequest); } function handleInitializeRequest(sender, args) { var prm = Sys.WebForms.PageRequestManager.getInstance(); var eid = args.get_postBackElement().id; if (eid.indexOf("DeleteLink")>0) { args.get_postBackElement().innerHTML = "
正在刪除..."; } else if (eid.indexOf("btnSubmit")>0) { document.getElementById("AjaxHolder_PostComment_ltSubmitMsg").innerHTML="正在提交..."; document.getElementById("AjaxHolder_PostComment_btnSubmit").disabled = true; } else if(eid.indexOf("refreshList")>0) { document.getElementById("AjaxHolder_PostComment_refreshList").innerHTML="
正在刷新..."; } } function TempSave(ElementID) { try { CommentsPersistDiv.setAttribute("CommentContent",document.getElementById(ElementID).value); CommentsPersistDiv.save("CommentXMLStore"); } catch(ex) { } } function Restore(ElementID) { CommentsPersistDiv.load("CommentXMLStore"); document.getElementById(ElementID).value=CommentsPersistDiv.getAttribute("CommentContent"); }