如何给 child class 属性 与 Dart 中的 superclass 不同的类型?

How to give child class property a different type from the superclass in Dart?

我正在处理实体和模型。 基本上我的实体描述了我的数据应该是什么样子,我的模型扩展了这些实体并有一个函数 toMap 将它们转换成 json object.

所以我有 2 个实体,Car 和 Brand,其中一个 Car 属性是 Brand 类型。

class Car {
  final String name;
  final Brand brand;

  const Car({
    required this.name,
    required this.brand,
  });
}

class Brand {
  final String label;
  final String color;

  const Brand({
    required this.label,
    required this.color,
  });
}

然后我有2个模型CarModel和BrandModel,分别扩展了Car和Brand。

class CarModel extends Car {
  CarModel({
    required String name,
    required BrandModel brand,
  }) : super(
          name: name,
          brand: brand,
        );

  Map<String, dynamic> toMap() {
    return {
      'name': name,
      'brand': brand,
    };
  }
}

class BrandModel extends Brand {
  BrandModel({
    required String label,
    required String color,
  }) : super(
          label: label,
          color: color,
        );

  Map<String, dynamic> toMap() {
    return {
      'label': label,
      'color': color,
    };
  }
}

如您所见,在 CarModel 中,我希望 brand 属性 具有类型 BrandModel。 但是,它采用其 superclass 类型。

所以最后,我的 CarModel 中的 brand 属性 的类型是 Brand.

我的问题是在 toMap 函数中,我想像这样使用 brand.toMap:

  Map<String, dynamic> toMap() {
    return {
      'name': name,
      'brand': brand.toMap(), // <= I want to use toMap() here, so I need brand to be of type BrandModel 
    };
  }

有没有办法强制我的 child class 为从超级 class 继承的 属性 设置不同的类型?

有趣的问题。您正试图实现 继承 。因此,当您在 CarModel 的构造函数中提供 required BrandModel brand, 时发生了什么,它扩展了 Car 具有 final Brand brand; 类型的字段。 BrandBrandModel object 获取必要的 attributes/definition,这是您无法在 Map<String, dynamic> toMap() 方法中调用 'brand': brand.toMap(), 的唯一原因因为 Brand class 没有这个方法。

Child具有Parent的所有attributes/functions/definition,但Parent不具有child的所有特性。

解决方案

因此,您需要将 toMap() 放在 Brand class 中。这里 Abstraction 进来了。在这个阶段,您可能不清楚 toMap() 方法,例如它在体内有什么,它会映射什么?简单地说,将其声明为 abstract。此外,将 class 声明为抽象 class。 抽象class应该有抽象方法。

现在,多态性进来了。在BrandModelclass中,toMap()方法将是一个覆盖方法。检查以下测试代码(完整示例)。

class Car {
  final String name;
  final Brand brand;

  const Car({
    required this.name,
    required this.brand,
  });
}

abstract class Brand {
  final String label;
  final String color;

  const Brand({
    required this.label,
    required this.color,
  });
  Map<String, dynamic> toMap();
}

class CarModel extends Car {
  CarModel({
    required String name,
    required BrandModel brand,
  }) : super(
          name: name,
          brand: brand,
        );

  Map<String, dynamic> toMap() {
    return {
      'name': name,
      'brand': brand.toMap(),
    };
  }
}
class BrandModel extends Brand {
  BrandModel({
    required String label,
    required String color,
  }) : super(
          label: label,
          color: color,
        );

  @override
  Map<String, dynamic> toMap() {
    return {
      'label': label,
      'color': color,
    };
  }
}

void main() {
  BrandModel brand = BrandModel(label:'Test',color : 'white');
  
  CarModel car = CarModel(name: 'T',brand : brand);
  
  print(car.toMap());
}