我猜这个问题的简短回答是“不行”,但我对在C# 4.0中检测动态关键字在运行时的能力很感兴趣,特别是作为方法的泛型类型参数。
为了提供一些背景信息,我们在库中共享了一个RestClient类,它接受一个类型参数来指定在反序列化响应时应使用的类型,例如:
然而,将dynamic作为第一个方法的类型参数使用会导致一个非常奇怪的错误,掩盖了实际问题,使得整个调试过程变得头疼。为了帮助其他使用API的程序员,我想尝试检测第一个方法中是否使用了dynamic,以便在编译时根本无法编译或在使用时抛出异常,提示“如果您需要动态响应类型,请使用另一个方法”。基本上可以采取以下两种方式之一:
为了提供一些背景信息,我们在库中共享了一个RestClient类,它接受一个类型参数来指定在反序列化响应时应使用的类型,例如:
public IRestResponse<TResource> Get<TResource>(Uri uri, IDictionary<string, string> headers)
where TResource : new()
{
var request = this.GetRequest(uri, headers);
return request.GetResponse<TResource>();
}
很遗憾(由于篇幅原因不再赘述),使用dynamic作为类型参数来返回动态类型并不能正常工作 - 我们不得不向类中添加第二个签名以返回动态响应类型:
public IRestResponse<dynamic> Get(Uri uri, IDictionary<string, string> headers)
{
var request = this.GetRequest(uri, headers);
return request.GetResponse();
}
然而,将dynamic作为第一个方法的类型参数使用会导致一个非常奇怪的错误,掩盖了实际问题,使得整个调试过程变得头疼。为了帮助其他使用API的程序员,我想尝试检测第一个方法中是否使用了dynamic,以便在编译时根本无法编译或在使用时抛出异常,提示“如果您需要动态响应类型,请使用另一个方法”。基本上可以采取以下两种方式之一:
public IRestResponse<TResource> Get<TResource>(Uri uri, IDictionary<string, string> headers)
where TResource is not dynamic
或者
public IRestResponse<TResource> Get<TResource>(Uri uri, IDictionary<string, string> headers)
where TResource : new()
{
if (typeof(TResource).isDynamic())
{
throw new Exception();
}
var request = this.GetRequest(uri, headers);
return request.GetResponse<TResource>();
}
这两种情况有可能吗?我们正在使用VS2010和.Net 4.0,但是如果可以使用更高版本的语言特性,我会对.Net 4.5的解决方案感兴趣。
dynamic
时,运行时的TResource
只是一个简单的object
。用户是否会尝试使用Get<object>
?这是否需要与Get<dynamic>
不同处理? - Tim S.dynamic
在这里是不被允许的。除此之外,我不知道。 - Tim S.