您好,登錄后才能下訂單哦!
這篇文章主要介紹基于Android服務器端程序的示例分析,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
在 iOS 的 APP 中,每個程序都在自己的沙盒中運行,一旦程序刪除了,應用的數據也就被清除了,所以大部分程序,需要保存數據的都會使用 iCloud 備份數據,但是如果是創作類的 APP,類似筆記之類的,如果要導出到電腦,就必須還要中轉一次,非常麻煩。所以也有很多 APP 就開始內置了 FTP 服務器,一旦啟動后,電腦只需要通過 FTP 客戶端鏈接就可以訪問 APP 內的數據了。
其實在Android中也有很多這些類似的 APP,為了方便和 PC 之間共享 APP 里的應用數據,也會有 FTP 或者WebDAV服務在 APP 里運行。但是Android不存在和 iOS 的那種沙盒問題,雖然 Android 也有沙盒。通常大部分的手機不會取得 root 權限,敏感的應用數據都會放在沙盒中,也就是 APP 內部數據目錄,位于 /data/data/com.xxx.xx/ 中,可以通過 Context.getFilesDir() 獲取到該路徑,如果手機沒有 root 權限,除了 APP 本身,誰也無法訪問這里面的數據。但是 Android 可以選擇將數據存放在外部沙盒中,也就是 APP 外部數據目錄,可以通過Context.getExternalFilesDir() 獲取到該路徑,甚至還有其他歪門邪道的 APP 在外置存儲里隨便建立文件夾 ...
內置以服務器端運行方式和外部進行數據交換的 APP 有很多,比如多看閱讀,Documents5 等等。
在實現上大部分都是啟動 Socket 監聽一個固定端口,然后處理 HTTP 請求,但是對于大部分 APP 碼農,處理 HTTP 是一件非常麻煩的事情。要處理 Header,對 POST 和 GET 的處理,對文件上傳和普通表單的處理等等,如果不借助第三方庫,這個功能想要寫好非常困難。
在第三方實現中有 AndroidAsync ,雖然沒看過多看的源代碼,但是估計十有八九也是采用了這個庫。
不過它也可以作為客戶端方式,作為監聽服務方式運行使用方法非常簡單:
AsyncHttpServer server = new AsyncHttpServer(); server.get("/", new HttpServerRequestCallback() { @Override public void onRequest(AsyncHttpServerRequest request, AsyncHttpServerResponse response) { response.send("Hello!!!"); } }); server.listen(5000);
對于大部分做過 WEB 的同學可能在提到服務器端程序時,肯定會想到 IIS 、Tomcat、Apache 這些。但是 IIS 是 Windows 平臺的,IIS 所依賴的 HTTP.SYS 是系統驅動級別的,移植是不可能移植的,這輩子都不可能移植的。 Tomcat 是運行在 JVM 虛擬機上的 JavaEE 容器,Android 雖然也使用 JAVA 語言,但是其虛擬機是 ART(4.4以前是 Dalvik),Apache 是 C/C++ 開發的,移植到 Android 還是很有希望的。這個各位看官可以去網上找找相關的教程,Apache 如何交叉編譯到 ARM,想做個伸手黨也可以,很多已經編譯好了的。
這里舉個栗子說說如何在 Android 上運行 httpd for arm,可以先將編譯好的 httpd 放入 raw 文件夾中,在 MainActivity 啟動時判斷是否在指定位置中,沒有則釋放。我通常是將其放在單獨的服務中運行,這樣就算 Activity 銷毀了,服務還會在后臺運行,這也是服務器必備的一個特性。
private File httpd; @Override public void onCreate() { super.onCreate(); httpd = new File(getFilesDir(), "httpd"); if (!httpd.exists()) { try { InputStream ins = getResources().openRawResource(R.raw.httpd); FileIOUtils.writeFileFromIS(httpd, ins); Runtime.getRuntime().exec("chmod 777 " + httpd.getAbsolutePath()); } catch (Exception e) { Log.e(TAG, "onCreate: ", e); } } }
在 Android 中有一個 Runtime 類,這個類主要是用來讓 Android 應用程序可以與它所在的運行環境進行交互,可以直接通過調用 Runtime.getRuntime() 的靜態方法來得到這個類的實例,再調用 exec 就可以執行命令,接下來我創建了一個二進制執行類,對其做了一個簡單的封裝。
public class BinExecuter { /** * 進程 PID */ private int pid; /** * 可執行二進制文件路徑 */ private String bin; /** * 啟動參數 */ private String paras; /** * 進程實例 */ private Process process; /** * 獲取 PID * @return */ public int getPid() { return pid; } /** * 構造函數 * @param bin 可執行二進制文件路徑 * @param paras 啟動參數 */ public BinExecuter(String bin, String paras) { this.bin = bin; this.paras = paras; } /** * 啟動進程 */ public void start() { try { process = Runtime.getRuntime().exec(bin + " " + paras); Field f = process.getClass().getDeclaredField("pid"); f.setAccessible(true); pid = f.getInt(process); f.setAccessible(false); } catch (Exception ex) { ex.printStackTrace(); } } /** * 結束進程 */ public void stop() { if (pid > 0) { try { Runtime.getRuntime().exec("kill -9 " + pid); } catch (Exception ex) { ex.printStackTrace(); } } } }
但是這還是不夠的,像 httpd 這類程序,啟動后,控制臺會有輸出。例如有客戶端請求了某個 url,或者出現什么錯誤,都會顯示在控制臺上。Android 上是沒有控制臺窗口的,那么如何捕捉控制臺輸出呢,簡單,重定向輸出到輸入流中即可。
InputStream outs = process.getInputStream(); InputStreamReader isrout = new InputStreamReader(outs); BufferedReader brout = new BufferedReader(isrout); String line; try { while ((line = brout.readLine()) != null) { log.d(line); } } catch (Exception ex) { ex.printStackTrace(); }
注意了,這里有個大歪鵝(while),主線程會被阻塞的,啟動另外的線程就行了,改造這個類,增加控制臺輸出的監聽,可以讓它變稍微強大一點。
/** author:yahch**/ public interface BinExecuteCallback { void onConsoleResponse(String text); } private BinExecuteCallback binExecuteCallback; public void setBinExecuteCallback(BinExecuteCallback binExecuteCallback) { this.binExecuteCallback = binExecuteCallback; }
在前段時間我開發的一個 Aria2 服務端中的對應用法如下:
@Override public int onStartCommand(Intent intent, int flags, int startId) { if (intent != null) { ariaConfig = (AriaConfig) intent.getSerializableExtra("config"); if (ariaConfig != null) { Log.d(TAG, ariaConfig.toString()); binExecuter = new BinExecuter(fileAria2c.getAbsolutePath(), ariaConfig.toString()); binExecuter.setBinExecuteCallback(new BinExecuter.BinExecuteCallback() { @Override public void onConsoleResponse(String text) { sendMessage(ARIA2_SERVICE_BIN_CONSOLE, text); } }); } } else { stopSelf(); } return super.onStartCommand(intent, flags, startId); } private void sendMessage(String name, String message) { MessageEvent genericEvent = new MessageEvent(name, message); EventBus.getDefault().post(genericEvent); }
通過 EventBus 把服務中截取的控制臺消息拋到 Activity 中,當然也可以使用廣播,我覺得 EventBus 還是要好用些。
現在 GO 語言也百花齊放,GO 天生就是為了服務端而生,而且跨平臺能力特別強大,在 Github 上已經有很多程序編譯為了 ARM 版本的,像 frp、caddy、filebrowser 這些,都可以移植在 Android 上,我們要做的,就是給他一個殼,控制它運行和停止,以及配置些參數。
以上是“基于Android服務器端程序的示例分析”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。