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"}
}

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 中发布的解决方案,我们可以通过以下方式实现
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;
}
public class BalanceBase: IBalace
{
}
需要更改余额类别
public Balance : BalanceBase
{
}
private IDictionary<string, Balance> GetBalances()
{
var postData = new Dictionary<string, object>();
var data = PostData<IDictionary<string, Balance>>("returnCompleteBalances", postData);
return data;
}