基于Android的服務(wù)器端程序?qū)嵗?/h1>
更新時(shí)間:2018年03月16日 09:41:56 作者:天兵公園
這篇文章主要介紹了基于Android的服務(wù)器端程序?qū)嵗?,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
在 iOS 的 APP 中,每個(gè)程序都在自己的沙盒中運(yùn)行,一旦程序刪除了,應(yīng)用的數(shù)據(jù)也就被清除了,所以大部分程序,需要保存數(shù)據(jù)的都會(huì)使用 iCloud 備份數(shù)據(jù),但是如果是創(chuàng)作類的 APP,類似筆記之類的,如果要導(dǎo)出到電腦,就必須還要中轉(zhuǎn)一次,非常麻煩。所以也有很多 APP 就開始內(nèi)置了 FTP 服務(wù)器,一旦啟動(dòng)后,電腦只需要通過 FTP 客戶端鏈接就可以訪問 APP 內(nèi)的數(shù)據(jù)了。
其實(shí)在Android中也有很多這些類似的 APP,為了方便和 PC 之間共享 APP 里的應(yīng)用數(shù)據(jù),也會(huì)有 FTP 或者WebDAV服務(wù)在 APP 里運(yùn)行。但是Android不存在和 iOS 的那種沙盒問題,雖然 Android 也有沙盒。通常大部分的手機(jī)不會(huì)取得 root 權(quán)限,敏感的應(yīng)用數(shù)據(jù)都會(huì)放在沙盒中,也就是 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 在外置存儲(chǔ)里隨便建立文件夾 ...
內(nèi)置以服務(wù)器端運(yùn)行方式和外部進(jìn)行數(shù)據(jù)交換的 APP 有很多,比如多看閱讀,Documents5 等等。
在實(shí)現(xiàn)上大部分都是啟動(dòng) Socket 監(jiān)聽一個(gè)固定端口,然后處理 HTTP 請(qǐng)求,但是對(duì)于大部分 APP 碼農(nóng),處理 HTTP 是一件非常麻煩的事情。要處理 Header,對(duì) POST 和 GET 的處理,對(duì)文件上傳和普通表單的處理等等,如果不借助第三方庫,這個(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);
對(duì)于大部分做過 WEB 的同學(xué)可能在提到服務(wù)器端程序時(shí),肯定會(huì)想到 IIS 、Tomcat、Apache 這些。但是 IIS 是 Windows 平臺(tái)的,IIS 所依賴的 HTTP.SYS 是系統(tǒng)驅(qū)動(dòng)級(jí)別的,移植是不可能移植的,這輩子都不可能移植的。 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 啟動(dòng)時(shí)判斷是否在指定位置中,沒有則釋放。我通常是將其放在單獨(dú)的服務(wù)中運(yùn)行,這樣就算 Activity 銷毀了,服務(wù)還會(huì)在后臺(tái)運(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í)行類,對(duì)其做了一個(gè)簡單的封裝。
public class BinExecuter {
/**
* 進(jìn)程 PID
*/
private int pid;
/**
* 可執(zhí)行二進(jìn)制文件路徑
*/
private String bin;
/**
* 啟動(dòng)參數(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 啟動(dòng)參數(shù)
*/
public BinExecuter(String bin, String paras) {
this.bin = bin;
this.paras = paras;
}
/**
* 啟動(dòng)進(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 這類程序,啟動(dòng)后,控制臺(tái)會(huì)有輸出。例如有客戶端請(qǐng)求了某個(gè) url,或者出現(xiàn)什么錯(cuò)誤,都會(huì)顯示在控制臺(tái)上。Android 上是沒有控制臺(tái)窗口的,那么如何捕捉控制臺(tái)輸出呢,簡單,重定向輸出到輸入流中即可。
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),主線程會(huì)被阻塞的,啟動(dòng)另外的線程就行了,改造這個(gè)類,增加控制臺(tái)輸出的監(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ù)端中的對(duì)應(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ù)中截取的控制臺(tái)消息拋到 Activity 中,當(dāng)然也可以使用廣播,我覺得 EventBus 還是要好用些。
現(xiàn)在 GO 語言也百花齊放,GO 天生就是為了服務(wù)端而生,而且跨平臺(tái)能力特別強(qiáng)大,在 Github 上已經(jīng)有很多程序編譯為了 ARM 版本的,像 frp、caddy、filebrowser 這些,都可以移植在 Android 上,我們要做的,就是給他一個(gè)殼,控制它運(yùn)行和停止,以及配置些參數(shù)。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
-
Android 桌面快捷方式實(shí)現(xiàn)實(shí)例詳解
這篇文章主要為大家介紹了Android 桌面快捷方式實(shí)現(xiàn)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪 2022-11-11
-
Android運(yùn)行時(shí)權(quán)限終極方案(PermissionX)
這篇文章主要介紹了Android運(yùn)行時(shí)權(quán)限終極方案(PermissionX),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧 2020-05-05
-
Android Rsa數(shù)據(jù)加解密的介紹與使用示例
RSA是第一個(gè)既能用于數(shù)據(jù)加密也能用于數(shù)字簽名的算法。它易于理解和操作,也很流行。想起自己曾經(jīng)使用過的Rsa非對(duì)稱加密算法,閑下來總結(jié)一下。方便自己和大家以后使用的時(shí)候參考借鑒。下面來一起看看吧。 2016-09-09
-
Android自定義實(shí)現(xiàn)側(cè)滑菜單效果
這篇文章主要為大家詳細(xì)介紹了Android自定義實(shí)現(xiàn)側(cè)滑菜單效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下 2019-01-01
-
Android編程實(shí)現(xiàn)畫板功能的方法總結(jié)【附源碼下載】
這篇文章主要介紹了Android編程實(shí)現(xiàn)畫板功能的方法,結(jié)合實(shí)例形式總結(jié)分析了Android基于自定義View與Canvas類實(shí)現(xiàn)畫板功能的具體操作步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下 2018-02-02
-
Android中TabLayout+ViewPager 簡單實(shí)現(xiàn)app底部Tab導(dǎo)航欄
TabLayout 是Android com.android.support:design庫的一個(gè)控件。本文主要給大家介紹TabLayout+ViewPager 簡單實(shí)現(xiàn)app底部Tab布局,需要的的朋友參考下 2017-02-02
-
Flutter移動(dòng)端進(jìn)行多渠道打包發(fā)布的全過程
在使用flutter開發(fā)的過程中,需要根據(jù)不同的環(huán)境,不同的包名來打包,下面這篇文章主要給大家介紹了關(guān)于Flutter移動(dòng)端進(jìn)行多渠道打包發(fā)布的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下 2022-06-06
最新評(píng)論
在 iOS 的 APP 中,每個(gè)程序都在自己的沙盒中運(yùn)行,一旦程序刪除了,應(yīng)用的數(shù)據(jù)也就被清除了,所以大部分程序,需要保存數(shù)據(jù)的都會(huì)使用 iCloud 備份數(shù)據(jù),但是如果是創(chuàng)作類的 APP,類似筆記之類的,如果要導(dǎo)出到電腦,就必須還要中轉(zhuǎn)一次,非常麻煩。所以也有很多 APP 就開始內(nèi)置了 FTP 服務(wù)器,一旦啟動(dòng)后,電腦只需要通過 FTP 客戶端鏈接就可以訪問 APP 內(nèi)的數(shù)據(jù)了。
其實(shí)在Android中也有很多這些類似的 APP,為了方便和 PC 之間共享 APP 里的應(yīng)用數(shù)據(jù),也會(huì)有 FTP 或者WebDAV服務(wù)在 APP 里運(yùn)行。但是Android不存在和 iOS 的那種沙盒問題,雖然 Android 也有沙盒。通常大部分的手機(jī)不會(huì)取得 root 權(quán)限,敏感的應(yīng)用數(shù)據(jù)都會(huì)放在沙盒中,也就是 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 在外置存儲(chǔ)里隨便建立文件夾 ...
內(nèi)置以服務(wù)器端運(yùn)行方式和外部進(jìn)行數(shù)據(jù)交換的 APP 有很多,比如多看閱讀,Documents5 等等。
在實(shí)現(xiàn)上大部分都是啟動(dòng) Socket 監(jiān)聽一個(gè)固定端口,然后處理 HTTP 請(qǐng)求,但是對(duì)于大部分 APP 碼農(nóng),處理 HTTP 是一件非常麻煩的事情。要處理 Header,對(duì) POST 和 GET 的處理,對(duì)文件上傳和普通表單的處理等等,如果不借助第三方庫,這個(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);
對(duì)于大部分做過 WEB 的同學(xué)可能在提到服務(wù)器端程序時(shí),肯定會(huì)想到 IIS 、Tomcat、Apache 這些。但是 IIS 是 Windows 平臺(tái)的,IIS 所依賴的 HTTP.SYS 是系統(tǒng)驅(qū)動(dòng)級(jí)別的,移植是不可能移植的,這輩子都不可能移植的。 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 啟動(dòng)時(shí)判斷是否在指定位置中,沒有則釋放。我通常是將其放在單獨(dú)的服務(wù)中運(yùn)行,這樣就算 Activity 銷毀了,服務(wù)還會(huì)在后臺(tái)運(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í)行類,對(duì)其做了一個(gè)簡單的封裝。
public class BinExecuter {
/**
* 進(jìn)程 PID
*/
private int pid;
/**
* 可執(zhí)行二進(jìn)制文件路徑
*/
private String bin;
/**
* 啟動(dòng)參數(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 啟動(dòng)參數(shù)
*/
public BinExecuter(String bin, String paras) {
this.bin = bin;
this.paras = paras;
}
/**
* 啟動(dòng)進(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 這類程序,啟動(dòng)后,控制臺(tái)會(huì)有輸出。例如有客戶端請(qǐng)求了某個(gè) url,或者出現(xiàn)什么錯(cuò)誤,都會(huì)顯示在控制臺(tái)上。Android 上是沒有控制臺(tái)窗口的,那么如何捕捉控制臺(tái)輸出呢,簡單,重定向輸出到輸入流中即可。
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),主線程會(huì)被阻塞的,啟動(dòng)另外的線程就行了,改造這個(gè)類,增加控制臺(tái)輸出的監(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ù)端中的對(duì)應(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ù)中截取的控制臺(tái)消息拋到 Activity 中,當(dāng)然也可以使用廣播,我覺得 EventBus 還是要好用些。
現(xiàn)在 GO 語言也百花齊放,GO 天生就是為了服務(wù)端而生,而且跨平臺(tái)能力特別強(qiáng)大,在 Github 上已經(jīng)有很多程序編譯為了 ARM 版本的,像 frp、caddy、filebrowser 這些,都可以移植在 Android 上,我們要做的,就是給他一個(gè)殼,控制它運(yùn)行和停止,以及配置些參數(shù)。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android 桌面快捷方式實(shí)現(xiàn)實(shí)例詳解
這篇文章主要為大家介紹了Android 桌面快捷方式實(shí)現(xiàn)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
Android運(yùn)行時(shí)權(quán)限終極方案(PermissionX)
這篇文章主要介紹了Android運(yùn)行時(shí)權(quán)限終極方案(PermissionX),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
Android Rsa數(shù)據(jù)加解密的介紹與使用示例
RSA是第一個(gè)既能用于數(shù)據(jù)加密也能用于數(shù)字簽名的算法。它易于理解和操作,也很流行。想起自己曾經(jīng)使用過的Rsa非對(duì)稱加密算法,閑下來總結(jié)一下。方便自己和大家以后使用的時(shí)候參考借鑒。下面來一起看看吧。2016-09-09
Android自定義實(shí)現(xiàn)側(cè)滑菜單效果
這篇文章主要為大家詳細(xì)介紹了Android自定義實(shí)現(xiàn)側(cè)滑菜單效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01
Android編程實(shí)現(xiàn)畫板功能的方法總結(jié)【附源碼下載】
這篇文章主要介紹了Android編程實(shí)現(xiàn)畫板功能的方法,結(jié)合實(shí)例形式總結(jié)分析了Android基于自定義View與Canvas類實(shí)現(xiàn)畫板功能的具體操作步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下2018-02-02
Android中TabLayout+ViewPager 簡單實(shí)現(xiàn)app底部Tab導(dǎo)航欄
TabLayout 是Android com.android.support:design庫的一個(gè)控件。本文主要給大家介紹TabLayout+ViewPager 簡單實(shí)現(xiàn)app底部Tab布局,需要的的朋友參考下2017-02-02
Flutter移動(dòng)端進(jìn)行多渠道打包發(fā)布的全過程
在使用flutter開發(fā)的過程中,需要根據(jù)不同的環(huán)境,不同的包名來打包,下面這篇文章主要給大家介紹了關(guān)于Flutter移動(dòng)端進(jìn)行多渠道打包發(fā)布的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06

