Flutter完整開發(fā)實戰(zhàn)詳解(三、 打包與填坑篇)

作為系列文章的第三篇,本篇將為你著重展示:Flutter開發(fā)過程的打包流程、APP包對比、細節(jié)技巧與問題處理,本篇主要描述的 Flutter 的打包、在開發(fā)過程中遇到的各類問題與細節(jié),算是對上兩篇的補全。

文章匯總地址:

Flutter 完整實戰(zhàn)實戰(zhàn)系列文章專欄

Flutter 番外的世界系列文章專欄

一、打包

首先我們先看結(jié)果,如下表所示,是 Flutter 與 React Native 、iOS 與 Android 的縱向與橫向?qū)Ρ?/strong> 。

項目 IOS Android
GSYGithubAppFlutter
flutter-ipa
flutter-apk
GSYGithubAppRN
rn-ipa
rn-apk

從上表我們可以看到:

  • Fluuter的 apk 會比 ipa 更小一些,這其中的一部分原因是 Flutter 使用的 Skia 在Android 上是自帶的。

  • 橫向?qū)Ρ?React Native ,雖然項目不完全一樣,但是大部分功能一致的情況下, Flutter 的 Apk 確實更小一些。這里又有一個細節(jié),rn 的 ipa 包體積小很多,這其實是因為 javascriptcore 在 ios上 是內(nèi)置的原因。

  • 對上述內(nèi)容有興趣的可以看看《移動端跨平臺開發(fā)的深度解析》

1、Android 打包

I'm Android

在 Android 的打包上,筆者基本沒有遇到什么問題,在android/app/build.grade文件下,配置applicationIdversionCodeversionName 和簽名信息,最后通過 flutter build app 即可完成編譯。編程成功的包在 build/app/outputs/apk/release 下。

2、iOS 打包與真機運行

在 iOS 的打包上,筆者倒是經(jīng)歷了一波曲折,這里主要講筆者遇到的問題。

首先你需要一個 apple 開發(fā)者賬號,然后創(chuàng)建證書、創(chuàng)建AppId,創(chuàng)建配置文件、最后在info.plist文件下輸入相關(guān)信息,更詳細可看官方的《發(fā)布的IOS版APP》的教程。

但由于筆者項目中使用了第三方的插件包如 shared_preferences 等,在執(zhí)行 Archive 的過程卻一直出現(xiàn)如下問題:

在 `Archive` 時提示找不到
#import <connectivity/ConnectivityPlugin.h>  ///file not found
#import <device_info/DeviceInfoPlugin.h>
#import <flutter_statusbar/FlutterStatusbarPlugin.h>
#import <flutter_webview_plugin/FlutterWebviewPlugin.h>
#import <fluttertoast/FluttertoastPlugin.h>
#import <get_version/GetVersionPlugin.h>
#import <package_info/PackageInfoPlugin.h>
#import <share/SharePlugin.h>
#import <shared_preferences/SharedPreferencesPlugin.h>
#import <sqflite/SqflitePlugin.h>
#import <url_launcher/UrlLauncherPlugin.h>

