传递c#泛型类型作为参数
本文关键字:参数 泛型类型 传递 | 更新日期: 2023-09-27 18:04:55
我有一个接口声明为
public interface ISomething<T> where T : class
在接口的某个地方我有一个声明为
的成员[JsonProperty("someProperty")]
[JsonConverter(typeof(ConcreteTypeConverter<List<T>>))]
List<T> SomePropertyList{ get; set; }
我在ConcreteTypeConverter<List<T>>
上得到一个错误,说它不能使用类型(T)作为参数。我的ConcreteTypeConverter类接受类型T,并返回JSON反序列化所需的T的具体实现。这里的场景是T可以有大约20种不同的类型。但是我希望避免有20个这样的接口——这就是为什么我选择了一个泛型接口。
用法类似于
ISomething<SomeType> variable = new Something<SomeType>();
var list = variable.SomePropertyList;
其中SomeType是t的实际实现。在这种情况下是否有任何方法使用泛型?
我的ConcreteConverterClass派生自JsonConverter(使用Newtonsoft.Json):
public class ConcreteTypeConverter<TConcrete> : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return true;
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
serializer.Serialize(writer, value);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
return serializer.Deserialize<TConcrete>(reader);
}
}
我的准确错误是:
Attribute Argument cannot use type parameters
仅当SomePropertyList
不是具体类型时才需要ConcreteTypeConverter
,例如
[JsonProperty("someProperty")]
[JsonConverter(typeof(ConcreteTypeConverter<List<T>>))]
IList<T> SomePropertyList{ get; set; }
如果这不是问题,只需将属性声明更改为
[JsonProperty("someProperty")]
List<T> SomePropertyList{ get; set; }
引用:
[JsonConverter(typeof(ConcreteTypeConverter<List<T>>))]
List<T> SomePropertyList{ get; set; }
我得到一个错误在ConcreteTypeConverter
由于泛型参数不能出现在属性声明中,所以我给你一个建议:
public class ConcreteTypeConverter : JsonConverter
移除TConcrete
。因为ReadJson
返回一个对象,所以泛型的力量就会逐渐消失。
,
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
return serializer.Deserialize<object>(reader);
}
将TConcrete
替换为对象。如果这还不足以使您的代码工作,请尝试解决它。你的方法返回一个object
。
请在实现该接口的类上应用该属性。
[JsonProperty("someProperty")]
[JsonConverter(typeof(ConcreteTypeConverter<List<T>>))]
正如之前所说,泛型参数不能出现在属性声明中,我遇到了同样的问题,以前的建议不适合我,我需要一个SingleValueToCollectionConverter,它可以像泛型一样工作。反思帮助了我:
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
foreach (var prop in objectType.GetProperties())
{
var type = prop.PropertyType;
if (!type.IsClass)
continue;
var destination = Activator.CreateInstance(objectType);
var result = reader.TokenType == JsonToken.StartArray
? serializer.Deserialize(reader, objectType)
: new List<object> { serializer.Deserialize(reader, type) };
return Mapper.Map(result, destination, result.GetType(), destination.GetType());
}
return null;
}
像这样使用(如果Data将是json中的Object,它将被转换为列表):
public class BaseResponse<TData>
{
[JsonConverter(typeof(SingleValueToCollectionConverter))]
public List<TData> Data { get; set; }
}
我希望它能帮助到别人