Flutter循序渐进==>封装、继承、多态、抽象类以及属性修改

发布于:2024-06-28 ⋅ 阅读:(12) ⋅ 点赞:(0)

导言

新学一门编程语言,最难以理解的莫过于类了。如果类没用,也就算了,它偏偏很有用,我们必须得掌握,不然怎么好意思说自己会面向对象编程呢?

抽象类(Abstract Class)在面向对象编程中扮演着非常重要的角色,主要用于定义一个或多个抽象方法的模板,供子类继承并实现这些抽象方法。抽象类不能被实例化,它的主要用途包括:

  1. 定义接口规范:抽象类可以包含抽象方法(没有具体实现的方法),这些方法只有声明没有具体的实现。子类继承抽象类时必须实现这些抽象方法,这样就强制要求子类遵循一定的规范,实现了功能的统一和接口的一致性。

  2. 提供默认实现:虽然抽象类的主要目的是定义接口,但它也可以提供一些具体实现的方法。这样,子类可以继承这些方法而无需重新实现,既提供了灵活性也保证了代码的重用。

  3. 设计模式与架构:在软件设计模式和架构中,抽象类经常用于定义框架或基础结构,而具体的实现细节留给子类完成。这有助于分离关注点,提高代码的模块化和可维护性。

  4. 提升代码的可扩展性:通过定义抽象类和抽象方法,可以轻松地增加新的子类来扩展功能,而无需修改现有的抽象类或其它已有的子类,符合开闭原则(Open-Closed Principle)。

把类搞的低小下,让从没编过程的都能听懂

封装

封装挺好,不用知道里面的复杂,直接拿起来就用。大自然就是很会封装的,每个人都封装得很好。在这里封装主要是把数据封装到一个类里面。如果是私有,就在前面_,这是Dart的搞法。

继承

子类继承了父类所有的特征,父传子,甲天下!

多态

子类可以青出于蓝胜于蓝,搞自己的,表现就是@override,覆盖父类的方法!

class General {
  //name和company可以叫字段、变量或者属性,都是一回事,不知道谁搞那么复杂?
  String _name = '';
  String _company = '';

  //构造函数,用来初始化变量,这是类General的构造函数,用于初始化新创建的General对象的属性。
  // 构造函数的参数与类的属性名称相同,因此使用了Dart的快捷语法(被称为“命名参数构造”)。
  // 当你创建类的实例时,传递给构造函数的参数值会自动赋给相应的属性。
  //老粗说法:就是调用类(实例化)的参数,给这个构造函数,构造函数传递给类的属性,是类和外界通信的桥梁.

  //试图限制General类的name属性不能被设置为"吕布",但是实际上在创建General类实例时,直接传递
  // 了'吕布'作为name参数,这一步绕过了setter方法的检查。这是因为构造函数直接访问了 _name 字
  // 段,而不是通过setter方法设置。所以,尽管在setter方法中设置了限制条件,但在构造函数中
  // 直接赋值时并没有执行这个限制检查。

  // 在构造函数中使用setter方法来确保name的值不被设置为"吕布"
  General(String name, String company) {
    this.name = name; // 这里间接调用了setter方法,进行值的设置及检查
    this._company = company;
  }

  // Getter for name 把_name的值用getter薅过来
  String get name => _name;

  // Setter for name with validation,开始修改,根据传进来的参数来搞事。
  set name(String value) {
    if (value != '吕布') {
      _name = value;
    } else{
      _name = '三姓家奴';
    }
    // 注意:如果传入的是"吕布",这里不做任何操作,相当于拒绝了该设置。
  }

  String get company => _company;
  // 注意:没有提到对_company的设置逻辑,所以我保留了直接的getter,没有setter。

  //方法就是类里面的函数,用来搞动作
  void selfIntro(){
    print('大家好,我是$_name,我来自$_company!');
  }
}

//搞个小类,前面已经有将军了,就没必要重新搞了,搞个刘备的将军,继承将军类
//定义了一个名为liubeiGeneral的新类,它通过extends General继承自General类。
// 这意味着liubeiGeneral类将拥有General类的所有属性和方法,并可以在此基础上扩展或修改。
class liubeiGeneral extends General{
  //liubeiGeneral类的构造函数通过使用super关键字调用父类General的构造函数。
  // 这里super.name和super.company分别指代父类构造函数中对应的参数,
  // 这样可以确保子类实例化时也能初始化父类的属性。这是一种简化的构造函数声明方式
  // ,用于直接委托给超类构造函数处理参数。
  liubeiGeneral(super.name, super.company);
//在子类liubeiGeneral中,新增了一个名为han的方法,用于打印一条关于效忠汉室的信息。
// 这个方法是liubeiGeneral类特有的,不在其父类General中。
  void han() {
    print('我们都是为汉室效忠!');
  }
  //我不喜欢老的开场白,自己写一个覆盖原来的
  @override
  void selfIntro(){
    print('我是$_name,我为$_company代言!');
  }
}

