Dart 2(八)泛型

泛型

如果您查看基本数组类型List的API文档,您将看到该类型实际上是List<E> <..>标记列表为具有正式类型参数的泛型(或参数化)类型a。按照惯例,类型变量有单字母名称,例如E、T、S、K和V。
E, T, K, V是泛型中常用的几个名称,实际上定义泛型时完全可以不使用它们.

不过这几个字母用得人多了,也就有了可读性上的意义,这样可以更好的进行协作.

E代表Element,元素.

T代表Type,类型.

K代表Key,键.

V代表Value,值.

为什么使用泛型?

泛型通常是为了类型安全而必需的,但是它们比仅仅允许您的代码运行有更多的好处:

  • 适当地指定泛型类型会生成更好的代码。
  • 您可以使用泛型来减少代码重复。
    如果想要List只包含字符串,可以将其声明为list <String>(读作“String列表”)。通过这种方式,您的程序员伙伴和您的工具可以检测到向列表分配非字符串可能是一个错误。这里有一个例子:
var names = List<String>();
names.addAll(['Seth', 'Kathy', 'Lars']);
names.add(42); // Error

使用泛型的另一个原因是为了减少代码重复。泛型允许您在许多类型之间共享一个接口和实现,同时还可以利用静态分析。例如,假设您创建了一个缓存对象的接口:

abstract class ObjectCache {
  Object getByKey(String key);
  void setByKey(String key, Object value);
}

您发现需要此接口的特定字符串版本,因此创建了另一个接口:

abstract class StringCache {
  String getByKey(String key);
  void setByKey(String key, String value);
}

稍后,您决定要此接口的特定于数字的版本……您就知道了。
泛型类型可以省去创建所有这些接口的麻烦。相反,您可以创建一个带有类型参数的接口:

abstract class Cache<T> {
  T getByKey(String key);
  void setByKey(String key, T value);
}

在这段代码中,T是替代类型。它是一个占位符,您可以将其视为开发人员稍后将定义的类型。

使用集合文字

List和Map文字可以参数化。参数化的字面值就像您已经看到的字面值一样,只是在左括号前面添加了<type>(用于列表)或<keyType, valueType>(用于映射)。下面是使用类型化文字的例子:

var names = <String>['Seth', 'Kathy', 'Lars'];
var pages = <String, String>{
  'index.html': 'Homepage',
  'robots.txt': 'Hints for web robots',
  'humans.txt': 'We are people, not machines'
};

使用带构造函数的参数化类型

若要在使用构造函数时指定一个或多个类型,请将类型放在尖括号(<…>)中,刚好在类名之后。例如:

var names = List<String>();
names.addAll(['Seth', 'Kathy', 'Lars']);
var nameSet = Set<String>.from(names);

下面的代码创建了一个具有整型键和类型视图值的映射:

var views = Map<int, View>();

泛型集合及其包含的类型

Dart泛型类型被具体化了,这意味着它们在运行时携带着自己的类型信息。例如,您可以测试集合的类型:

var names = List<String>();
names.addAll(['Seth', 'Kathy', 'Lars']);
print(names is List<String>); // true

注意:相反,Java中的泛型使用擦除,这意味着泛型类型参数在运行时被删除。在Java中,可以测试对象是否为列表,但不能检查它是否为列表<String>。

限制参数化类型

在实现泛型类型时,您可能希望限制其参数的类型。您可以使用extend来实现这一点。

class Foo<T extends SomeBaseClass> {
  // Implementation goes here...
  String toString() => "Instance of 'Foo<$T>'";
}

class Extender extends SomeBaseClass {...}

可以使用SomeBaseClass或它的任何子类作为通用参数:

var someBaseClassFoo = Foo<SomeBaseClass>();
var extenderFoo = Foo<Extender>();

也可以不指定通用参数:

var foo = Foo();
print(foo); // Instance of 'Foo<SomeBaseClass>'

指定任何非SomeBaseClass类型将导致错误:

var foo = Foo<Object>();

使用泛型方法

最初,Dart的通用支持仅限于类。一种较新的语法,称为泛型方法,允许对方法和函数进行类型参数:

T first<T>(List<T> ts) {
  // Do some initial work or error checking, then...
  T tmp = ts[0];
  // Do some additional checking or processing...
  return tmp;
}

这里的第一个泛型类型参数(<T>)允许您在几个地方使用类型参数T:

  • 在函数的返回类型(T)中。
  • 在参数的类型中(列表<T>)。
  • 在局部变量(T tmp)的类型中。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • object 变量可指向任何类的实例,这让你能够创建可对任何数据类型进程处理的类。然而,这种方法存在几个严重的问题...
    CarlDonitz阅读 934评论 0 5
  • 写在开头:本人打算开始写一个Kotlin系列的教程,一是使自己记忆和理解的更加深刻,二是可以分享给同样想学习Kot...
    胡奚冰阅读 1,452评论 1 3
  • 付费就是检便宜! 凡事能用钱买的其实都是便宜的。 这句话可能很多人都不能理解,因为在我们的脑海里,钱才是最宝贵的东...
    俞到更好的自己阅读 198评论 2 0
  • 身为一个石头画爱好者(零基础小白),看到简书和京东手机发起的《大奖寻人|寻找能看到音乐的人,一起画音乐》这个活动很...
    呆兮阅读 654评论 0 3
  • 离别是,相思的种子 时间的雨露在 不停地浇灌着思念 生根发芽,茁壮成长 不是每一棵思念的树 都会随着时间而逝去 也...
    温差先生阅读 181评论 0 2