Flutter基本知识整理笔记

  • 与iOS开发的异同。例:Widget和UIView,交互等
  • 生命周期,Widget
  • 常用Widget介绍
  • 路由跳转
  • 网络请求,json解析

一、与原生开发的区别

首先要了解flutter。什么是flutter?flutter和ReactNative、Weex有什么区别?和原生开发有什么区别?

什么是flutter?先引用一段官方解释:

Flutter is Google’s mobile app SDK for crafting high-quality native interfaces on iOS and Android in record time. Flutter works with existing code, is used by developers and organizations around the world, and is free and open source.

官方的说法很简单,flutter就是一个移动应用的SDK,它跨平台(iOS和Android),也支持混合开发,且开源免费。我们可以使用flutter编写高质量的具有原生体验的应用。

要彻底了解flutter,我们必须要先了解下FuchsiaSkiaDart

Fuchsia

Fuchsia是谷歌开发的一款全新的操作系统,他不同于谷歌之前开发的系统,如ChromeOS、Android都是基于Linux内核,而Fuchsia是基于全新内核Zircon,对于硬件要求更低,性能更加卓越。

对于这款系统,谷歌一直很低调,就连最初的亮相也是因为于2016年8月份在Github开源的项目而公之于众而非官宣,但是对于这款操作系统的愿景和目的,谷歌也一直很隐晦,但是从谷歌的动作不难推测,它是想开发一款能够运行在所有不同设备的系统,智能手机、平板电脑、个人电脑等,胃口不可谓不大,大众一度推测,这款全新系统大有日后逐步取代Android的趋势。
而Flutter跟Fuchsia有什么关系呢?

Flutter是Fuchsia操作系统构建UI的SDK。这种关系,不需多言,Flutter的定位一下清晰了很多。

Skia

Skia是一款用C++开发的性能彪悍的2d图像绘制引擎,于2005年被谷歌收购。因为其出色的绘制表现,被广泛应用,如ChromeOS、ChromiumOS、FirefoxOS、Android等产品,当然,Flutter的底层2D图像引擎用的就是Skia

现在我们明白了,原来Flutter的实现思路,完全不同于React Native,Weex之类通过建立JSBridge/JSCore来桥接iOS/Android的原生控件,而是做的比较彻底,==直接基于全新的2d图像引擎Skia来绘制图像==

这是一项伟大的突破,因为,真正的跨平台,如果不甩开JSBridge,就如同带着脚铐跳舞。而Flutter做到了,这也是其相比于其他方案的绝对优势之一。

Dart

Flutter的开发语言是Dart

Dart是谷歌推出的编程语言,谷歌一直想拥有一门能够完美编写前端、后端的语言,而自家的Go语言,是静态强类型、编译型,显然不适合谷歌想Productive编写前端应用的愿景。

为什么没有选择JS呢?JS作为一名编程语言界的冉冉新星,粉丝拥簇,Google而没有选择它,理由很社会:Dart语言组就在自己隔壁,Flutter需要的一些新特性,Dart能够快速落地实现,而如果选择了JS,Google如果想为语言添加新特性,就必须经过各种委员会和浏览器提供商的决议...。

而事实上,Dart的确做到了亲兄弟间的密切支持,2018年2月份发布的Dart2.0,2019年12月份最新发布的Dart2.1,就是为Flutter量身定制了许多语法特性以及改善了很多语法结构。

当然,选择Dart,还有其他更坚定的理由:

  • Dart支持即时编译JIT(Just In Time)和事前编译AOT(Ahead of Time)
    JIT为Flutter的HotReload(热加载)提供了可能。热加载是,当你在编辑器上修改了代码想看效果,不像传统的iOS/Android开发那样必须重新编译运行等待好长时间才能看结果(当然,现在Android/RN等也可以)。而Flutter的热加载,实现了在不到1秒的时间内就能看到修改后的结果,真是溜到飞起。要知道,热加载是开发者提高生产力的关键。而AOT将Dart编译成更加高效的本地代码,代码性能和用户体验更加卓越。

  • Dart是一门易于上手的语言,事实上,Dart作为一门现代化语言,集百家之长,可以随处可见Java、Swift、JS的影子,降低了跨语言开发的学习成本。

  • 富有语言表现力,如对函数式编程的支持相比于其他语言如Java要强太多。

  • 性能卓越。如在VM方面,谷歌凭借着其在Dalvik、Go的技术沉淀,Dart VM在内存回收、吞吐量都有着不错的表现。

flutter和ReactNative、Weex有什么区别?
这个问题在上面介绍flutter中已经给出答案了。最大区别就是flutter是抛开JSBridge的方式,而是建立在自己的2d图形绘制引擎上绘制UI。也就是说和原生UI组件毫无关系,不受原生API的影响。而ReactNative和Weex其实都是在通过jsbridge调用原生控件实现。

下面通过一张表来看下整体的一个对比:

比较内容 Flutter Weex ReactNative
平台实现 通过Dart虚拟机编译成机器码 1、Vue编写的Web页面编译成JSBundle;2、Native端解析DOM,生成真实的Native控件; 1、React编写JS文件,如果是UI界面,会映射到Virtual DOM;2、通过C++编写的Bridge调用原生的API,控件则是根据DOM映射到原生的View;
绘制引擎 Skia 1、2D绘制:JSCore+Skia;2、3D绘制:JSCore+OpenGL ES 1、2D绘制:JS V8+Skia;2、3D绘制:JS V8+OpenGL ES
开发语言 Dart Vue React
上手难度 一般 容易 较难
社区 丰富、谷歌团队支持 较小、阿里支持(目前托管Apache) 活跃、Facebook支持
跨平台 Android、iOS、Web Android、iOS、Web Android、iOS
热更新 支持 支持 支持
性能 自带绘制引擎、号称性能接近原生,但目前实际效果较原生还是有些距离 因为多了一层JS解析,渲染慢一些 同Weex

讲了这么多,那么和原生开发有什么区别呢?

以 iOS 为例。在 iOS 中,你在 UI 中创建的大部分视图都是UIView 的实例。而在构造布局时,这些视图也可以作为其他视图的容器。

在 Flutter 中,同UIView能够进行类比的就是Widget了。但Widget和 iOS 里的视图并不能同等对待,不过当你想要了解 Flutter 的工作原理时,你可以把它理解为“声明和构造 UI 的方法”。

然而,WidgetUIView还是有着相当一部分区别的。首先,widget 拥有着不同的生命周期:整个生命周期内它是不可变的,且只能够存活到被修改的时候。一旦 widget 实例或者它的状态发生了改变, Flutter 框架就会创建一个新的由 Widget 实例构造而成的树状结构。而在 iOS 里,修改一个视图并不会导致它重新创建实例,它作为一个可变对象,只会绘制一次,只有在发生 setNeedsDisplay() 调用之后才会发生重绘。

还有,和 UIView 不同,Flutter 的 widget 是很轻量的,一部分原因就是源于它的不可变特性。因为它并不是视图,也不直接绘制任何内容,而是作为对 UI 及其特性的一种描述,而被“注入”到视图中去。

Flutter 包含了 Material Components 库。内容都是一些遵循了 Material Design 设计规范 的组件。Material Design 是一种灵活的 支持全平台 的设计体系,其中也包括了 iOS。

但是 Flutter 的灵活性和表现力使其能够适配任何的设计语言。在 iOS 中,你可以通过 Cupertino widgets 来构造类似于 Apple iOS 设计语言 的接口。

二、flutterWidgetState的生命周期

因为在 Flutter 中,widget 都是不可变的,所以也不能够直接对其修改。所以,你必须通过修改 widget 的 state 来达到更新视图的目的。

于是就引入了 Stateful widget 和 Stateless widget 的概念。

StatelessWidget的生命周期

StatelessWidget的生命周期很简单,只有build。

StatefulWidget的生命周期

StatefulWidget的生命周期比较复杂,我们先创建一个继承StatefulWidget的子类和对应的State来探究其生命周期。

class _Body extends StatefulWidget {
  @override
  State<StatefulWidget> createState() {
    return new _BodyState();
  }
}


class _BodyState extends State {

  bool showPage = false;

  @override
  Widget build(BuildContext context) {
    return new Center(
      child: new Row(
        children: <Widget>[
          new RaisedButton(onPressed: (){
            setState(() {
              showPage = !showPage;
            });
          }, child: new Text('切换'),),
            showPage ? new Text('不展示page') : new _MyhomePage(),
        ],
      ),
    );
  }
}

class _MyhomePage extends StatefulWidget {
  // 指需要变化的那部分UI


  @override
  StatefulElement createElement() {
    __p('createElement');
    return super.createElement();
  }


  @override
  State<StatefulWidget> createState() {
    __p('createState');
    return new _CountState();
  }
}

class _CountState extends State<_MyhomePage> {

  var i = 0;

  @override
  void initState() {
    super.initState();
    __p('initState');
  }

  @override
  void didUpdateWidget(_MyhomePage oldWidget) {
    super.didUpdateWidget(oldWidget);
    __p('didUpdateWidget');
  }

  @override
  void deactivate() { // 使无效
    super.deactivate();
    __p('deactivate');
  }

  @override
  void dispose() { // 处理
    super.dispose();
    __p('dispose');
  }

  @override
  void didChangeDependencies() {
    super.didChangeDependencies();
    __p('didChangeDependencies');
  }

  @override
  Widget build(BuildContext context) {
    __p('build');
    return new Container(
      child: new FlatButton(
      onPressed: (){
        setState(() {
          __p('setState');
          i++;
        });
      },
      child: new Text('点击$i',textDirection: TextDirection.ltr,),
    ),
    width: 100,
    height: 40,
    color: Colors.red,
    margin: EdgeInsets.symmetric(horizontal: 50,vertical: 50),
    );
  }
}

启动程序后打印结果如下:

flutter: ==createElement
flutter: ==createState
flutter: ==initState
flutter: ==didChangeDependencies
flutter: ==build

点击按钮后打印结果如下:

