传递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

传递c#泛型类型作为参数

仅当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; }
}

我希望它能帮助到别人