//抽象类
abstract class HanGeneral {
  void fight();
}
//把抽象类写详细,抽象类是老领导,搞战略,子类是马仔,去跑腿执行
class ShuGeneral extends HanGeneral {
  @override
  void fight(){
    print('我们誓死捍卫汉室!');
  }
}

void main(){
  //类打个括号就是实例化,并把参数传递给构造函数,name为'赵云',company为'刘备集团'。
  General ZhaoYun = General('吕布', '刘备集团');
  //调用Zhaoyun实例(就是具体的赵云人)的selfIntro方法,就打印出
  //大家好,我是赵云,我来自刘备集团!
  ZhaoYun.selfIntro();

  liubeiGeneral Guanyu = liubeiGeneral('吕布', '刘备集团');
  Guanyu.selfIntro();
  Guanyu.han();
  ShuGeneral().fight();
}

代码

class General {
  //name和company可以叫字段、变量或者属性,都是一回事,不知道谁搞那么复杂?
  String _name = '';
  String _company = '';

  //构造函数,用来初始化变量,这是类General的构造函数,用于初始化新创建的General对象的属性。
  // 构造函数的参数与类的属性名称相同,因此使用了Dart的快捷语法(被称为“命名参数构造”)。
  // 当你创建类的实例时,传递给构造函数的参数值会自动赋给相应的属性。
  //老粗说法:就是调用类(实例化)的参数,给这个构造函数,构造函数传递给类的属性,是类和外界通信
  //的桥梁.

  //试图限制General类的name属性不能被设置为"吕布",但是实际上在创建General类实例时,直接传递
  // 了'吕布'作为name参数,这一步绕过了setter方法的检查。这是因为构造函数直接访问了 _name 字
  // 段,而不是通过setter方法设置。所以,尽管在setter方法中设置了限制条件,但在构造函数中
  // 直接赋值时并没有执行这个限制检查。


  // 在构造函数中使用setter方法来确保name的值不被设置为"吕布"
  General(String name, String company) {
    this._name = name; // 这里间接调用了setter方法,进行值的设置及检查
    this._company = company;
  }

  // Getter for name 把_name的值用getter薅过来
  String get name => _name;

  // Setter for name with validation,开始修改,根据传进来的参数来搞事。
  set name(String value) {
    if (value != '吕布') {
      _name = value;
    } else{
      _name = '三姓家奴';
    }
    // 注意:如果传入的是"吕布",这里不做任何操作,相当于拒绝了该设置。
  }

  String get company => _company;
  // 注意:没有提到对_company的设置逻辑,所以我保留了直接的getter,没有setter。

  //方法就是类里面的函数,用来搞动作
  void selfIntro(){
    print('大家好,我是$_name,我来自$_company!');
  }
}

//搞个小类,前面已经有将军了,就没必要重新搞了,搞个刘备的将军,继承将军类
//定义了一个名为liubeiGeneral的新类,它通过extends General继承自General类。
// 这意味着liubeiGeneral类将拥有General类的所有属性和方法,并可以在此基础上扩展或修改。
class liubeiGeneral extends General{
  //liubeiGeneral类的构造函数通过使用super关键字调用父类General的构造函数。
  // 这里super.name和super.company分别指代父类构造函数中对应的参数,
  // 这样可以确保子类实例化时也能初始化父类的属性。这是一种简化的构造函数声明方式
  // ,用于直接委托给超类构造函数处理参数。
  liubeiGeneral(super.name, super.company);
//在子类liubeiGeneral中,新增了一个名为han的方法,用于打印一条关于效忠汉室的信息。
// 这个方法是liubeiGeneral类特有的,不在其父类General中。
  void han() {
    print('我们都是为汉室效忠!');
  }
  //我不喜欢老的开场白,自己写一个覆盖原来的
  @override
  void selfIntro(){
    print('我是$_name,我为$_company代言!');
  }
}

//抽象类
abstract class HanGeneral {
  void fight();
}

class ShuGeneral extends HanGeneral {
  @override
  void fight(){
    print('我们誓死捍卫汉室!');
  }
}

void main(){
  //类打个括号就是实例化,并把参数传递给构造函数,name为'赵云',company为'刘备集团'。
  General ZhaoYun = General('吕布', '刘备集团');
  //调用Zhaoyun实例(就是具体的赵云人)的selfIntro方法,就打印出
  //大家好,我是赵云,我来自刘备集团!
  ZhaoYun.selfIntro();

  liubeiGeneral Guanyu = liubeiGeneral('吕布', '刘备集团');
  Guanyu.selfIntro();
  Guanyu.han();
  ShuGeneral().fight();
}

运行结果