1. 程式人生 > >android 編譯調用C代碼

android 編譯調用C代碼

line out name roi 上網 嵌入 hive set str

博客地址:www.zalezone.cn



前言

需求來源

這幾天幫別人做一個簡單的androidclient,也沒什麽功能。主要就是調用C代碼來對手機的Wifi網絡進行設置,於是也就引出了技術難點所在。怎樣去調用C程序達到我們所須要的效果。

解決方式

對於這個。我想出了兩種解決方式:

  1. 第一種方案是利用JNI來進行本地調用。關於什麽是JNI呢。JNI事實上是Java Native Interface的簡稱。也就是java本地接口,它提供了若幹API實現了java和其它語言的通信(主要是C和C++)。

  2. 另外一種方法是將要運行的C代碼編譯成可運行文件

    ,然後將這個可運行文件和程序一起打包成APK,在須要使用的時候調用這個可運行文件。

終於選擇

最後我選擇了另外一種方案,理由是另外一種方法在我已經有了可運行文件的條件下整體來說比較簡單。可操作性強。而第一種方案的話由於還要下載android的NDK,NDK是一系列工具的集合,提供了幫助開發人員高速開發C或則C++的動態庫,並能自己主動將so和java應用一起打包成apk,十分方便。

技術實現

可運行文件

首先須要得到一個可運行文件。當然想要的到可運行文件並非想象中的那麽簡單,不是在linux中直接gcc就能到的,這裏須要對C代碼進行交叉編譯獲得能夠在android機子上運行的可運行文件,詳細怎樣對C文件進行交叉編譯,這裏就不再贅述。大家能夠上網查找一下。

另外,NDK也是個不錯的工具。

資源存儲

這裏的資源存儲頁算是個小坑。尋常我們在寫java程序的時候,假設要打開一個文件的話就直接輸入路徑。比方假設所要使用的文件就在項目的文件夾下,直接輸入文件名稱就能夠調用了,可是這裏的執行環境是嵌入式設備,不是PC,這就涉及到一個問題,資源怎樣存儲了。

這裏先談一下Android中的asset目錄res/raw目錄的異同:

  • 同樣點
    • 兩者文件夾下的文件在打包後都會原封不動的保存在apk包中,不會被編譯成二進制。

  • 不同點
    • res/raw中的文件會被映射到R.java中。訪問的時候直接使用資源ID就可以。而assets文件夾下的文件不會被映射到R.java。

    • res/raw不能夠有文件夾結構,而assets文件夾下能夠再建立文件夾。

資源獲取

這裏順帶說一下res/raw下的文件資源的讀取方法,通過下面方式獲取輸入流來進行寫操作

1
InputStream is =getResources().openRawResource(R.id.filename);

接下來才是我用到的讀取assets下的方法。相同也是通過獲取輸入流的方式來進行寫操作

123
AssetManager am = null; am = getAssets(); InputStream is = am.open("filename");

註意點:據說Assert僅僅能放單個文件不超過1M的文件。可是不是真的詳細還沒考證過。假設碰到問題了應該考慮一下這個註意點。

盡管讀取是成功了,可是要用shell腳本運行的話,應該在手機的存儲上應該有這個文件,光是讀取的話在手機裏面是找不見的,所以我們須要一個存文件的操作。

這裏我寫了一個存文件的函數。當中將獲取assets中數據的方法也結合進去了。

123456789101112131415
public  void copyDataToSD(String outFileName)throws IOException{	InputStream myInputStream;	OutputStream myOutputStream = new FileOutputStream(outFileName);	myInputStream = this.getAssets().open("a.out");	byte[] buffer = new byte[1024];	int length = myInputStream.read(buffer);	while (length > 0) {		myOutputStream.write(buffer, 0, length);		length = myInputStream.read(buffer);	}	myOutputStream.flush();	myInputStream.close();	myOutputStream.close();}

然後我定義的傳入的outFileName是定義的文件路徑加文件名稱

12
private static String EXE_PATH = "data/data/com.example.g3wifi/a.out";private static File exe_file;

shell命令運行

到這裏的話就是“萬事俱備。僅僅欠東風”了,我們須要運行所得到的可運行文件。由於android是基於linux的。所以一些主要的命令還是支持的,在android中要運行shell命令的話就按例如以下格式就可以:

12345678
public  void exeC(String cmd)throws IOException{	Runtime runtime =Runtime.getRuntime();	Process process = runtime.exec(cmd);    //Process process = runtime.exec(new String[]{"su","reboot"});//能夠運行兩條命令    //這能夠得到運行shell命令後的結果    BufferedReader ie = new BufferedReader(new InputStreamReader(process.getErrorStream()));}





android 編譯調用C代碼