C#JSON反序列化:类型是接口或抽象类,不能实例化

本文关键字:抽象类 不能 实例化 接口 反序列化 类型 C#JSON | 更新日期: 2025-02-19 13:02:38

我正在使用这个C#项目,该项目使用API与在线交易平台Poloniex进行通信。

这个代码应该在钱包中获得余额:

var x = await polo_client.Wallet.GetBalancesAsync();

尽管此代码给出了以下错误:

获取钱包时出错:无法创建Jojatekk.PoloniexAPI.WalletTools.IBalance类型的实例。
类型是接口或抽象类,不能实例化
路径"1CR.available",第1行,位置20。

在Helper.cs中点击此处:

[SuppressMessage("Microsoft.Usage", "CA2202:Do not dispose objects multiple times")]
internal static T DeserializeObject<T>(this JsonSerializer serializer, string value)
{ 
    using (var stringReader = new StringReader(value))
    {
        using (var jsonTextReader = new JsonTextReader(stringReader))
        {
            return (T)serializer.Deserialize(jsonTextReader, typeof(T));
        }
    } 
}

调用它的代码是:

public T GetData<T>(string command, params object[] parameters)
{
    var relativeUrl = CreateRelativeUrl(command, parameters);
    var jsonString = QueryString(relativeUrl);
    var output = JsonSerializer.DeserializeObject<T>(jsonString);
    return output;
}

有人能告诉我为什么反序列化这个JSON响应时出错吗?

响应是JSON,下面是一个示例:

{
"1CR":{"available":"0.00000000","onOrders":"0.00000000","btcValue":"0.00000000"},
"ABY":{"available":"0.00000000","onOrders":"0.00000000","btcValue":"0.00000000"}
}

C#JSON反序列化:类型是接口或抽象类,不能实例化

JSON与否,错误是自我描述的。不能实例化接口或抽象类。它们只是它们所代表的功能或对象结构的蓝图。

例如,你不能这样做:

var something = new ISomething();

但是,你可以这样做:

ISomething something = new Something();

您所需要的只是该接口的一些具体实现。类似于:

public JsonReceived : IBalance
{
// rest of the implementation
}

或者可能已经由第三方提供,请查看他们的SDK文档。

这就是我为使它工作所做的。感谢Guru.com的Chris让这一切顺利进行。很棒的家伙,价格也很好。你可以在这里找到他:http://www.guru.com/freelancers/deatc

private IDictionary<string, IBalance> GetBalances()
{
    var postData = new Dictionary<string, object>();
    var data = PostData<IDictionary<string, Balance>>("returnCompleteBalances", postData);
    var returnData = new Dictionary<string, IBalance>();
    foreach (string key in data.Keys)
    {
        returnData.Add(key, data[key]);
    }
    return returnData;
}

由于对象的DeSerialization导致的错误,该对象在内部创建了一个Jobject并将其作为输入Generic,在上述场景中,由于我们无法创建Interface的对象,它正在抛出错误,根据您在GitHub 中发布的解决方案,我们可以通过以下方式实现

  • 添加一个Hashtable作为PostData方法的泛型输入
  • 将wallet.cs更改为例如:
    IBalance CreateFactory(string typeOfBalance, string jsonString)
            {
                IBalance balance = null;
                switch (typeOfBalance)
                {
                    case "balanceClassIdentification":
                        balance = new Balance();
                        balance = Newtonsoft.Json.JsonConvert.DeserializeObject<Balance>(jsonString);
                        break;
                    case "":
                    // write logic if Ibalance interface implemented in any childClass ex: class stockBalance: IBalance , then balance = new StockBalance();balance = Newtonsoft.Json.JsonConvert.DeserializeObject<stockBalance>(jsonString);
                    default:
                        balance = new Balance();
                        balance = Newtonsoft.Json.JsonConvert.DeserializeObject<Balance>(jsonString);
                        break;
                }
                return balance;
            }
            private IDictionary<string, IBalance> GetBalances()
            {
                IDictionary<string, IBalance> data = null;
                var postData = new Dictionary<string, object>();
                //var data = PostData<IDictionary<string, object>>("returnCompleteBalances", postData);
                Hashtable output = PostData<Hashtable>("returnCompleteBalances", postData);
                if (output.Count > 0)
                {
                    data = new Dictionary<string, IBalance>();
                }
                foreach (string jsonObjKey in output.Keys)
                {
                    IBalance balace = CreateFactoty("balanceClassIdentification", output[jsonObjKey].ToString()); // this "balanceClassIdentification" just used as a string but we can add the condition in createFactory according to the type of object that wallet
                    data.Add(jsonObjKey, balace);
                }
                return data;
            }
    

  • 创建另一个实现Ibalance的中间类
  • public class BalanceBase: IBalace
     {
     } 
    

    需要更改余额类别

     public Balance : BalanceBase
        {
        }
    

  • 如果IBalance除了balance没有实现其他类
  • 可以直接使用GetBalance()方法的返回类型作为IDictionary并改变的实施
     private IDictionary<string, Balance> GetBalances()
            {
                var postData = new Dictionary<string, object>();
                var data = PostData<IDictionary<string, Balance>>("returnCompleteBalances", postData);
                return data;
            }