2016-09-30 120 views
0

我们正在使用.Net 4.6构建Web Api应用程序我们正在尝试JsonConvert.DeserializeObject一个复杂对象。该对象具有复杂对象的列表,并且该对象是一个接口。 因此,我们有一个Task类具有TaskDetails列表和类有IBehaviorJson使用Interface作为属性反序列化一个复杂对象

public class Task 
{ 
    public int Id { get; set; } 
    public int TaskTypeId { get; set; } //TYPE 
    public List<TaskDetail> TaskDetails { get; set; } 
} 

public class TaskDetail 
{ 
    public int Id { get; set; } 
    public IBehavior Behavior { get; set; } 
} 

的属性和IBehavior是一个空的接口...

public interface IBehavior 
{ 
} 

和具体行为.. 。

public class PartPick : IBehavior 
{ 
    public bool AllowMultiplePicks { get; set; } 
    public bool RunLightsOnly { get; set; } 
    public bool StandardLightMode { get; set; } 

} 

或者

public class TorqueTool : IBehavior 
{ 
    public short PSet { get; set; } 
    public short RundownsRequired { get; set; } 
    public int MultiSpindleMask { get; set; } 
} 

JSON的有效载荷是:

{ 
    "id": 10000, 
    "name": "Attach Spoiler", 
    "taskTypeId": 1, 
    "behavior": { 
     "pSet": 1, 
     "rundownsRequired": 1, 
     "multiSpindleMask": 4, 
     "multiSpindleMaskString": "0010000000000000" 
    } 
} 

我收到以下错误:

"The request is invalid.","modelState":{"task.taskDetails[0].behavior.pSet":["Could not create an instance of type Bl.Models.EPA.IBehavior. Type is an interface or abstract class and cannot be instantiated. Path 'taskDetails[0].behavior.pSet'

我试图创建一个JsonBodyModelbinder:IModelBinder 在反序列化的方法是:

private static T DeserializeObjectFromJson(string json) 
{ 
    var binder = new TypeNameSerializationBinder(""); 

    var obj = JsonConvert.DeserializeObject<T>(json, new JsonSerializerSettings 
    { 
     TypeNameHandling = TypeNameHandling.Auto, 
     Binder = binder 
    }); 
    return obj; 
} 

它在httpConfiguration中配线:

config.Services.Insert(typeof(ModelBinderProvider), 0, 
      new SimpleModelBinderProvider(typeof(IBehavior), new JsonBodyModelBinder<IBehavior>())); 

我也试着把下面的属性放在TaskDetail中的IBehavior属性上。

[JsonConverter(typeof(JsonBodyModelBinder<IBehavior>))] 
public IBehavior Behavior { get; set; } 
+0

的可能的复制[铸造界面在JSON.NET反序列化(http://stackoverflow.com/questions/5780888/casting-interfaces-for-deserialization-in-json-净) – BWA

回答

0

事实证明,我可以将TaskDetail中的属性从IBehavior更改为动态,并且可以工作。然后,当我需要使用具体类型时,我有一个可以返回具体类型的工厂(即torqueTool)。在这一点上我可以做

JsonConvert.DeserializeObject<TorqueTool>(taskDetail.Behavior.ToString()); 
0

我相信Deserializer不知道该怎么映射接口,所以提供一些Deserializer提示可以做到这一点。

public class TaskDetail 
{ 
    TaskDetails(PartPick partPick) 
    { 
     Behavior = partPick; 
    } 

    TaskDetails(TorqueTool torqueTool) 
    { 
     Behavior = torqueTool; 
    } 

    public int Id { get; set; } 
    public IBehavior Behavior { get; set; } 
} 
0

也许this answer将适合你。基本上他们提出的是传递实现这个接口的类的实例。

public class TaskDetail 
{ 
    public TaskDetail(TorqueTool behavior){ 
     Behavior = behavior; 
    } 
    public int Id { get; set; } 
    public IBehavior Behavior { get; set; } 
} 
0

如果您正在使用的DataContractSerializer,您应该使用"KnownTypes"属性来定义应用程序,您所期待实现该接口的类型。

如果您正在使用XML序列化程序,则需要使用this问题中所见的每种类型的“XmlInclude”属性对其进行标记。