C#JSON反序列化:Type是接口或抽象类,无法实例化

时间:2016-05-06 10:03:02

标签: c# json json-deserialization

我正在使用这个使用APIs to communicate with the online trading platform Poloniex的C#项目。

此代码应该可以在钱包中获得余额:

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

虽然此代码会出现此错误:

  

获取钱包时出错:无法创建Jojatekok.PoloniexAPI.WalletTools.IBalance类型的实例。
  Type是接口或抽象类,无法实例化   路径'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"}
}

3 个答案:

答案 0 :(得分:1)

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

例如,你不能这样做:

var something = new ISomething();

但是,你可以这样做:

ISomething something = new Something();

您只需要该界面的一些具体实现。类似的东西:

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

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

答案 1 :(得分:1)

这是我为使其发挥作用所做的工作。感谢来自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;
}

答案 2 :(得分:0)

由于对象的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实施其他类别 你可以直接使用GetBalance()方法的返回类型 作为IDictionary 并改变实施

     private IDictionary<string, Balance> GetBalances()
            {
    
                var postData = new Dictionary<string, object>();
                var data = PostData<IDictionary<string, Balance>>("returnCompleteBalances", postData);
    
                return data;
            }