乐趣区

Flutter三之搞定Dart一

前言一:接下来一段时间我会陆续更新一些列 Flutter 文字教程

更新进度: 每周至少两篇;

更新地点: 首发于公众号,第二天更新于掘金、思否等地方;

更多交流: 可以添加我的微信 372623326,关注我的微博:coderwhy

希望大家可以 帮忙转发,点击在看,给我更多的创作动力。

前言二:若干年后,你会发现,选错了公司和行业的损失远比选错了语言来的大。

而且计算机这个行业,只会一种编程语言显然是不现实的,接收一门新的语言,并没有你想象中的难!

一. Dart 介绍和安装

1.1. 认识 Dart

Google 为 Flutter 选择了 Dart 就已经是既定的事实,无论你多么想用你熟悉的语言,比如 JavaScript、Java、Swift、C++ 等来开发 Flutter,至少目前都是不可以的。

在讲解 Dart 的过程中,我会假定你已经有 一定的编程语言基础,比如 JavaScript、Java、Python、C++ 等。

其实如果你对编程语言 足够的自信,Dart 的学习过程甚至可以直接忽略:

  • 因为你学过 N 种编程语言之后,你会发现他们的 差异是并不大
  • 无非就是 语法上的差异 + 某些语言有某些特性,而某些语言没有某些特性而已;
  • 在我初次接触 Flutter 的时候,并没有专门去看 Dart 的语法,而是对于某些语法不太熟练的时候回头去了解而已;

所以,如果你对编程语言已经足够了解,可以跳过我们接下来的 Dart 学习:

  • 我也并不会所有特性都一一罗列,我会挑出比较重要的语言特性来专门讲解;
  • 某些特性可能会等到后面讲解 Flutter 的一些知识的时候单独拿出来讲解;

下面,我们就从安装 Dart 开始吧!

1.2. 安装 Dart

为什么还需要安装 Dart 呢?

事实上在安装 Flutter SDK 的时候,它已经内置了 Dart 了,我们完全可以直接使用 Flutter 去进行 Dart 的编写并且运行。

但是,如果你想单独学习 Dart,并且运行自己的 Dart 代码,最好去安装一个 Dart SDK。

下载 Dart SDK

到 Dart 的官方,根据不同的操作系统下载对应的 Dart

  • 官方网站:https://dart.dev/get-dart

无论是什么操作系统,安装方式都是有两种:通过工具安装 直接下载 SDK,配置环境变量

1. 通过工具安装

  • Windows 可以通过 Chocolatey
  • macOS 可以通过 homebrew
  • 具体安装操作官网网站有详细的解释

2. 直接下载 SDK,配置环境变量

  • 下载地址:https://dart.dev/tools/sdk/ar…
  • 我采用了这个安装方式。
  • 下载完成后,根据路径配置环境变量即可。

1.3. VSCode 配置

学习 Dart 过程中,我使用 VSCode 作为编辑器

  • 一方面编写代码非常方便,而且界面风格我也很喜欢
  • 另一方面我可以快速在终端看到我编写代码的效果

使用 VSCode 编写 Dart 需要安装 Dart 插件:我目前给这个 VSCode 装了四个插件

  • Dart 和 Flutter 插件是为 Flutter 开发准备的
  • Atom One Dark Theme 是我个人比较喜欢的一个主题
  • Code Runner 可以点击右上角的按钮让我快速运行代码

二. Hello Dart

2.1. Hello World

接下来,就可以步入正题了。学习编程语言,从祖传的 Hello World 开始。

在 VSCode 中新建一个 helloWorld.dart 文件,添加下面的内容:

main(List<String> args) {print('Hello World');
}

然后在终端执行 dart helloWorld.dart,就能看到Hello World 的结果了。

完成了这个执行过程之后,以你之前学习的编程语言来看,你能得到多少信息呢?

2.2. 程序的分析

接下来,就是我自己的总结:

  • 一、Dart 语言的入口也是 main 函数,并且必须显示的进行定义;
  • 二、Dart 的入口函数 main 是没有返回值的;
  • 三、传递给 main 的命令行参数,是通过 List<String> 完成的。

    • 从字面值就可以理解 List 是 Dart 中的集合类型。
    • 其中的每一个 String 都表示传递给 main 的一个参数;
  • 四、定义字符串的时候,可以使用单引号或双引号;
  • 五、每行语句必须使用分号结尾,很多语言并不需要分号,比如 Swift、JavaScript;

三. 定义变量

3.1. 明确声明(Explicit)

明确声明变量的方式, 格式如下:

变量类型 变量名称 = 赋值;

示例代码:

String name = 'coderwhy';
int age = 18;
double height = 1.88;
print('${name}, ${age}, ${height}'); // 拼接方式后续会讲解

注意事项: 定义的变量可以修改值, 但是不能赋值其他类型

String content = 'Hello Dart';
content = 'Hello World'; // 正确的
content = 111; // 错误的, 将一个 int 值赋值给一个 String 变量

3.2. 类型推导(Type Inference)

类型推导声明变量的方式, 格式如下:

var/dynamic/const/final 变量名称 = 赋值;

3.3.1. var 的使用

var 的使用示例:

  • runtimeType 用于获取变量当前的类型
var name = 'coderwhy';
name = 'kobe';
print(name.runtimeType); // String

var 的错误用法:

var age = 18;
age = 'why'; // 不可以将 String 赋值给一个 int 类型

3.3.2. dynamic 的使用

如果确实希望这样做, 可以使用 dynamic 来声明变量:

  • 但是在开发中, 通常情况下不使用 dynamic, 因为类型的变量会带来潜在的危险
dynamic name = 'coderwhy';
print(name.runtimeType); // String
name = 18;
print(name.runtimeType); // int

3.3.3. final&const 的使用

final 和 const 都是用于定义常量的, 也就是定义之后值都不可以修改

final name = 'coderwhy';
name = 'kobe'; // 错误做法

const age = 18;
age = 20; // 错误做法

final 和 const 有什么区别呢?

  • const 在赋值时, 赋值的内容必须是在编译期间就确定下来的
  • final 在赋值时, 可以动态获取, 比如赋值一个函数
String getName() {return 'coderwhy';}