flutter: ==setState
flutter: ==build

当点击切换将_MyHomePage从tree中移除时,打印结果如下:

flutter: ==deactivate
flutter: ==dispose

因此,总结下来StatefulWidget的生命周期就可以用如下这张图表示。


StatefulWidget的生命周期
flutter中常见的Widget

文本Text

new Text(
            'Hello Hello HelloHello你好',
            // 文字样式
            style: new TextStyle(
              inherit: false,
              color: Colors.red, // 文字颜色
              backgroundColor: Colors.blue, // 背景色
              fontSize: 20, // 字体大小
              fontWeight: FontWeight.w400, // 字体粗细
              fontStyle: FontStyle.italic, // 正常还是斜体
              letterSpacing: 2, // 字符间距
              wordSpacing: 5, // 实测为单词间的间距
              textBaseline: TextBaseline.ideographic, // 基线模式
              height:2, // ??
              decoration: TextDecoration.underline, // 下划线
              decorationColor: Colors.black, // 划线的颜色
              decorationStyle: TextDecorationStyle.wavy, // 这个style可能控制画实线,虚线,两条线,点, 波浪线等
              decorationThickness:2, // 线的厚度
            ),
          )

Text

富文本RichText

RichText(
      text: TextSpan(
          text: '登录即视为同意',
          style: TextStyle(color: Colors.black, fontSize: 16),
          children: [
            TextSpan(
                text: '《xx协议》',
                style: TextStyle(color: Colors.blue),
                recognizer: new TapGestureRecognizer()
                  ..onTap = () {
                    print('点击协议');
                  }),
            WidgetSpan(child: Icon(Icons.account_balance)),
          ]),
    );

其中RichText的属性定义为:final InlineSpan text;而继承自InlineSpan的类则有TextSpanWidgetSpan。*WidgetSpan有个child属性,支持各种widget,因此我们可以轻易的实现图文排布的功能。

富文本

图片Image

先看看Image类构造器中的必填参数final ImageProvider image;其中ImageProvider是个抽象类,它的抽象子类有以下这些:

  • AssetBundleImageProvider
  • ResizeImage
  • NetworkImage
  • FileImage
  • MemoryImage

Image就是通过ImageProvider的各种实现类来加载图片的。

  • Image.asset:用来加载本地资源图片
  • Image.file:用来加载本地(File文件)图片
  • Image.network:用来加载网络图片
  • Image.memory:用来加载Uint8List资源(byte数组)图片

Image.asset的使用如下,

Column(
      children: <Widget>[
        Image.asset('images/25.jpg'),
        Image(image: AssetImage('images/icon_ppt_step.png'),width: 40,height: 90,),
      ],
    );

并且需要在配置文件pubspec.yaml中需要声明图片的路径:

flutter:

  # The following line ensures that the Material Icons font is
  # included with your application, so that you can use the icons in
  # the material Icons class.
  uses-material-design: true
  assets:
     - images/25.jpg
     - images/icon_ppt_step.png     #表示引入单张图片
     # 或者如下写法,不用每张图片路径都写
     - images/   #表示引入整个images文件夹的图片资源

千万要注意不要多空格,一旦格式不对,否则不能加载图片。

网络图片加载:

Image.network('http://n.sinaimg.cn/sports/2_img/upload/cf0d0fdd/107/w1024h683/20181128/pKtl-hphsupx4744393.jpg')

按钮

class 描述
RaisedButton 凸起的按钮,其实就是Material Design风格的Button
FlatButton 扁平化的按钮
OutlineButton 线框按钮
IconButton 图标按钮
ButtonBar 按钮组
FloatingActionButton 浮动按钮

布局

Container 矩形布局

  • 构造函数
Container({
    Key key,
    this.alignment,
    this.padding,
    Color color,
    Decoration decoration,
    this.foregroundDecoration,
    double width,
    double height,
    BoxConstraints constraints,
    this.margin,
    this.transform,
    this.child,
  })
  • 属性解析

key:Container唯一标识符,用于查找更新。

alignment:控制child的对齐方式,如果container或者container父节点尺寸大于child的尺寸,这个属性设置会起作用,有很多种对齐方式。

padding:decoration内部的空白区域,如果有child的话,child位于padding内部。padding与margin的不同之处在于,padding是包含在content内,而margin则是外部边界,设置点击事件的话,padding区域会响应,而margin区域不会响应。

color:用来设置container背景色,如果foregroundDecoration设置的话,可能会遮盖color效果。

decoration:绘制在child后面的装饰,设置了decoration的话,就不能设置color属性,否则会报错,此时应该在decoration中进行颜色的设置。

foregroundDecoration:绘制在child前面的装饰。

width:container的宽度,设置为double.infinity可以强制在宽度上撑满,不设置,则根据child和父节点两者一起布局。

height:container的高度,设置为double.infinity可以强制在高度上撑满。

constraints:添加到child上额外的约束条件。

margin:围绕在decoration和child之外的空白区域,不属于内容区域。

transform:设置container的变换矩阵,类型为Matrix4。

child:container中的内容widget。


参考资料:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351