通過 Android Studio 運行到 iOS 模擬器時沒有任何問題,說明這不是第三方包問題。通過查找問題發(fā)現(xiàn),在 iOS 執(zhí)行 Archive 之前,需要執(zhí)行 flutter build release,如下圖在命令執(zhí)行之后,Pod 的執(zhí)行目錄會發(fā)現(xiàn)改變,并且生成打包需要的文件。(ps 普通運行時自動又會修改回來

文件變化

但是實際在執(zhí)行 flutter build release 后,問題依然存在,最終翻山越嶺(╯‵□′)╯︵┻━┻,終于找到兩個答案:

  • Issue#19241 下描述了類似問題,但是他們因為路徑問題導(dǎo)致,經(jīng)過嘗試并不能解決。

  • Issue#18305 真實的解決了這個問題,居然是因為 Pod 的工程沒引入:

open iOS/Runner.xcodeproj

I checked Runner/Pods is empty in Xcode sidebar.

drop Pods/Pods.xcodeproj into Runner/Pods.

"Valid architectures" to only "arm64" (I removed armv7 armv7s) 

最后終于成功打包,心累啊(//////)。同時如果希望直接在真機上調(diào)試 Flutter,可以參考 :《Flutter基礎(chǔ)—開發(fā)環(huán)境與入門》 下的 iOS 真機部分。

二、細節(jié)

這里主要講一些小細節(jié)

1、AppBar

在 Flutter 中 AppBar 算是常用 Widget ,而 AppBar 可不僅僅作為標(biāo)題欄和使用,AppBar上的 leadingbottom 同樣是有用的功能。

  • AppBar 的 bottom 默認支持 TabBar, 也就是常見的頂部 Tab 的效果,這其實是因為TabBar 實現(xiàn)了 PreferredSizeWidgetpreferredSize
    所以只要你的控件實現(xiàn)了 preferredSize,就可以放到 AppBar 的 bottom 中使用。比如下圖搜索欄,這是TabView下的頁面又實用了AppBar。
image
  • leading :通常是左側(cè)按鍵,不設(shè)置時一般是 Drawer 的圖標(biāo)或者返回按鈕。

  • flexibleSpace :位于 bottomleading 之間。

2、按鍵

Flutter 中的按鍵,如 FlatButton 默認是否有邊距和最小大小的。所以如果你想要無 padding、margin、border 、默認大小 等的按鍵效果,其中一種方式如下:

///
new RawMaterialButton(
        materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
        padding: padding ?? const EdgeInsets.all(0.0),
        constraints: const BoxConstraints(minWidth: 0.0, minHeight: 0.0),
        child: child,
        onPressed: onPressed);

如果在再上 Flex ,如下所示,一個可控的填充按鍵就出來了。

new RawMaterialButton(
        materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
        padding: padding ?? const EdgeInsets.all(0.0),
        constraints: const BoxConstraints(minWidth: 0.0, minHeight: 0.0),
        ///flex
        child: new Flex(
          mainAxisAlignment: mainAxisAlignment,
          direction: Axis.horizontal,
          children: <Widget>[],
        ),
        onPressed: onPressed);

3、StatefulWidget 賦值

這里我們以給 TextField 主動賦值為例,其實 Flutter 中,給有狀態(tài)的 Widget 傳遞狀態(tài)或者數(shù)據(jù),一般都是通過各種 controller 。如 TextField 的主動賦值,如下代碼所示:


 final TextEditingController controller = new TextEditingController();

 @override
 void didChangeDependencies() {
    super.didChangeDependencies();
    ///通過給 controller 的 value 新創(chuàng)建一個 TextEditingValue
    controller.value = new TextEditingValue(text: "給輸入框填入?yún)?shù)");
 }

 @override
  Widget build(BuildContext context) {
    return new TextField(
     ///controller
      controller: controller,
      onChanged: onChanged,
      obscureText: obscureText,
      decoration: new InputDecoration(
        hintText: hintText,
        icon: iconData == null ? null : new Icon(iconData),
      ),
    );
  }

其實 TextEditingValueValueNotifier,其中 value 的 setter 方法被重載,一旦改變就會觸發(fā) notifyListeners 方法。而 TextEditingController 中,通過調(diào)用 addListener 就監(jiān)聽了數(shù)據(jù)的改變,從而讓UI更新。

當(dāng)然,賦值有更簡單粗暴的做法是:傳遞一個對象 class A 對象,在控件內(nèi)部使用對象 A.b 的變量綁定控件,外部通過 setState({ A.b = b2}) 更新

4、GlobalKey

在Flutter中,要主動改變子控件的狀態(tài),還可以使用 GlobalKey。 比如你需要主動調(diào)用 RefreshIndicator 顯示刷新狀態(tài),如下代碼所示。


 GlobalKey<RefreshIndicatorState> refreshIndicatorKey;
  
 showForRefresh() {
    ///顯示刷新
    refreshIndicatorKey.currentState.show();
  }

  @override
  Widget build(BuildContext context) {
    refreshIndicatorKey =  new GlobalKey<RefreshIndicatorState>();
    return new RefreshIndicator(
      key: refreshIndicatorKey,
      onRefresh: onRefresh,
      child: new ListView.builder(
        ///·····
      ),
    );
  }

5、Redux 與主題

使用 Redux 來做 Flutter 的全局 State 管理最合適不過,由于Redux內(nèi)容較多,如果感興趣的可以看看 篇章二 ,這里主要通過 Redux 來實現(xiàn)實時切換主題的效果。

如下代碼,通過 StoreProvider 加載了 store ,再通過 StoreBuilder 將 store 中的 themeData 綁定到 MaterialApp 的 theme 下,之后在其他 Widget 中通過 Theme.of(context) 調(diào)你需要的顏色,最終在任意位置調(diào)用 store.dispatch 就可實時修改主題,效果如后圖所示。

class FlutterReduxApp extends StatelessWidget {
  final store = new Store<GSYState>(
    appReducer,
    initialState: new GSYState(
      themeData: new ThemeData(
        primarySwatch: GSYColors.primarySwatch,
      ),
    ),
  );

  FlutterReduxApp({Key key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    /// 通過 StoreProvider 應(yīng)用 store
    return new StoreProvider(
      store: store,
      ///通過 StoreBuilder 獲取 themeData
      child: new StoreBuilder<GSYState>(builder: (context, store) {
        return new MaterialApp(
            theme: store.state.themeData,
            routes: {
              HomePage.sName: (context) {
                return HomePage();
              },
            });
      }),
    );
  }
}
主題

6、Hotload 與 Package

Flutter 在 Debug 和 Release 下分別是 JITAOT 模式,而在 DEBUG 下,是支持 Hotload 的,而且十分絲滑。但是需要注意的是:如果開發(fā)過程中安裝了新的第三方包 ,而新的第三方包如果包含了原生代碼,需要停止后重新運行哦。

pubspec.yaml 文件下就是我們的包依賴目錄,其中 ^ 代表大于等于,一般情況下 upgradeget 都能達到下載包的作用。但是:upgrade 會在包有更新的情況下,更新 pubspec.lock 文件下包的版本

三、問題處理

  • Waiting for another flutter command to release the startup lock :如果遇到這個問題:
  1、打開flutter的安裝目錄/bin/cache/ 
  2、刪除lockfile文件 
  3、重啟AndroidStudio
  • dialog下的黃色線
    yellow-lines-under-text-widgets-in-flutter:showDialog 中,默認是沒使用 Scaffold ,這回導(dǎo)致文本有黃色溢出線提示,可以使用 Material 包一層處理。

  • TabBar + TabView + KeepAlive 的問題
    可以通過 TabBar + PageView 解決,具體可見 篇章二

自此,第三篇終于結(jié)束了!(//////)

資源推薦

完整開源項目推薦:
GSYTech
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,488評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,034評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,327評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,554評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 71,337評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,883評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,975評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,114評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,625評論 1 332
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,555評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,737評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,244評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 43,973評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,362評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,615評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,343評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 47,699評論 2 370

推薦閱讀更多精彩內(nèi)容