main(List<String> args) {const name = getName(); // 错误的做法, 因为要执行函数才能获取到值
  final name = getName(); // 正确的做法}

final 和 const 小案例:

  • 首先, const 是不可以赋值为 DateTime.now()
  • 其次, final 一旦被赋值后就有确定的结果, 不会再次赋值
// const time = DateTime.now(); // 错误的赋值方式
final time = DateTime.now();
print(time); // 2019-04-05 09:02:54.052626

sleep(Duration(seconds: 2));
print(time); // 2019-04-05 09:02:54.052626

const 放在赋值语句的右边,可以共享对象,提高性能:

  • 这里可以暂时先做了解,后面讲解类的常量构造函数时,我会再次提到这个概念
class Person {const Person();
}

main(List<String> args) {final a = const Person();
  final b = const Person();
  print(identical(a, b)); // true

  final m = Person();
  final n = Person();
  print(identical(m, n)); // false
}

四. 数据类型

4.1. 数字类型

对于数值来说,我们也不用关心它是否有符号,以及数据的宽度和精度等问题。只要记着整数用 int,浮点数用double 就行了。

不过,要说明一下的是 Dart 中的 intdouble可表示的范围并不是固定的,它取决于运行 Dart 的平台。

// 1. 整数类型 int
int age = 18;
int hexAge = 0x12;
print(age);
print(hexAge);

// 2. 浮点类型 double
double height = 1.88;
print(height);

字符串和数字之间的转化:

// 字符串和数字转化
// 1. 字符串转数字
var one = int.parse('111');
var two = double.parse('12.22');
print('${one} ${one.runtimeType}'); // 111 int
print('${two} ${two.runtimeType}'); // 12.22 double

// 2. 数字转字符串
var num1 = 123;
var num2 = 123.456;
var num1Str = num1.toString();
var num2Str = num2.toString();
var num2StrD = num2.toStringAsFixed(2); // 保留两位小数
print('${num1Str} ${num1Str.runtimeType}'); // 123 String
print('${num2Str} ${num2Str.runtimeType}'); // 123.456 String
print('${num2StrD} ${num2StrD.runtimeType}'); // 123.46 String

4.2. 布尔类型

布尔类型中,Dart 提供了一个 bool 的类型, 取值为 true 和 false

// 布尔类型
var isFlag = true;
print('$isFlag ${isFlag.runtimeType}');

注意: Dart 中不能判断非 0 即真, 或者非空即真

Dart 的类型安全性意味着您不能使用 if(非 booleanvalue)或 assert(非 booleanvalue)之类的代码。

  var message = 'Hello Dart';
  // 错误的写法
  if (message) {print(message)
  }

4.3. 字符串类型

Dart 字符串是 UTF-16 编码单元的序列。您可以使用单引号或双引号创建一个字符串:

// 1. 定义字符串的方式
var s1 = 'Hello World';
var s2 = "Hello Dart";
var s3 = 'Hello\'Fullter';
var s4 = "Hello'Fullter";

可以使用三个单引号或者双引号表示多行字符串:

// 2. 表示多行字符串的方式
var message1 = '''
哈哈哈
呵呵呵
嘿嘿嘿 ''';

字符串和其他变量或表达式拼接: 使用 ${expression}, 如果表达式是一个标识符, 那么 {} 可以省略

// 3. 拼接其他变量
var name = 'coderwhy';
var age = 18;
var height = 1.88;
print('my name is ${name}, age is $age, height is $height');

4.4. 集合类型

4.4.1. 集合类型的定义

对于集合类型,Dart 则内置了最常用的三种:List / Set / Map

其中,List可以这样来定义:

// List 定义
// 1. 使用类型推导定义
var letters = ['a', 'b', 'c', 'd'];
print('$letters ${letters.runtimeType}');

// 2. 明确指定类型
List<int> numbers = [1, 2, 3, 4];
print('$numbers ${numbers.runtimeType}');

其中,set可以这样来定义:

  • 其实,也就是把 [] 换成 {} 就好了。
  • SetList 最大的两个不同就是:Set是无序的,并且元素是不重复的。
// Set 的定义
// 1. 使用类型推导定义
var lettersSet = {'a', 'b', 'c', 'd'};
print('$lettersSet ${lettersSet.runtimeType}');

// 2. 明确指定类型
Set<int> numbersSet = {1, 2, 3, 4};
print('$numbersSet ${numbersSet.runtimeType}');

最后,Map是我们常说的字典类型,它的定义是这样的:

// Map 的定义
// 1. 使用类型推导定义
var infoMap1 = {'name': 'why', 'age': 18};
print('$infoMap1 ${infoMap1.runtimeType}');

// 2. 明确指定类型
Map<String, Object> infoMap2 = {'height': 1.88, 'address': '北京市'};
print('$infoMap2 ${infoMap2.runtimeType}');

4.4.2. 集合的常见操作

了解了这三个集合的定义方式之后,我们来看一些最基础的公共操作

第一类,是所有集合都支持的获取长度的属性length

// 获取集合的长度
print(letters.length);
print(lettersSet.length);
print(infoMap1.length);

第二类, 是添加 / 删除 / 包含操作

  • 并且,对 List 来说,由于元素是有序的,它还提供了一个删除指定索引位置上元素的方法
// 添加 / 删除 / 包含元素
numbers.add(5);
numbersSet.add(5);
print('$numbers $numbersSet');

numbers.remove(1);
numbersSet.remove(1);
print('$numbers $numbersSet');

print(numbers.contains(2));
print(numbersSet.contains(2));

// List 根据 index 删除元素
numbers.removeAt(3);
print('$numbers');

第三类,是 Map 的操作

  • 由于它有 key 和 value,因此无论是读取值,还是操作,都要明确是基于 key 的,还是基于 value 的,或者是基于 key/value 对的。
// Map 的操作
// 1. 根据 key 获取 value
print(infoMap1['name']); // why

// 2. 获取所有的 entries
print('${infoMap1.entries} ${infoMap1.entries.runtimeType}'); // (MapEntry(name: why), MapEntry(age: 18)) MappedIterable<String, MapEntry<String, Object>>

// 3. 获取所有的 keys
print('${infoMap1.keys} ${infoMap1.keys.runtimeType}'); // (name, age) _CompactIterable<String>

// 4. 获取所有的 values
print('${infoMap1.values} ${infoMap1.values.runtimeType}'); // (why, 18) _CompactIterable<Object>

// 5. 判断是否包含某个 key 或者 value
print('${infoMap1.containsKey('age')} ${infoMap1.containsValue(18)}'); // true true

// 6. 根据 key 删除元素
infoMap1.remove('age');
print('${infoMap1}'); // {name: why}

五. 函数

5.1. 函数的基本定义

Dart 是一种真正的面向对象语言,所以即使函数也是对象,所有也有类型, 类型就是 Function。

这也就意味着函数可以作为变量定义或者作为其他函数的参数或者返回值.

函数的定义方式:

返回值 函数的名称(参数列表) {
  函数体
  return 返回值
}

按照上面的定义方式, 我们定义一个完整的函数:

int sum(num num1, num num2) {return num1 + num2;}

Effective Dart 建议对公共的 API, 使用类型注解, 但是如果我们省略掉了类型, 依然是可以正常工作的

sum(num1, num2) {return num1 + num2;}

另外, 如果函数中只有一个表达式, 那么可以使用箭头语法(arrow syntax)

  • 注意, 这里面只能是一个表达式, 不能是一个语句
sum(num1, num2) => num1 + num2;

5.2. 函数的参数问题

函数的参数可以分成两类: 必须参数和可选参数

前面使用的参数都是必须参数.

5.2.1. 可选参数

可选参数可以分为 命名可选参数 位置可选参数

定义方式:

命名可选参数: {param1, param2, ...}
位置可选参数: [param1, param2, ...]

命名可选参数的演示:

// 命名可选参数
printInfo1(String name, {int age, double height}) {print('name=$name age=$age height=$height');
}

// 调用 printInfo1 函数
printInfo1('why'); // name=why age=null height=null
printInfo1('why', age: 18); // name=why age=18 height=null
printInfo1('why', age: 18, height: 1.88); // name=why age=18 height=1.88
printInfo1('why', height: 1.88); // name=why age=null height=1.88

位置可选参数的演示:

// 定义位置可选参数
printInfo2(String name, [int age, double height]) {print('name=$name age=$age height=$height');
}

// 调用 printInfo2 函数
printInfo2('why'); // name=why age=null height=null
printInfo2('why', 18); // name=why age=18 height=null
printInfo2('why', 18, 1.88); // name=why age=18 height=1.88

命名可选参数, 可以指定某个参数是必传的(使用 @required, 有问题)

// 命名可选参数的必须
printInfo3(String name, {int age, double height, @required String address}) {print('name=$name age=$age height=$height address=$address');
}

5.2.2. 参数默认值

参数可以有默认值, 在不传入的情况下, 使用默认值

  • 注意, 只有可选参数才可以有默认值, 必须参数不能有默认值
// 参数的默认值
printInfo4(String name, {int age = 18, double height=1.88}) {print('name=$name age=$age height=$height');
}

Dart 中的 main 函数就是一个接受可选的列表参数作为参数的, 所以在使用 main 函数时, 我们可以传入参数, 也可以不传入

5.3. 函数是一等公民

在很多语言中, 函数并不能作为一等公民来使用, 比如 Java/OC. 这种限制让编程不够灵活, 所以现代的编程语言基本都支持函数作为一等公民来使用, Dart 也支持.

这就意味着你可以将函数赋值给一个变量, 也可以将函数作为另外一个函数的参数或者返回值来使用.

main(List<String> args) {
  // 1. 将函数赋值给一个变量
  var bar = foo;
  print(bar);

  // 2. 将函数作为另一个函数的参数
  test(foo);

  // 3. 将函数作为另一个函数的返回值
  var func =getFunc();
  func('kobe');
}

// 1. 定义一个函数
foo(String name) {print('传入的 name:$name');
}

// 2. 将函数作为另外一个函数的参数
test(Function func) {func('coderwhy');
}

// 3. 将函数作为另一个函数的返回值
getFunc() {return foo;}

5.4. 匿名函数的使用

大部分我们定义的函数都会有自己的名字,比如前面定义的 foo、test 函数等等。

但是某些情况下,给函数命名太麻烦了,我们可以使用没有名字的函数,这种函数可以被称之为匿名函数 (anonymous function),也可以叫lambda 或者closure

main(List<String> args) {
  // 1. 定义数组
  var movies = ['盗梦空间', '星际穿越', '少年派', '大话西游'];

  // 2. 使用 forEach 遍历: 有名字的函数
  printElement(item) {print(item);
  }
  movies.forEach(printElement);

  // 3. 使用 forEach 遍历: 匿名函数
  movies.forEach((item) {print(item);
  });
  movies.forEach((item) => print(item));
}

5.5. 词法作用域

dart 中的词法有自己明确的作用域范围,它是根据代码的结构 ({}) 来决定作用域范围的

优先使用自己作用域中的变量,如果没有找到,则一层层向外查找。

var name = 'global';
main(List<String> args) {
  // var name = 'main';
  void foo() {
    // var name = 'foo';
    print(name);
  }

  foo();}

5.6. 词法闭包

闭包可以访问其词法范围内的变量,即使函数在其他地方被使用,也可以正常的访问。

main(List<String> args) {makeAdder(num addBy) {return (num i) {return i + addBy;};
  }

  var adder2 = makeAdder(2);
  print(adder2(10)); // 12
  print(adder2(6)); // 8

  var adder5 = makeAdder(5);
  print(adder5(10)); // 15
  print(adder5(6)); // 11
}

5.7. 返回值问题

所有函数都返回一个值。如果没有指定返回值,则语句返回 null; 隐式附加到函数体。

main(List<String> args) {print(foo()); // null
}

foo() {print('foo function');
}

备注:所有内容首发于公众号,之后除了 Flutter 也会更新其他技术文章,TypeScript、React、Node、uniapp、mpvue、数据结构与算法等等,也会更新一些自己的学习心得等,欢迎大家关注

退出移动版