【问题标题】:Configure Json.net to URL encode certain field配置 Json.net 对特定字段进行 URL 编码
【发布时间】:2017-09-12 15:32:07
【问题描述】:

假设我有一个这样的模型类:

public class MyModelClass
{
    [JsonProperty("first_field"]
    public string FirstField { get; set; }

    [JsonProperty("second_field"]
    public string SecondField { get; set; }

    public MyModelClass(string first, string second)
    {
        FirstField = first;
        SecondField = second;
    }
}

假设我有一个这种类型的实例:

var myObject = new MyModelClass("blablabla", "<>@%#^^@!%");

当我使用 Json.net 将此对象转换为 Json 字符串时,我得到如下信息:

{ 'first_field': 'blablabla', 'second_field': '<>@%#^^@!%' }

有没有办法配置 Json.net 以便“SecondField”的内容是 URL 编码的?我需要编写自己的自定义转换器还是有更简单的方法?

【问题讨论】:

  • 你不能使用属性使“真实”属性对序列化程序不可见,并有一个仅用于序列化的替代属性吗?

标签: c# json.net


【解决方案1】:

如果它只是一个类中的一个字段,您可以简单地在您的类中添加一个只读属性来处理编码,并对其进行注释,使其在序列化期间取代原始属性:

    public class MyModelClass
    {
        ...

        [JsonIgnore]
        public string SecondField { get; set; }

        [JsonProperty("second_field")]
        private string UrlEncodedSecondField
        {
            get { return System.Web.HttpUtility.UrlEncode(SecondField); }
        }

        ...
    }

演示小提琴:https://dotnetfiddle.net/MkVBVH


如果您需要跨多个类的多个字段,您可以使用类似于Selectively escape HTML in strings during deserialization 中的解决方案,并进行一些小的更改:

  1. 创建一个自定义 UrlEncode 属性并让解析器查找具有该属性的属性,而不是缺少 AllowHtml 属性。
  2. HtmlEncodingValueProvider 更改为UrlEncodingValueProvider 并让它在GetValue 而不是SetValue 中应用编码(这样它就对序列化而不是反序列化进行编码)。

生成的代码如下所示:

public class UrlEncodeAttribute : Attribute { }

public class CustomResolver : DefaultContractResolver
{
    protected override IList<JsonProperty> CreateProperties(Type type, MemberSerialization memberSerialization)
    {
        IList<JsonProperty> props = base.CreateProperties(type, memberSerialization);

        // Find all string properties that have a [UrlEncode] attribute applied
        // and attach an UrlEncodingValueProvider instance to them
        foreach (JsonProperty prop in props.Where(p => p.PropertyType == typeof(string)))
        {
            PropertyInfo pi = type.GetProperty(prop.UnderlyingName);
            if (pi != null && pi.GetCustomAttribute(typeof(UrlEncodeAttribute), true) != null)
            {
                prop.ValueProvider = new UrlEncodingValueProvider(pi);
            }
        }

        return props;
    }

    protected class UrlEncodingValueProvider : IValueProvider
    {
        PropertyInfo targetProperty;

        public UrlEncodingValueProvider(PropertyInfo targetProperty)
        {
            this.targetProperty = targetProperty;
        }

        // SetValue gets called by Json.Net during deserialization.
        // The value parameter has the original value read from the JSON;
        // target is the object on which to set the value.
        public void SetValue(object target, object value)
        {
            targetProperty.SetValue(target, (string)value);
        }

        // GetValue is called by Json.Net during serialization.
        // The target parameter has the object from which to read the string;
        // the return value is the string that gets written to the JSON
        public object GetValue(object target)
        {
            string value = (string)targetProperty.GetValue(target);
            return System.Web.HttpUtility.UrlEncode(value);
        }
    }
}

要使用自定义解析器,首先使用新的[UrlEncode] 属性装饰您想要进行 URL 编码的任何属性:

public class MyModelClass
{
    [JsonProperty("first_field")]
    public string FirstField { get; set; }

    [UrlEncode]
    [JsonProperty("second_field")]
    public string SecondField { get; set; }

    ...
}

然后,像这样序列化您的模型:

var myObject = new MyModelClass("blablabla", "<>@%#^^@!%");

var settings = new JsonSerializerSettings
{
    ContractResolver = new CustomResolver(),
    Formatting = Formatting.Indented
};

string json = JsonConvert.SerializeObject(myObject, settings);

演示小提琴:https://dotnetfiddle.net/iOOzFr

【讨论】:

  • 这看起来是一个很好的解决方案,但似乎令人难以置信,我们需要 3 个类来实现这个看似简单的目标。
【解决方案2】:

@brian-rogers 提供的答案非常好,但我想提出一个我认为更简单的替代方案:

public class UrlEncodingConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof(string);
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        if (value == null) return;
        var encodedValue = System.Web.HttpUtility.UrlEncode((string)value);
        writer.WriteValue(encodedValue);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        if (reader.Value == null) return null;
        var decodedValue = System.Web.HttpUtility.UrlDecode(reader.Value.ToString());
        return decodedValue;
    }
}

可以这样使用:

public class MyModelClass
{
    [JsonProperty("first_field")]
    public string FirstField { get; set; }

    [JsonProperty("second_field")]
    [JsonConverter(typeof(UrlEncodingConverter))]
    public string SecondField { get; set; }
}

【讨论】:

    【解决方案3】:

    试试这个:

    var myObject = new MyModelClass("blablabla", WebUtility.UrlEncode("<>@%#^^@!%"));
    

    您需要将System.Net 添加到您的使用指令中

    【讨论】:

    • 不幸的是,我需要“myObject”来包含未编码的值。我只需要在转换为 Json 时对值进行编码
    • 然后要么在需要时使用WebUtility.UrlDecode,要么仅在调用函数将其转换为 JSON 时使用 UrlEncode 方法
    猜你喜欢
    • 1970-01-01
    • 2021-11-17
    • 1970-01-01
    • 1970-01-01
    • 2015-11-29
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多