无法反序列化具有<t>列表的对象,其中T是从接口派生的

时间:2018-09-10 20:57:39

标签: c# json serialization deserialization json-deserialization

我有一个类型为Job的对象,并且已成功将其序列化为JSON文件,但遇到此问题

  

无法将System.String强制转换或转换为   qConnector_v2._0.Model.Job

这是我进行序列化的方式:

_Job.Tasks.Add(new Email());
_Job.Tasks.Add(new UNC());
_Job.Tasks.Add(new FTP());
_Job.Tasks.Add(new WebApi());
_Job.Tasks.Add(new WebService());
_Job.Tasks.Add(new StoredProcedure());
_Job.Tasks.Add(new Odbc());
Newtonsoft.Json.JsonSerializer serializer = new Newtonsoft.Json.JsonSerializer();
serializer.Converters.Add(new Newtonsoft.Json.Converters.JavaScriptDateTimeConverter());
serializer.NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore;
serializer.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.Auto;
serializer.Formatting = Newtonsoft.Json.Formatting.Indented;

using (StreamWriter sw = new StreamWriter(@"C:\Temp\res.json"))
using (Newtonsoft.Json.JsonWriter writer = new Newtonsoft.Json.JsonTextWriter(sw))
{
    serializer.Serialize(writer, _Job, typeof(Model.Job));
}

这是我进行反序列化的方式:

Model.Job obj = Newtonsoft.Json.JsonConvert.DeserializeObject<Model.Job>(File.ReadAllText(@"C:\Temp\res.json"), new Newtonsoft.Json.JsonSerializerSettings
{
    TypeNameHandling = Newtonsoft.Json.TypeNameHandling.Auto,
});

有关更多详细信息,请参见以下类的结构:

主要类别-工作

class Job : IDisposable
{
    private Guid _ID;
    public Job()
    {
        ID = Guid.NewGuid();

    }
    public Guid ID { get { return _ID; } internal set { _ID = value; } }
    public DateTime ScheduledTime { get; set; }
    public Configurations Configuration { get; set; }
    public Scheduler Scheduler { get; set; }
    public List<Task> Tasks { get; set; }

    #region IDisposable Support
    private bool disposedValue = false; // To detect redundant calls

    protected virtual void Dispose(bool disposing)
    {
        if (!disposedValue)
        {
            if (disposing)
            {
                // TODO: dispose managed state (managed objects).
            }

            // TODO: free unmanaged resources (unmanaged objects) and  a finalizer below.
            // TODO: set large fields to null.

            disposedValue = true;
        }
    }

    // TODO:  a finalizer only if Dispose(bool disposing) above has code to free unmanaged resources.
    ~Job()
    {
        // Do not change this code. Put cleanup code in Dispose(bool disposing) above.
        Dispose(false);
    }

    // This code added to correctly implement the disposable pattern.
    public void Dispose()
    {
        // Do not change this code. Put cleanup code in Dispose(bool disposing) above.
        Dispose(true);
        // TODO: uncomment the following line if the finalizer is overridden above.
        // GC.SuppressFinalize(this);
    }
    #endregion
}

这是Task声明,如果使用此接口,我会派生更多类型

public interface Task : IDisposable
{
    //public Task() { }
    string Title { get; set; }
    bool ExecuteTask();/* { return true; }*/
    bool? NotifyOnSuccess { get; set; }
    bool? NotifyOnFailure { get; set; }


}

这是从“任务”界面派生的类型的列表:

Task Types Derived from Interface Task

以下是从“任务类型”派生的一种类的示例:

public class Email : Task
{     
    public string Recipients { get; set; }
    public string Subject { get; set; }
    public string Body { get; set; }
    public string AttachmentPath { get; set; }
    public string From { get; set; }
    public  string Title { get; set; }
    public  bool? NotifyOnSuccess { get; set; }
    public  bool? NotifyOnFailure { get; set; }

    public void Dispose()
    {
        throw new NotImplementedException();
    }

    public  bool ExecuteTask()
    {
        return true;
    }

}

这是JSON输出。

注意:Scheduler类中的字段的值已分配,但我不知道为什么在输出中显示它们的值不同。

