欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于Android的服務(wù)器端程序?qū)嵗?/h1>
 更新時(shí)間:2018年03月16日 09:41:56   作者:天兵公園  
這篇文章主要介紹了基于Android的服務(wù)器端程序?qū)嵗【幱X得挺不錯的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧

在 iOS 的 APP 中,每個(gè)程序都在自己的沙盒中運(yùn)行,一旦程序刪除了,應(yīng)用的數(shù)據(jù)也就被清除了,所以大部分程序,需要保存數(shù)據(jù)的都會使用 iCloud 備份數(shù)據(jù),但是如果是創(chuàng)作類的 APP,類似筆記之類的,如果要導(dǎo)出到電腦,就必須還要中轉(zhuǎn)一次,非常麻煩。所以也有很多 APP 就開始內(nèi)置了 FTP 服務(wù)器,一旦啟動后,電腦只需要通過 FTP 客戶端鏈接就可以訪問 APP 內(nèi)的數(shù)據(jù)了。

其實(shí)在Android中也有很多這些類似的 APP,為了方便和 PC 之間共享 APP 里的應(yīng)用數(shù)據(jù),也會有 FTP 或者WebDAV服務(wù)在 APP 里運(yùn)行。但是Android不存在和 iOS 的那種沙盒問題,雖然 Android 也有沙盒。通常大部分的手機(jī)不會取得 root 權(quán)限,敏感的應(yīng)用數(shù)據(jù)都會放在沙盒中,也就是 APP 內(nèi)部數(shù)據(jù)目錄,位于 /data/data/com.xxx.xx/ 中,可以通過 Context.getFilesDir() 獲取到該路徑,如果手機(jī)沒有 root 權(quán)限,除了 APP 本身,誰也無法訪問這里面的數(shù)據(jù)。但是 Android 可以選擇將數(shù)據(jù)存放在外部沙盒中,也就是 APP 外部數(shù)據(jù)目錄,可以通過Context.getExternalFilesDir() 獲取到該路徑,甚至還有其他歪門邪道的 APP 在外置存儲里隨便建立文件夾 ...

內(nèi)置以服務(wù)器端運(yùn)行方式和外部進(jìn)行數(shù)據(jù)交換的 APP 有很多,比如多看閱讀,Documents5 等等。

在實(shí)現(xiàn)上大部分都是啟動 Socket 監(jiān)聽一個(gè)固定端口,然后處理 HTTP 請求,但是對于大部分 APP 碼農(nóng),處理 HTTP 是一件非常麻煩的事情。要處理 Header,對 POST 和 GET 的處理,對文件上傳和普通表單的處理等等,如果不借助第三方庫,這個(gè)功能想要寫好非常困難。

在第三方實(shí)現(xiàn)中有 AndroidAsync ,雖然沒看過多看的源代碼,但是估計(jì)十有八九也是采用了這個(gè)庫。

不過它也可以作為客戶端方式,作為監(jiān)聽服務(wù)方式運(yùn)行使用方法非常簡單:

AsyncHttpServer server = new AsyncHttpServer();
server.get("/", new HttpServerRequestCallback() {
  @Override
  public void onRequest(AsyncHttpServerRequest request, AsyncHttpServerResponse response) {
    response.send("Hello!!!");
  }
});
server.listen(5000);

對于大部分做過 WEB 的同學(xué)可能在提到服務(wù)器端程序時(shí),肯定會想到 IIS 、Tomcat、Apache 這些。但是 IIS 是 Windows 平臺的,IIS 所依賴的 HTTP.SYS 是系統(tǒng)驅(qū)動級別的,移植是不可能移植的,這輩子都不可能移植的。 Tomcat 是運(yùn)行在 JVM 虛擬機(jī)上的 JavaEE 容器,Android 雖然也使用 JAVA 語言,但是其虛擬機(jī)是 ART(4.4以前是 Dalvik),Apache 是 C/C++ 開發(fā)的,移植到 Android 還是很有希望的。這個(gè)各位看官可以去網(wǎng)上找找相關(guān)的教程,Apache 如何交叉編譯到 ARM,想做個(gè)伸手黨也可以,很多已經(jīng)編譯好了的。

這里舉個(gè)栗子說說如何在 Android 上運(yùn)行 httpd for arm,可以先將編譯好的 httpd 放入 raw 文件夾中,在 MainActivity 啟動時(shí)判斷是否在指定位置中,沒有則釋放。我通常是將其放在單獨(dú)的服務(wù)中運(yùn)行,這樣就算 Activity 銷毀了,服務(wù)還會在后臺運(yùn)行,這也是服務(wù)器必備的一個(gè)特性。

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 中有一個(gè) Runtime 類,這個(gè)類主要是用來讓 Android 應(yīng)用程序可以與它所在的運(yùn)行環(huán)境進(jìn)行交互,可以直接通過調(diào)用 Runtime.getRuntime() 的靜態(tài)方法來得到這個(gè)類的實(shí)例,再調(diào)用 exec 就可以執(zhí)行命令,接下來我創(chuàng)建了一個(gè)二進(jìn)制執(zhí)行類,對其做了一個(gè)簡單的封裝。

public class BinExecuter {

  /**
   * 進(jìn)程 PID
   */
  private int pid;

  /**
   * 可執(zhí)行二進(jìn)制文件路徑
   */
  private String bin;

  /**
   * 啟動參數(shù)
   */
  private String paras;

  /**
   * 進(jìn)程實(shí)例
   */
  private Process process;

  /**
   * 獲取 PID
   * @return
   */
  public int getPid() {
    return pid;
  }

  /**
   * 構(gòu)造函數(shù)
   * @param bin 可執(zhí)行二進(jìn)制文件路徑
   * @param paras 啟動參數(shù)
   */
  public BinExecuter(String bin, String paras) {

    this.bin = bin;
    this.paras = paras;

  }

  /**
   * 啟動進(jìn)程
   */
  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();
    }
  }

  /**
   * 結(jié)束進(jìn)程
   */
  public void stop() {
    if (pid > 0) {
      try {
        Runtime.getRuntime().exec("kill -9 " + pid);
      } catch (Exception ex) {
        ex.printStackTrace();
      }
    }
  }
}

但是這還是不夠的,像 httpd 這類程序,啟動后,控制臺會有輸出。例如有客戶端請求了某個(gè) url,或者出現(xiàn)什么錯誤,都會顯示在控制臺上。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();
}

注意了,這里有個(gè)大歪鵝(while),主線程會被阻塞的,啟動另外的線程就行了,改造這個(gè)類,增加控制臺輸出的監(jiān)聽,可以讓它變稍微強(qiáng)大一點(diǎn)。

/** author:yahch**/
public interface BinExecuteCallback {
  void onConsoleResponse(String text);
}

private BinExecuteCallback binExecuteCallback;

public void setBinExecuteCallback(BinExecuteCallback binExecuteCallback) {
  this.binExecuteCallback = binExecuteCallback;
}

在前段時(shí)間我開發(fā)的一個(gè) Aria2 服務(wù)端中的對應(yīng)用法如下:

@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 把服務(wù)中截取的控制臺消息拋到 Activity 中,當(dāng)然也可以使用廣播,我覺得 EventBus 還是要好用些。

現(xiàn)在 GO 語言也百花齊放,GO 天生就是為了服務(wù)端而生,而且跨平臺能力特別強(qiáng)大,在 Github 上已經(jīng)有很多程序編譯為了 ARM 版本的,像 frp、caddy、filebrowser 這些,都可以移植在 Android 上,我們要做的,就是給他一個(gè)殼,控制它運(yùn)行和停止,以及配置些參數(shù)。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

最新評論