带有私有构造函数的类型的JsonSerializationException

时间:2013-03-10 21:04:22

标签: serialization json.net ravendb nlog

我正在将NLog日志记录语句保存到我的RavenDb数据库中。 LogEventInfo类(表示日志语句)具有属性LogLevel,具有私有构造函数。 LogLevel(Info,Warn等)的实例是通过调用私有构造函数的静态,只读属性创建的。

问题是我希望从数据库中读取消息并查询它们会导致Json.Net序列化错误:

  

无法找到用于NLog.LogLevel类型的构造函数。一个类应该有一个默认的构造函数,一个带参数的构造函数或一个用JsonConstructor属性标记的构造函数。路径'Level.Name'。

如何解决错误?可以在这里创建某种Raven索引吗?

2 个答案:

答案 0 :(得分:1)

我认为最简单的方法是创建一个自定义类来保存所有日志信息。

答案 1 :(得分:1)

它像我一样与我合作

create a CustomCreationConverter并将其与对象的反序列化一起使用

public class EventInfoConverter : CustomCreationConverter<LogEventInfo>
    {
        private string _level { get; set; }
        public EventInfoConverter(string s)
        {
            JToken eventinfo = JObject.Parse(s);
            var childs = eventinfo.Children();
            foreach (var item in childs)
            {
                if (((Newtonsoft.Json.Linq.JProperty)item).Name == "Level")
                {
                    var m = ((Newtonsoft.Json.Linq.JProperty)item).Value.Children();
                    foreach (var item1 in m)
                    {
                        _level = ((Newtonsoft.Json.Linq.JProperty)item1).Value.ToString();
                        break;
                    }

                    break;
                }
            }
        }

        public override LogEventInfo Create(Type objectType)
        {
            LogEventInfo eventInfo = new LogEventInfo();
            switch (_level)
            {
                case "Info":
                         eventInfo = new LogEventInfo(LogLevel.Info, "", "");
                    break;
                case "Debug":
                    eventInfo = new LogEventInfo(LogLevel.Debug, "", "");
                    break;
                case "Error":
                    eventInfo = new LogEventInfo(LogLevel.Error, "", "");
                    break;
                case "Warn":
                    eventInfo = new LogEventInfo(LogLevel.Warn, "", "");
                    break;
                default:
                    break;
            }
            return eventInfo;
        }
    }
  • 在反序列化中:

    NLog.Logger _logger = NLog.LogManager.GetCurrentClassLogger();
    _logger.Log(Newtonsoft.Json.JsonConvert.DeserializeObject<LogEventInfo>(eventInfo, new EventInfoConverter(eventInfo)));