• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

Fluter基础巩固之Dart语言详解 二

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

继续学习枯燥的Dart语言语法,目前的耐得住寂寞是为了将来学得“爽”做准备的!!!

Dart 提供了 Exception 和 Error 类型, 以及一些子类型。还可以定义自己的异常类型。但是,Dart 代码可以抛出任何非 null 对象为异常,不仅仅是实现了 Exception 或者 Error 的对象。

Exception类型:

其中常见的Exception如下:

Error类型:

抛出:

所有的 Dart 异常是非检查异常。 方法不一定声明了他们所抛出的异常, 并且你不要求捕获任何异常【这个Kotlin也差不多】。
Dart 代码可以抛出任何非 null 对象为异常,不仅仅是实现了 Exception 或者 Error 的对象。

捕获:

  • 可以使用on 或者 catch 来声明捕获语句,也可以 同时使用。使用 on 来指定异常类型,使用 catch 来 捕获异常对象。
  • catch() 可以带有一个或者两个参数, 第一个参数为抛出的异常对象, 第二个为堆栈信息 (一个 StackTrace 对象)。

    还可以用on来指定异常类型,如下:

    如果抛一个非error的呢?

    还可以用精确异常:

    也能这样写:

    完整的写法可以如下:

  • 可以使用rethrow把捕获的异常重新抛出。

类:

构造函数:

命名构造函数:

使用命名构造函数可以为一个类实现多个构造函数, 或者使用命名构造函数来更清晰的表明你的意图。

重定向构造函数:

一个重定向构造函数是没有代码的,在构造函数声明后,使用冒号调用其他构造函数。

初始化列表:

  • 在构造函数体执行之前可以初始化实例参数。 使用逗号分隔初始化表达式。
  • 初始化列表非常适合用来设置 final 变量的值。

调用超类构造函数:

  • 超类命名构造函数不会传递,如果希望使用超类中定义的命名构造函数创建子类,则必须在子类中实现该构造函数。
  • 如果超类没有默认构造函数, 则你需要手动的调用超类的其他构造函数。
  • 调用超类构造函数的参数无法访问 this。
  • 在构造函数的初始化列表中使用 super(),需要把它放到最后。

 

其中要注意一个细节:

常量构造函数:

  • 定义const构造函数要确保所有实例变量都是final。
  • const关键字放在构造函数名称之前。

注意必须是final修饰变量,换成const或去掉都是要报错的:

有啥用呢?再来看:

这种写法在未来flutter的学习中会大量运用到的。

工厂构造函数【用得很多】:

  • 工厂构造函数是一种构造函数,与普通构造函数不同,工厂函数不会自动生成实例,而是通过代码来决定返回的实例对象。
  • 如果一个构造函数并不总是返回一个新的对象,则使用 factory 来定义这个构造函数。
  • 工厂构造函数无法访问this。

 

要记住这样的写法,因为在未来实际场景中会大量看到~~

Setter和Getter:

  • 每个实例变量都隐含的具有一个 getter, 如果变量不是 final 的则还有一个 setter。
  • 可以通过实行 getter 和 setter 来创建新的属性, 使用 get 和 set 关键字定义 getter 和 setter。
  • getter 和 setter 的好处是,你可以开始使用实例变量,后来 你可以把实例变量用函数包裹起来,而调用你代码的地方不需要修改。

代码来瞅一下:

抽象类:

  • 不能被实例化,除非定义一个工厂构造函数。
  • 抽象类通常用来定义接口, 以及部分实现。
  • 抽象类通常具有抽象方法,抽象方法不需要关键字,以分号结束即可。
  • 接口方式使用时,需要重写抽象类的成员变量和方法,包括私有的。
  • 一个类可以implement一个普通类。Dart任何一个类都是接口
  • 一个类可以implement多个接口。

比如说按摩可以有多个种类,咱们先来定义一个抽象的按摩类:

接下来来创建具体的子类,如下:

调用一下:

以上就是一种工厂方法的典型使用,需要特别注意:在Dart中木有interface关键字,abstract就可以当接口来用,下面再来看以继承的方式来使用:

这是第二种工厂模式的写法,对其有个大致印象既可,在之后的实际开发中会有用武之地滴。

可调用类:

实现call()方法可以让类像函数一样能够被调用。

Mixin:

  • 子类没有重写超类A方法的前提下,如果2个或多个超类拥有相同签名的A方法,那么子类会以继承的最后一个超类中的A方法为准。
  • 如果子类自己重写了A方法则以本身的A方法为准。

下面以这个场景为例:

下面来看下代码,看mixin是个啥东东?

void main() {
  Bicycle().transport();
  Motocycle().transport();
  Car().transport();
}

//交通抽象类
abstract class Transportation {
  void transport();
}

//自行车
class Bicycle extends Transportation {
  String safeIndex() => "low";

  String powerUnit() => "2个轮子";

  String energy() => "脚蹬";

  @override
  void transport() {
    print('Bicycle:\npowerUnit: ${powerUnit()}, safeIndex: ${safeIndex()}, energy: ${energy()}');
  }
}

//摩托车
class Motocycle extends Transportation {
  String safeIndex() => "low";

  String powerUnit() => "2个轮子";

  String energy() => "汽油";

  @override
  void transport() {
    print('Motocycle:\npowerUnit: ${powerUnit()}, safeIndex: ${safeIndex()}, energy: ${energy()}');
  }
}

//汽车
class Car extends Transportation {
  String safeIndex() => "middle";

  String powerUnit() => "4个轮子";

  String energy() => "汽油";

