本文目錄一覽:
- 1、java如何調用C語言程序
- 2、java如何調用c語言源文件並進行執行?
- 3、java如何調用c++裏面的方法?
- 4、Java如何調用C語言代碼 ?
- 5、java調用c語言編寫的可執行文件
- 6、java用jna調用C語言dll接口中的回調函數怎麼寫
java如何調用C語言程序
用 Runtime 的 exec 方法的確是可行的。
假設我們已經把以下的 C 程序編繹成 adder.exe:
#include stdio.h
int main() { /* 簡單地循環打印標準輸入上的兩個整數之和 */
int a, b, lineNumber = 0;
while (scanf(“%d %d”, a, b))
printf(“Line# %d \t %d + %d == %d\n”, ++lineNumber, a, b, a + b);
return 0;
}
以下的 Java 程序可以在啟動 adder.exe 後,跟 adder.exe 的標準輸入和輸出接軌,然後持續不斷地向它發送數據和索取結果:
import java.io.*;
class C {
public static void main(String[] args) throws Exception {
final Process proc = Runtime.getRuntime().exec(“adder.exe”);
// 用另一個線程把參數送到 proc 的標準輸入上去。
new Thread() {
public void run() {
OutputStream stdin = proc.getOutputStream();
for (int i = 0; ; i++) {
try {
Thread.sleep(1); // 要休息片刻才看得到 I/O 的緩存效果。
stdin.write((i + ” ” + i + “\n”).getBytes());
} catch (Exception ex) {
ex.printStackTrace();
}
}
}
}.start();
// 主線程負責讀取並打印 proc 的標準輸出。
BufferedReader stdout = new BufferedReader(new InputStreamReader(proc.getInputStream()));
for (String line; null != (line = stdout.readLine()); )
System.out.println(line);
}
}
循環里的 Thread.sleep(1) 純粹是為了凸顯 I/O 的緩存效果。
我測試時看到大約 900 行的緩存量(用 32-bit XP 和 Java 1.6)。
java如何調用c語言源文件並進行執行?
要在java中調用c語言的庫,需要使用Java提供了JNI。
舉例說明
在c語言中定義一個 void sayHello()函數(打印Hello World);然後在Java中調用這個函數顯示Hello Word.
現在分別從Java和C語言兩部分說明:
1. Java 部分
首先定義一個HelloNative,在其中申明sayHello函數,函數要申明為Native 類型的.如下:
public class HelloNative {
public native void sayHello();
}
編譯這個類,生成class文件:
javac HelloWorld.java
利用javah生成需要的h文件
javah HelloNative
生成的 h文件大概如下:
/* DO NOT EDIT THIS FILE – it is machine generated */
#include jni.h
/* Header for class HelloNative */
#ifndef _Included_HelloNative
#define _Included_HelloNative
#ifdef __cplusplus
extern “C” {
#endif
/*
* Class: HelloNative
* Method: sayHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloNative_sayHello
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
可以看一下上面自動生成的程序,程序include了jni.h,這個頭文件在 $JAVA_HOME下的include文件夾下. 還可以發現生成的函數名是在之前的函數名前面加上了Java_HelloNative。
2. C語言部分
根據上面生成的h文件編寫相應的代碼實現,建立一個 HelloNative.cpp用來實現顯示Hello World的函數.如下:
#include stdio.h
#include “HelloNative.h”
JNIEXPORT void JNICALL Java_HelloNative_sayHello(JNIEnv *, jobject)
{
printf(“Hello World!\n”);
}
代碼編寫完成之後,我們再用gcc編譯成庫文件,命令如下;
gcc -fPIC -I/usr/lib/jvm/java-7-openjdk-i386/include -I/usr/lib/jvm/java-7-openjdk-i386/include/linux -shared -o libHelloNative.so HelloNative.cpp
這樣就會在當前目錄下生成一個libHelloNative.so的庫文件.這時需要的庫已經生成,在C語言下的工作已經完成了.
接下來需要在Java中編寫一個程序測試一下.在程序前,需要將我們的庫載入進去.載入的方法是調用Java的 System.loadLibrary(“HelloNative”);
public class TestNative
{
static {
try {
System.loadLibrary(“HelloNative”);
}
catch(UnsatisfiedLinkError e) {
System.out.println( “Cannot load hello library:\n ” + e.toString() );
}
}
public static void main(String[] args) {
HelloNative test = new HelloNative();
test.sayHello();
}
}
但是再編譯後,運行的時候,問題又出現了.
Cannot load hello library:
java.lang.UnsatisfiedLinkError: no HelloNative in java.library.path
Exception in thread “main” java.lang.UnsatisfiedLinkError: HelloNative.sayHello()V
at HelloNative.sayHello(Native Method)
at TestNative.main(TestNative.java:13)
載入庫失敗,但是庫明明就是放在當前文件夾下的,怎麼會載入失敗呢?
用System.getProperty(“java.library.path”)查看,發現java.library.path中並不u存在當前的目錄.主要有以下的幾個解決辦法:
1) 將生成的庫複製到java.library.path有的路徑中去,當然這樣不是很好
2) 設置環境變量export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH ,將當前的目錄加入到LD_LIBRARY_PATH中
3) 設置java 的選項,將當前的目錄加入到其中 .java -Djava.library.path=. $LD_LIBRARY_PATH
這樣之後程序就能夠成功的運行了.可以看見顯示的”Hello World!”了
java如何調用c++裏面的方法?
一、JAVA中所需要做的工作
在JAVA程序中,首先需要在類中聲明所調用的庫名稱,如下:
在這裡,庫的擴展名字可以不用寫出來,究竟是DLL還是SO,由系統自己判斷。 還需要對將要調用的方法做本地聲明,關鍵字為native。並且只需要聲明,而不需要具體實現。如下:
然後編譯該JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就會生成C/C++的頭文件。
例如程序testdll.java,內容為:
用javac testdll.java編譯它,會生成testdll.class。
再用javah testdll,則會在當前目錄下生成testdll.h文件,這個文件需要被C/C++程序調用來生成所需的庫文件。
二、C/C++中所需要做的工作
對於已生成的.h頭文件,C/C++所需要做的,就是把它的各個方法具體的實現。然後編譯連接成庫文件即可。再把庫文件拷貝到JAVA程序的路徑下面,就可以用JAVA調用C/C++所實現的功能了。
接上例子。我們先看一下testdll.h文件的內容:
在具體實現的時候,我們只關心兩個函數原型和這裡JNIEXPORT和JNICALL都是JNI的關鍵字,表示此函數是要被JNI調用的。而jint是以JNI為中介使JAVA的int類型與本地的int溝通的一種類型,我們可以視而不見,就當做int使用。函數的名稱是JAVA_再加上java程序的package路徑再加函數名組成的。參數中,我們也只需要關心在JAVA程序中存在的參數,至於JNIEnv*和jclass我們一般沒有必要去碰它。
好,下面我們用testdll.cpp文件具體實現這兩個函數:
編譯連接成庫文件,本例是在WINDOWS下做的,生成的是DLL文件。並且名稱要與JAVA中需要調用的一致,這裡就是goodluck.dll
把goodluck.dll拷貝到testdll.class的目錄下,java testdll運行它,就可以觀察到結果了。
Java如何調用C語言代碼 ?
需要JNI(Java Native Interface)技術即,本地調用接口。
可以將C、C++、VB等其他編程語言編製的程序代碼封裝成對用戶不可見的底層class。
而在java程序中僅聲明其方法名和參數表以及返回值即可。
大致的操作過程請百度 JNI 選擇第一個搜索結果即可找到。
java調用c語言編寫的可執行文件
C代碼寫成這種形式吧:
#includestdio.h
#includestring.h
int main(string argv[]){
char a[100];
strcpy(a,argv[0]);
}
傳入的字串字符個數不能超過100個字符,用指針可以避免這一問題,C++代碼如下:
#includeiostream
using namespace std;
typedef char *String;
int main(String argv[]){
String str;
str=argv[0];
}
Java調用代碼如下所示:
Runtime.getRuntime().exec(“a.exe testString”)
testString是你自己輸入的測試字串
java用jna調用C語言dll接口中的回調函數怎麼寫
簡單的你會寫,那就好辦,直接上代碼
public interface TestAPI extends StdCallLibrary {
interface LoginReply extends StdCallCallback {
/**
* 登錄回調
*/
void invoke(String userId, long retCode, String retMsg);
}
/**
* 註冊一個登錄回調方法.此方法是dll提供的
*/
void RegisterLoginReply(LoginReply loginReply);
}
/**
* 實現登錄回調方法
*/
private class LoginReplyCallBack implements TestAPI.LoginReply {
@Override
public void invoke(String userId, long retCode, String retMsg) {
log.info(“登錄回調方法:{},{},{}”, userId, retCode, retMsg);
}
}
// 設置登錄回調
api.RegisterLoginReply(loginReplyCallBack);
原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/200897.html