在 dart 中实例化一个泛型类

Jan*_*ert 3 generics dart

我已经使用 typedef 查看了有关 stackoverflow 的示例,但看起来它主要用于回调,因此不确定它是否与我正在处理的内容相关。我正在使用执行 RPC 的泛型实现一个类......

abstract class Message {

  int created = new DateTime.now().millisecondsSinceEpoch;
  Map map = new Map();

  Map toJson();
  void fromJson(String json){
    map = JSON.decode(json);
    this.created = map["created"];
  }

  String toString() {
    return JSON.encode(this);
  }

  Message(){
    map["created"] = created;
  }

}
Run Code Online (Sandbox Code Playgroud)

___Request 和 ___Response 都扩展了 Message:

import 'Message.dart';

class TestResponse extends Message {

  String test;
  String message;

  Map toJson() {
    map["test"] = this.test;
    return map;
  }

  fromJson(String json) {
    super.fromJson(json);
    this.test = map["test"];
    this.message = map["message"];
  }

}
Run Code Online (Sandbox Code Playgroud)

现在,当我尝试执行一个隐藏所有发送和接收消息样板的通用 RPC 类时,我需要创建响应类的新实例以将其发回。(我更喜欢做 RPC.submit,但这给了我一个错误,说静态静态成员不能引用类型参数,所以我这里的另一个选择是可能滥用构造函数语法,例如 RPC.submit(json, uri).getResponse () ...)

import 'package:http/browser_client.dart';
import 'Message.dart';

class RPC<REQ extends Message, RES extends Message> {

  RES submit(REQ req, String uri){
    var client = new BrowserClient();
    var url = "http://localhost:9090/application-api" + uri;
    RES res = new RES(); // <----- can't do this
    client.post(url, body: req.toString()).then((response){
      print("Response status: ${response.statusCode}");
      print("Response body: ${response.body}");
      res.fromJson(response.body);

    });
    return res;  
  }

}
Run Code Online (Sandbox Code Playgroud)

在我的提交方法中,我显然可以传入一个“RES res”的实例并使用它,但我希望它可以在通用RPC内部完成而无需太多额外的样板,这在dart中以某种方式可能吗?

Gün*_*uer 5

似乎与http://dartbug.com/10667有关

我在类似情况下所做的是创建一个静态映射,将类型映射到封闭的构造函数。我使用消息类型和每个创建该类型的新实例的闭包来初始化映射。然后我使用 type 参数查找闭包并调用返回的闭包以获取新实例。

var factories = {
  'A': () => new A(),
  'B': () => new B(),
  'C': () => new C(),
};

...

var a = factories['A']();
Run Code Online (Sandbox Code Playgroud)

您可以将工厂集成到类中

class A {
  static A createNew() => new A();
}

var factories = {
  'A': A.createNew,
  'B': B.createNew,
  'C': C.createNew,
};
...
var a = factories['A']();
Run Code Online (Sandbox Code Playgroud)