  @override
  void transport() {
    print('Car:\npowerUnit: ${powerUnit()}, safeIndex: ${safeIndex()}, energy: ${energy()}');
  }
}

运行如下:

好,接下来要改造了,从上面的代码来看:

好,咱们将里面的行为再单独抽象一下,如下:

接下来咱们使用Mixin的方式来改造代码:

void main() {
  Bicycle().transport();
  Motocycle().transport();
  Car().transport();
}

//交通抽象类
abstract class Transportation {
  void transport();
}

//2个轮子
abstract class TwoWheelTransportation {
  String powerUnit() => "2个轮子";
}

//4个轮子
abstract class FourWheelTransportation {
  String powerUnit() => "2个轮子";
}

//低安全系数
abstract class LowSafeTransportation {
  String safeIndex() => "low";
}

//中安全系数
abstract class MiddleSafeTransportation {
  String safeIndex() => "middle";
}

//人工动力
abstract class BodyEnergyTransportation {
  String energy() => "脚蹬";
}

//汽油动力
abstract class GasEnergyTransportation {
  String energy() => "汽油";
}

//自行车
class Bicycle extends Transportation with LowSafeTransportation, BodyEnergyTransportation, TwoWheelTransportation {

  @override
  void transport() {
    print(
        'Bicycle:\npowerUnit: ${powerUnit()}, safeIndex: ${safeIndex()}, energy: ${energy()}');
  }
}

//摩托车
class Motocycle extends Transportation with LowSafeTransportation, GasEnergyTransportation, TwoWheelTransportation {

  @override
  void transport() {
    print(
        'Motocycle:\npowerUnit: ${powerUnit()}, safeIndex: ${safeIndex()}, energy: ${energy()}');
  }
}

//汽车
class Car extends Transportation with MiddleSafeTransportation, GasEnergyTransportation, FourWheelTransportation {
  String safeIndex() => "middle";

  String powerUnit() => "4个轮子";

  String energy() => "汽油";

  @override
  void transport() {
    print(
        'Car:\npowerUnit: ${powerUnit()}, safeIndex: ${safeIndex()}, energy: ${energy()}');
  }
}

其运行结果跟之前是一样的:

这里还涉及到一个顺序问题,下面再来看下:

调用一下:

这个结果可以发现:

这俩打印又能说明啥现象呢?

这块东东先有个大体印象吧,没有真实的项目做为操练也不可以意会得非常深刻。。

泛型:

泛型函数:

Dart1.21开始可以使用泛型函数。
泛型函数可以在以下几个地方使用类型参数:
<1> 函数的返回值类型。
<2> 参数的类型。
<3> 局部变量的类型。

构造函数泛型:

要在使用构造函数时指定一个或多个类型,可将类型放在类名称后面的尖括号<...>中。

泛型限制:

实现泛型类型时,您可能希望限制其参数的类型,可以在<>里面使用extends,其实也就是泛型的协变跟逆变,但是在Dart中木有super限制,如下:

与Java的区别:

  • Java中的泛型信息是编译时的,泛型信息在运行时是不存在的。
  • Dart的泛型类型是固化的,在运行时也有可以判断的具体类型。

如何理解?看程序:

也就是说在Java中,可以测试对象是否为List,但无法测试它是否是List<String>。

库:

使用核心库:

  • import 后的必须参数为库 的 URI。(Uniform Resource Identifier统一资源标识符)
  • 对于内置的库,URI 使用特殊的 dart: scheme。
  • 对于其他的库,你可以使用文件系统路径或者 package: scheme。

载入三方库:

pubspec.yaml声明需要引用的库,使用Packages get进行拉取。 

那咱们来找一个三方库来使用一下,先上官网搜一下三方库:

 

所以咱们将其依赖添加至咱们的工程:

添加完依赖之后,就可以使用了,如下:

载入文件:

如何来引用咱们自己的Dart文件呢?下面新建待引用的Dart文件:

然后引用一下:

指定库前缀:

如果两个库有冲突的标识符,可以为其中一个或两个库都指定前缀,比如:MyLib1.dart 和 MyLib2.dart 都有一个名字为 MyLib 的类, 此时解决冲突就可以指定前缀来解决,如下:

选择性载入:

如果只使用库的一部分功能,则可以选择需要导入的内容。 

下面来演示一下:

这功能还挺灵活的。。

延迟载入:

  • 使用 await 关键字暂停代码执行一直到库加载完成。
  • 可提高程序启动速度。
  • 用在不常使用的功能。
  • 用在载入时间过长的包。
  • 执行 A/B 测试,例如 尝试各种算法的 不同实现。

下面来使用一下,我们在请求完数据之后再来加载文件:

看下结果:

感觉Dart好强大呀,值得一学!!!

自定义库:

  • part 可以把一个库分开到多个 Dart 文件中。
  • 或者我们想让某一些库共享它们的私有对象的时候,可以需要使用part。
  • import不会完全共享作用域,而part之间是完全共享的。如果说在A库中import了B库,B库import了C库,A库是没有办法直接使用C库的对象的。而B,C若是A的part,那么三者共享所有对象。并且包含所有导入。

啥意思,比如说有三个Dart文件:

 

 而mylib.dart利用part整合了tool.dart和util.dart文件的功能,最终我们使用时只要引用mylib则可以拥有三个文件的功能,如下:

咱们来试一下:

 

 

运行:


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
7.Classes-类(Dart中文文档)发布时间:2022-07-13
下一篇:
dart中list的map方法获取index发布时间:2022-07-13
热门推荐
热门话题
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap