如何给 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;
类型的字段。 Brand
从 BrandModel
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应该有抽象方法。
现在,多态性进来了。在BrandModel
class中,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());
}
我正在处理实体和模型。 基本上我的实体描述了我的数据应该是什么样子,我的模型扩展了这些实体并有一个函数 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;
类型的字段。 Brand
从 BrandModel
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应该有抽象方法。
现在,多态性进来了。在BrandModel
class中,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());
}