{
  "ID": "9ebcec6c-8adb-421d-882b-da51b106f14b",
  "ScheduledTime": new Date(
    1536674974648
  ),
  "Configuration": {
    "CommunicationLanguage": 0,
    "NotifyBy": 0,
    "RelayServer": "Relay Server",
    "Port": "Port",
    "Username": "Username",
    "Passwoprd": "Password",
    "FromAddress": "From address",
    "Recipients": "",
    "StoredProcedureName": "Stored procedure name",
    "ConnectionString": "Connection String",
    "ProfileName": "Profile name"
  },
  "Scheduler": {
    "ScheduleType": 0,
    "Occurs": 0,
    "RecursEvery": 1,
    "OccursOnceAt": "17:04:47.3262587",
    "DurationStartDate": new Date(
      1536613487322
    ),
    "DurationEndDate": new Date(
      1536613487318
    ),
    "dayOrderInMonth": 0,
    "DayName": 0
  },
  "Tasks": [
    {
      "$type": "qConnector_v2._0.Model.WebApi, qConnector v2.0",
      "BaseURL": "d",
      "Route": "d",
      "Params": "d",
      "Header": "d",
      "NotifyOnSuccess": true,
      "NotifyOnFailure": false,
      "Title": "api Test"
    },
    {
      "$type": "qConnector_v2._0.Model.WebService, qConnector v2.0",
      "Title": "ws test",
      "Url": "",
      "Name": "",
      "Method": "",
      "NotifyOnSuccess": true,
      "NotifyOnFailure": false
    },
    {
      "$type": "qConnector_v2._0.Model.Odbc, qConnector v2.0",
      "Title": "odbc",
      "DataSource": "Data source",
      "Database": "Database",
      "User": "Username",
      "Password": "Password",
      "Type": "Odbc Type",
      "Query": "QUERY",
      "DestinationDataSource": "Data source",
      "DestinationDatabase": "Database",
      "DestinationUsername": "Username",
      "DestinationPassword": "Password",
      "DestinationStoreProcedure": "Stored Procedure Name",
      "DestinationLocation": "UNC Destination",
      "NotifyOnSuccess": false,
      "NotifyOnFailure": true
    },
    {
      "$type": "qConnector_v2._0.Model.FTP, qConnector v2.0",
      "Title": "ftp",
      "LocalPath": "Local path",
      "Server": "Server Path",
      "Password": "Password",
      "Username": "Username",
      "Download": false,
      "Upload": true,
      "NotifyOnSuccess": true,
      "NotifyOnFailure": false
    },
    {
      "$type": "qConnector_v2._0.Model.Excel, qConnector v2.0",
      "Title": "excel",
      "Location": "UNC path to load Excel file",
      "DataSource": "Data source",
      "Database": "Database",
      "Username": "Username",
      "Password": "Password",
      "StoredProcedure": "Stored Procedure Name",
      "SheetName": "Sheet Name (optional)",
      "NotifyOnSuccess": true,
      "NotifyOnFailure": true
    },
    {
      "$type": "qConnector_v2._0.Model.UNC, qConnector v2.0",
      "Title": "unc test",
      "Type": 2,
      "Source": "Source Path",
      "Destination": "Destination Path",
      "NotifyOnSuccess": true,
      "NotifyOnFailure": true
    },
    {
      "$type": "qConnector_v2._0.Model.Email, qConnector v2.0",
      "Recipients": "recipient email,recipient email,recipient email",
      "Subject": "Subject",
      "Body": "BODY",
      "AttachmentPath": "Attachment Path",
      "From": "From",
      "Title": "email test",
      "NotifyOnSuccess": false,
      "NotifyOnFailure": true
    },
    {
      "$type": "qConnector_v2._0.Model.StoredProcedure, qConnector v2.0",
      "Title": "TITLE",
      "NotifyOnSuccess": true,
      "NotifyOnFailure": false,
      "DataSource": "Data source",
      "Database": "Database",
      "Username": "Username",
      "Password": "Password",
      "SqlCommand": "SQL Command - SP Call"
    }
  ]
}

这是Scheduler类的结构:

public class Scheduler : IDisposable
{
    public enum SchedulerType { Recurring, OneTime };
    public enum SchedulerFrequency { Daily, Weekly, Monthly };
    public enum DayOrderInMonth { First, Second, Third, Fourth, Last };
    public DateTime? Scheduled { get; set; }
    public SchedulerType ScheduleType { get; set; }
    public DateTime? OneTimeDate { get; set; }
    public TimeSpan? OneTimeTime { get; set; }
    public SchedulerFrequency Occurs { get; set; }
    public int? RecursEvery { get; set; }
    public TimeSpan? OccursOnceAt { get; set; }
    public DateTime? DurationStartDate { get; set; }
    public DateTime? DurationEndDate { get; set; }
    public string WeekDays { get; set; }
    public List<DayOfWeek> WeekDaysList { get; set; }
    public DayOrderInMonth dayOrderInMonth { get; set; }
    public DayOfWeek DayName { get; set; }
    public int? EveryNumberOfMonth { get; set; }
    public int? DayNumberEachMonth { get; set; }
    public int? TheOptionRecursEveryMonth { get; set; }
    public int? RecursEveryWeek { get; set; }
}

1 个答案:

答案 0 :(得分:0)

解决方案非常简单, 而不是使用

Newtonsoft.Json.JsonSerializer serializer = new Newtonsoft.Json.JsonSerializer();
serializer.Converters.Add(new Newtonsoft.Json.Converters.JavaScriptDateTimeConverter());
serializer.NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore;
serializer.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.Auto;
serializer.Formatting = Newtonsoft.Json.Formatting.Indented;

using (StreamWriter sw = new StreamWriter(@"C:\Temp\res.json"))
using (Newtonsoft.Json.JsonWriter writer = new Newtonsoft.Json.JsonTextWriter(sw))
{
    serializer.Serialize(writer, _Job, typeof(Model.Job));
}

我只需要删除以下行:

  

serializer.Converters.Add(新   Newtonsoft.Json.Converters.JavaScriptDateTimeConverter());

因此代码将是:

Newtonsoft.Json.JsonSerializer serializer = new Newtonsoft.Json.JsonSerializer();
serializer.NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore;
serializer.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.Auto;
serializer.Formatting = Newtonsoft.Json.Formatting.Indented;

using (StreamWriter sw = new StreamWriter(@"C:\Temp\res.json"))
using (Newtonsoft.Json.JsonWriter writer = new Newtonsoft.Json.JsonTextWriter(sw))
{
    serializer.Serialize(writer, _Job, typeof(Model.Job));
}