中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

基于Android服務器端程序的示例分析

發布時間:2021-08-23 13:55:34 來源:億速云 閱讀:153 作者:小新 欄目:移動開發

這篇文章主要介紹基于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服務器端程序的示例分析”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

肃南| 惠安县| 铜川市| 临汾市| 岳西县| 永安市| 乌鲁木齐市| 广平县| 琼海市| 板桥市| 华安县| 乐亭县| 那曲县| 北辰区| 海晏县| 永善县| 凉山| 淮阳县| 博客| 徐汇区| 威宁| 绥江县| 衡阳市| 襄樊市| 乌拉特后旗| 宁乡县| 凤阳县| 浑源县| 馆陶县| 梓潼县| 黎川县| 东宁县| 琼中| 北流市| 浦东新区| 鸡东县| 崇信县| 清丰县| 玉田县| 东至县| 巧家县|