![Flutter从0基础到App上线](https://wfqqreader-1252317822.image.myqcloud.com/cover/259/33831259/b_33831259.jpg)
3.3 基本数据类型
在Dart中,有几种内置的数据类型:数值型、字符串、布尔、集合、键值对等。
3.3.1 数值型
Dart的数值包含Int和Double两种类型,分别表示整数和小数。
Int:用于表示整数的int类型,长度为64位,因此,可以表示的数值范围是-263~263-1。编译为JavaScript语言的Dart使用JavaScript的数值范围-253~253-1。
Double:用于表示小数的double类型,提供长度为64位的双精度,其范围是-1.7E308~1.7E308。
从Dart 2.1版本开始,如果声明是double类型的变量,而值却是一个整数,那么这个变量就会自动转换成double类型(在Dart 2.1之前的版本中,这样的写法将会报错)。比如:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_19.jpg?sign=1739701638-DZTctrHm1weNyP1SVIqyRdGUdpq2603h-0-c68f2c400c2b98963b0ca12743df791b)
输出结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_20.jpg?sign=1739701638-zeYXMHKcHqfdjQVNAck3N6Zb2MXRCAnY-0-6b7b4a9ebe5d82364dbfd20fca8244a7)
由此可见,之前作为int类型的值(1)自动转换成了double类型的值。
3.3.2 字符串
Dart中的字符串表示一个由多个字符组成的序列,采用UTF-16编码。在声明字符串时,可以用单引号也可以用双引号,但要注意其中的区别。示例如下:
使用单引号声明的字符串:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_21.jpg?sign=1739701638-f6LWeMfv2rwAg6CDOzYJUjPGFZUSyJo1-0-4a34d5c33d1ad4ae8bfe9e244537508e)
使用双引号声明的字符串:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_22.jpg?sign=1739701638-i7QtFZlP0Xvk5oRNtTsE5tIRQrn6IelR-0-0e5bf500347fc5705d757e9a0ff5a593)
你可能会发现,在使用单引号声明的字符串中,有一个反斜杠(\),这个反斜杠称为转义字符。其目的是区分字符串中的单引号是作为值的内容存在的,而不是代表字符串值的结束。类似地,如果使用双引号来声明字符串,当字符串值中包含双引号时,也要在双引号前加反斜杠,意思是这个双引号是其值的一部分,而非意味着字符串的结束。
接下来,想象这样一个真实情况。当用户成功登录后,给出一条欢迎信息,如“张先生,欢迎您的使用”。在这里,无论是哪位用户登录系统,后面的欢迎词都不变,变的只是前面的称呼。对于这样的情况,可以使用类似于下面的代码片段简化代码量,提取不变的部分:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_23.jpg?sign=1739701638-sv8Ks1xKAWuAg5wZeq00wGpQoWYfppKc-0-815bdf13988f79d90bdb9c362ef79f6d)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_24.jpg?sign=1739701638-gs2iitpkXPg0llihkCaWzijd376I5wEO-0-eda51785674eb02d226451b91a3a0393)
在这段代码中,使用了{表达式}。如果表达式是一个变量,则可省略大括号来插入另一个变量值,最后组成一句话。在这段代码中,welcome变量的值是不变的,可以将其声明为final。
当然,对于上例,也可以使用加号(+)联结它们,具体写法如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_25.jpg?sign=1739701638-s3sohBW1mVmtkKHTEfyEgLsg3VUiJYpd-0-702dc99c66416bce4616d852e8446a83)
输出结果相同。
要想判断两个字符串(或其他对象)的内容是否完全一致,可以使用连等号(==),如下例所示:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_26.jpg?sign=1739701638-F8jtLLgKTsq4zhg2PL6Bszl2aL8JhYy0-0-18890d6b2faabcdb7504fe5d2a3a9fae)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_27.jpg?sign=1739701638-ON95NxQfq9miiESZZEZCsgXsVVt3rjRU-0-50afc6aa05efa4c162f4483bca2ac5a8)
接下来看下面这段代码:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_28.jpg?sign=1739701638-lhYjPKaORyHYumRhvdMeXUNkwoItrVgV-0-6fa3322866f8c35aca50d32f734b978d)
其中,“\n”是换行符。想象一下,如果几十道菜的菜单这样延续下去,可读性就非常差。因此,我们可以使用三个单引号或三个双引号将上面的变量声明改为如下所示:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_29.jpg?sign=1739701638-XYJ9W0ZZD76qVCOIa5JpHu3SXNP1U4Eu-0-7ebd7123fd921b62e926e31b116df079)
这样的代码更易读、整洁,而且也不需要再手动添加换行符,引号内的所有内容将以原样输出。然后使用print()方法在控制台输出longStr的值,将得到如下结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_30.jpg?sign=1739701638-c2w5bDFjX3jBUWYZjiMpbsBcR6pS8M5k-0-93f71ad6bfdff5f15f38d98acba8d6e9)
在某些特定的情况下,我们还需要在数字和字符串之间进行转换。比如,一个字符串的内容是1000,如果想得到其乘以5的结果,由于字符串是无法直接参与算术运算的,所以就需要先将其转换为int类型。代码片段如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_31.jpg?sign=1739701638-IWeKSgWrPe5R99kPpkr8K8LGntuJHalR-0-20d6315008526e1db31a76c4646fe522)
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_32.jpg?sign=1739701638-VDILJDW2cphrHMBrWizb8u6LL2Vquczd-0-80ffb70c8e3de55cc070be6c1f1bab49)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_33.jpg?sign=1739701638-jo5Ck43vXbNfAtXkirxN5QEpT3FBDTsd-0-0d4075d8860d896b577f4477eac761d1)
int类型转换为String类型,代码如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_34.jpg?sign=1739701638-LeF6dznhok234H9Yt43UA809SRk70nyW-0-028b0dc47dd565123e5a50c5032b5fa0)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_35.jpg?sign=1739701638-04eS3gNNXjxZa3fuljSvTyv07Z8DNW6Y-0-a8df3102c5efbf441dab59f5f927a6ba)
注意,虽然输出的结果看上去一样,但类型不同。类似地,对于double类型的转换也可采用同样的方式。
字符串值和带有一个或多个表达式插值的字符串都属于编译时常量。
3.3.3 布尔
布尔类型有两个结果值:true和false,用于表示真和假或作为条件判断的依据,可以使用var声明,也可以使用bool,如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_36.jpg?sign=1739701638-nRtG0l65uF3PLQ0DBsZIelvhYkt9z6nD-0-57f27fcc9d612b492f468db8265a0b33)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_37.jpg?sign=1739701638-KVa5tXclmcQG2IHVtzZ1fB8FJMroWuD3-0-1f576abbe2eb52ce649dc183bd18f0a6)
3.3.4 集合
Dart 2.0版本提供了三种核心集合类型,分别是列表(List)对象、集合(Set)对象和映射(Map)对象。
1.列表对象
在Dart中,使用数组表示列表对象。可以像如下所示,声明一个列表对象:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_38.jpg?sign=1739701638-dZCDcx98gDaRRweLvdGoUuGl4FxuHVHn-0-5a26696aeb5e0d38c356979198363947)
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_39.jpg?sign=1739701638-cD8oEreeRfIthew6XQsOk7asMrWgWQn2-0-cde27b0cd3237939d6828186e406754b)
当然,也可以用列表来声明。列表中的元素可以包含其他类型,如double类型和String类型,甚至可以将它们混合使用,如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_40.jpg?sign=1739701638-4g6WE9234v855KLwdldJUr1bMQwEWKny-0-6ed85a38c31e1be6c58ca0667993013e)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_41.jpg?sign=1739701638-xOYvOwEbfh6sEugZTevS3ZYsgvyagMdp-0-3e370a9888b3ec7515761fa4066eff30)
此外,与其他的编程语言类似,Dart中的列表也支持获取长度,即元素个数。调用length()方法即可获取某个列表的元素个数,如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_42.jpg?sign=1739701638-nxoRuSppNylPIgRm1xin6GKueA8HCp2O-0-f469fbfe5696d5883970d01ad9aba490)
其中,listExp变量是上面例子中的列表对象。
运行代码片段结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_43.jpg?sign=1739701638-I1b9aP038u2GckA0fnTj1BBTte15kJZQ-0-d75be18b3cd6fde4190b64fab6c30813)
列表中的元素索引下标从0开始到list.length-1结束,总长度是list.length。如果要取出第2个元素,对上例而言,就应写成listExp[1]。
为什么要使用这样的方法获取长度呢?通常来讲,当需要遍历某个列表中的所有对象时,会使用列表的长度作为重要的结束依据。换言之,取出列表中的每个元素需要进行遍历,到最后一个元素结束遍历。如果这个列表的长度可能发生改变,就不能把长度写成一个定值。而在上例中,获取到的长度却总是正确的,从而保证了代码逻辑既不会落下该遍历的元素,又不会产生下标越界的错误。
2.集合对象
集合对象是Dart编程语言中的第二种集合类型,和t列表对象不同,它没有顺序且不允许重复。声明集合对象很简单,如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_44.jpg?sign=1739701638-z1kRTVaS5vqvc9uDWZZVxuqvUKDyEZwz-0-e320485a06d67d4492ad94875ec428d3)
和列表对象类似,我们也可以使用集合来声明集合对象,并且也支持不同类型的混合使用,代码片段如下所示:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_45.jpg?sign=1739701638-sIHyGllPriOPqYr44gbo3yfPpEhnnMUB-0-311800ecce6f25a87278f66b6f462a5f)
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_46.jpg?sign=1739701638-DN74dC1n0lSARuxCapLCIK02d4ksf3g9-0-960e27324179d11ef48c9e47e22043d2)
输出结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_47.jpg?sign=1739701638-SHvgPxneXe8J9V6vwKar0D0zGz0MhELB-0-676790b69223d2f0fada25cc54df91a0)
从输出结果来看,似乎集合是有序的,但实际上,其内部是无序的,而且也不能像列表对象那样通过下标来取值;但是集合仍能通过set.length()方法获取集合大小,即元素个数。
除此之外,集合还提供了add(),remove(),contains(),clear()等方法。
add()方法为动态添加元素提供了方便。通过add()方法可以随时向集合对象中添加元素。当然,由于集合本身不具有包含重复对象的特性,因此添加一个重复的元素将是无效的代码片段,如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_48.jpg?sign=1739701638-BqwPYLVPb8zK3dbwYXif4UOe8e6bItc3-0-aa27de2462ad7fbb9e7555bb7f3ad127)
运行结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_49.jpg?sign=1739701638-bJOLxEDppXqa1eyMBXqwLBxiF4ZAeTx9-0-a2e2ff0f4f08afbb5a4a7cf95ad54b76)
其中,Elan作为新添加的值被追加到名为setExp的集合对象中,而setExp已经含有Bob,因此新添加的Bob无效。
remove()方法提供了删除某个元素的功能,现在将Bob从集合中删除,参考下面的代码:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_50.jpg?sign=1739701638-AO9SbcMOcdelOzmn5jxv8vnm5TRGKUau-0-1ca55ff29854cced6fe5377b2a34bc1b)
运行结果如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_51.jpg?sign=1739701638-KyTnFf2bcXEbcHmWFDWmt4WSs1HCemMN-0-1b3641ca9ba316da3077bcd09362cdf3)
可以看到,Bob已经从集合中移除了。
那么,我们怎么来确定集合中是否已经包含了某个元素呢?这时,就要用到contains()方法了。现在,分别测试Alice和Fiona是否包含在集合中,具体代码如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_52.jpg?sign=1739701638-2mNB1y3McAsarsB20yhnky5g1Ukd1b0V-0-fff94cffb3322f8c392717cc156bdd7f)
输出结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_53.jpg?sign=1739701638-njK6n7xPDFX1NXEtgpeeX6aujgr5oACL-0-1ef955f43927b01afeaedaba699a7344)
由此可见,Alice在集合中,Fiona不在集合中。
如果我们想一次性清空整个集合,就可以调用clear()方法。彻底清空setExp集合,如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_54.jpg?sign=1739701638-nq86CoknZPs3rVx69nD0AmhQDI1Wkvof-0-79f3019e68ce20cae371c4e8843d3a37)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_55.jpg?sign=1739701638-8AirwzFjCsbnrDkAA3sQaW1Q7MdovUEq-0-4521084bcb4c7447baea40a340701639)
到此,整个集合被清空了。
实际上,集合对象提供的实用和方便的方法还有很多。比如,一次性追加多个元素可以使用addAll()方法;类似地,还有removeAll()方法和containsAll()方法等。在实际开发中,这些方法提供了很多便利,掌握并灵活运用它们,可以提高开发效率。
3.映射对象
通常意义上讲,映射对象由一个或多个key(键)- value(值)对组成,每个键值对是一个元素,键和值可分别为任意类型。其中,相同的键在映射对象中不支持重复,相同的值可以和不同的键对应。比如,我们要送新年礼物给每个人,代码片段如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_56.jpg?sign=1739701638-CjaaiuntEwOILouRWKXsTn2xv3GmYKxJ-0-bdab4d4d8d22dd144ae906d7d85fca15)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_57.jpg?sign=1739701638-IofBH3gLPzgpDa6PM3w2qEakt4MOU6cJ-0-7b491bfeaf4c651b16018547b7b20000)
在上面的代码中,冒号左侧是键,右侧是值。而左侧的人(键)是不会重复的,右侧的礼物(值)是可以重复的。因为前文限定了每个人只能有一件礼物,也就意味着,一旦决定送给某人某件礼物,如果他想得到其他的,就只能用另一件礼物来替代。代码如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_58.jpg?sign=1739701638-LB9H0sIAKpAj8WEc3EGWTBCwL21e6vRR-0-c6c6f1b9e7ece675bddca7aed65ae63a)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_59.jpg?sign=1739701638-ybhL1CqtWhNzXIPj5Pi6FICJVlnQA5Ms-0-39055b04edc4032d5b55946dd59d4f1d)
因此,童童得到了乐高玩具,不再是精装书。这里要特别注意,在修改某个键对应的值时,务必要把键写对。上例代码如果写成如下所示:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_60.jpg?sign=1739701638-39pxd9yszwnznhvPB4tBTmp4A9iDOi5S-0-c8df6db5ec5b912b5e3840c11cce622c)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_61.jpg?sign=1739701638-9lOIXDSyTpG3aZ3yZmw7Sb7CZnilUziQ-0-51ed89fa539e788171a2e5819934ef1d)
可见,童童依然会得到精装书,而乐高玩具送给了彤彤。除此之外,如果想要知道送给某人的礼物是什么,也可以凭借键去获取,代码如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_62.jpg?sign=1739701638-XlJCfqbdUtJvvk9TYbrGBd4wheN0z7AH-0-7b916e51aed5f86e60bd7adf8831bb57)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_63.jpg?sign=1739701638-67ZHv4gqOdrqSCAkc5RUQxLNZTMDiMWE-0-e87c25608b842e72103c6c87222a5e4f)
和集合对象类似,要想删除某个元素可使用remove()方法,只是要用键来删除。再回到这个案例中,删掉刚刚错误添加进去的彤彤,操作如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_64.jpg?sign=1739701638-imBon28ORnGTJWZGOrvgZL0RelrZrn7K-0-f00a130bcb7b33844f52950b2f03d08a)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_65.jpg?sign=1739701638-dFdU2S8yW5LoIiKoMsFA3YUy8KYhWo8G-0-2f7c7660b7e96c61c10fd28663ce537f)
到此,由于失误添加进去的彤彤已被移除。那么,能不能避免这种加入新人的错误呢?
答案是肯定的。我们在对某个键值操作前,可以先判断这个键是否存在。比如,先判断一下有没有彤彤这个人,如果有,再进行操作;如果没有,就是找错人了。这样一来,就成功地避免了出现的失误。具体代码如下:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_66.jpg?sign=1739701638-K39SSKBPhREmTParpW1Gsmx9FXkEeXgR-0-4ad5e0846a942eb3415172a52b9d5e8e)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_67.jpg?sign=1739701638-RXHm26klnfjHhmOsAs75iSdAaC4BEs8B-0-68d946c60dc9b2beb087d7bb7a3586d1)
和列表、集合类似,映射同样支持使用length()方法获取整个映射对象的长度,即元素个数。
3.3.5 UTF-32编码表示法
前面提到在Dart中字符串是UTF-16编码的,那么对于超过16位编码的情况怎么处理呢?此时,就要借助Runes类型,利用它可以处理高达32位Unicode编码的字符串。在一般情况下,要表示一个32位Unicode字符,写法是“\uXXXX”,其中X代表一个十六进制数。比如,要输出一个心形(♥)的图案,可以参照如下的写法:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_68.jpg?sign=1739701638-jWXQZwBUW58tHmqLaz4dywJ69xjzM4ck-0-ca016748e159fd1954923958ffd9fe30)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_69.jpg?sign=1739701638-8FPD2HgD9gCB17VK4m9kJ8MF8zeeodh3-0-3d655a5c0fb18264a8411443ea6a9bf9)
除此之外,还有一些特殊的不是4个数字的情况,就需要使用大括号将数字部分括起来。比如要输出一个表情:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_70.jpg?sign=1739701638-G0YsbggU4h35qjshUjC5Mm5bvRnVtgdR-0-de8055c0c92d2ba29e354b82fe44b7a3)
运行结果:
![img](https://epubservercos.yuewen.com/9CF474/18096059801207706/epubprivate/OEBPS/Images/txt003_71.jpg?sign=1739701638-gr8720TLJm9n6LOB7rWJNui3UQLu0xFr-0-c70d8c7c9cd4de79a09ff4439c8f5e7e)
由此可见,我们可以简单地使用var来声明,也可以使用Runes来创建Runes对象,然后输出结果。