Flutter 学习笔记

Dart

官方文档:https://dart.cn/guides/language/language-tour

变量

const 和 final 的区别

const值在编译时确定,final值在运行时确定。

方法

函数简写

如果函数体只有一个表达式,则函数可以简写为:

bool isNoble(int atomicNumber) => _nobleGases[atomicNumber] != null;

参数

Dart 函数的参数有两种形式:必要参数可选参数。必要参数位于所有参数的前面,可选参数则位于必要参数的后面。

如果一个函数声明里面有多个可选参数,那么调用方如何确定某个可选参数值是准确传给了目标参数呢?

Dart 里面将可选参数分为了两种:可选命名参数可选位置参数

可选命名参数

使用{}将方法的某些参数划为可选命名参数。

可选命名参数通过在调用时明确给出目标参数的名字,来准确定位参数传递。

void getDetail(String name, {String branch = "master"}) {

}

void test() {
  // 这里明确指定了 "release" 是传给 branch 参数的
  getDetail("name", branch: "release");
}

如果习惯了 Swift 或者 Objc 里面调用方法时都会带上所有参数的名字,比如:

func test(name: String, branch: String, msg: String = "empty") {

}

test(name: "name", branch: "master")

那么在可选命名参数前面加上required就可以了。被特别标记为required的参数不再是可选参数,而是调用时必须给出该参数的名字:

// 这里的 branch 不是可选参数,而是必要参数,且在调用时必须把参数名写出来
void getInfo(String name, {required String branch, String msg = "empty"}) {
  print("branch is " + branch);
  print("msg is " + msg);
}

void main(List<String> args) {
  getInfo("name", branch: "master")
}
可选位置参数

使用[]将方法的某些参数划为可选位置参数。

可选位置参数基于可选参数们的先后顺序,来确定参数值是传递给哪一个可选参数的。

void getInfo(String name, [String branch = "master", String msg = "empty"]) {
  print("branch is " + branch);
  print("msg is " + msg);
}

void main(List<String> args) {
  // 这里的 "release" 是传递给 branch 参数的。
  // 因为 branch 参数的位置先于 msg
  getInfo("name", "release");
}
// 输出
// branch is release
// msg is empty

Flutter

Widget

Flutter 中一切用于显示的都是 Widget。Widget 又分为无状态 StatelessWidget有状态 StatefulWidget两种。

Widget 之间通过childchildren嵌套。

StatelessWidget

StatelessWidget 在创建之后不会更改。

StatefulWidget

StatefulWidget 通过一个State<T>来驱动 Widget 重新构建刷新。

State 的生命周期如下图:

life_cycle
图源: Flutter Apprentice Learn to Build Cross-Platform Apps, 2nd Edition (Mike Katz Kevin D. Moore Vincent Ngo etc.)

其中:

  1. initState() 只会调用一次,类似于 iOS 中的 viewDidLoad().
  2. didChangeDependencies() 可以调用多次。在 initState() 后会立即调用一次;之后当依赖的 InheritedWidget rebuild 时也会再调用一次。
  3. build() 是(重新)绘制 Widget 时会调用。
  4. didUpdateWidget() 在组件状态发生变化时调用。一般情况下,父 Widget 调用 setState() 后,子 Widget 就会调用 didUpdateWidget() 方法。
  5. deactivate() 是当 State 要被移除出树中时调用。但调用这个方法不代表这个 State 对象要被销毁了,因为被移除出树的 State 是能再加回来的。
  6. dispose() 是当 State 要被销毁时调用的。

InheritedWidget 数据共享

Read more

2025 年度总结

2025 年度总结

今天是 2026 年 1 月 1 号,又是新的一年。这个元旦没有安排出行任务,就在家里休息休息,或者出门溜达溜达。昨天休了一天全薪病假,做了体检,写了年终绩效总结,晚上干了一顿烤肉,没有时间写个人的年度总结。今天起早写写总结。 以下「今年」指 2025 年。 职业发展 算起来,我已经毕业工作四年多了。职业发展整体上还算稳定,没有碰到过糟心事,遇到的领导们也都对我关怀有加。今年又晋升一次,薪资迈上新的台阶。越往上升,越觉得离职业生涯的终点越近,逼迫自己赶紧找个靠谱稳定的副业,到 35 岁没人要的时候能养活自己。 最近两年 AI 大模型的崛起,提高了许多行业的可替代性。码农虽然不是首当其冲的,但危机感已经弥漫在各个论坛博客公共平台上面。没有人能准确预测到未来发展,但做好两手准备是很有必要的。码农不能再只低着头守着自己的键盘和屏幕,也要往外看,接触社会上的各种信息,打破信息壁垒。掌握的信息越多,出路就越多。

By Gray
联通 FTTR 宽带从路由器设置自动重启和穿墙功率

联通 FTTR 宽带从路由器设置自动重启和穿墙功率

几个月前把家里宽带换成了联通的千兆 FTTR 宽带,包含一主一从两个点位。配套光猫设备是华为的星光 F50 尊享版。 主点位放置在客厅茶几上,方便连接电视。从点位放在卧室门口,那里恰好有一个不耽误过路的小拐角可以放路由器。平常我们基本不在客厅活动,其他区域最近的 Wi-Fi 信号源是从路由器,因此我们大多数的设备连接的都是从路由器。从路由器的工作负荷很大。 从路由器个头小主路由器很多,散热不咋地。工作时间久了发热就容易发生数据包堵塞,丢包延迟高。需要把它电源拔掉重启。从宽带开通到现在,数据包堵塞影响网络的情况每个月会发生一次。有一次还影响了居家办公的视频会议。宽带维修师傅也给不出有效的法子,建议就是定期插拔从路由器电源。 从路由器和书房之间隔了两堵墙。信号到我书桌那个位置时,千兆网速已经衰减到只有 400-500Mbps 了,折损将近一半。叠加路由器发热的 debuff,书桌位置的网速最差的时候几乎和百兆宽带差不多。 我尝试过在光猫后台管理将路由器功率设置到「穿墙」模式,但没有任何作用。今天在后台研究了一番发现,原来我之前设置的功率是仅对主路由器生效,从路由器还是标准功率。要修

By Gray