ASP.NET Core Post Json 请求参数模型类型转换

内容纲要

在后端中,当有分布式需求时,我们常常使用 64位 的数字类型表示字段类型,但是前端并不支持超过 16 位长度的数值类型,所以后端长度较大的 long、ulong 数值传到后端,其准确度就已经丢失。

解决方法就是将 ulong、long 转为字符串,传递给后端。

    public class TentantQueryDto
    {
        public ulong Id { get; set; }
    }

这是一个后端模型, Id 为 64 位数值,我们要做到前端传递 string,自动转 ulong;后端传递 ulong,自动返回 string 给前端。

首先定义两个转换器:

    public class ULongToStringConverter : JsonConverter<ulong>
    {
        public override ulong Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
        {
            string stringValue = reader.GetString();
            if (string.IsNullOrWhiteSpace(stringValue)) return 0;

            if (ulong.TryParse(stringValue, out var value)) return value;
            return 0;
        }

        public override void Write(Utf8JsonWriter writer, ulong value, JsonSerializerOptions options)
        {
            writer.WriteStringValue(value.ToString());
        }
    }
    public class StringToULongConverter : JsonConverter<string>
    {
        public override string Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
        {
            return reader.GetString();
        }

        public override void Write(Utf8JsonWriter writer, string value, JsonSerializerOptions options)
        {
            ulong result = 0;
            if (string.IsNullOrWhiteSpace(value)) result = 0;
            if (ulong.TryParse(value, out var number)) result = number;

            writer.WriteNumberValue(result);
        }
    }

这两个转换器没有什么特别的地方,只是将数值和string互转。

然后定义特性,这个特性告诉我们应该使用什么字段、怎么在前后端传递 Json 时,自动转换参数类型:

    /// <summary>
    /// string 和 ulong 转换器
    /// </summary>
    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Enum | AttributeTargets.Property, AllowMultiple = false)]
    public class ULongConverterAttribute : JsonConverterAttribute
    {
        public override JsonConverter CreateConverter(Type typeToConvert)
        {
            if (typeToConvert == typeof(ulong))
                return new ULongToStringConverter();
            if (typeToConvert == typeof(string))
                return new StringToULongConverter();

            return base.CreateConverter(typeToConvert);
        }
    }

在要被转换的字段上使用该特性:

    public class TentantQueryDto
    {
        [ULongConverter]
        public ulong Id { get; set; }
    }

这样,我们无需另外加上转换代码,在前后端传递数据时, ASP.NET Core 调用转换器自动转换。

点赞

发表评论

邮箱地址不会被公开。 必填项已用*标注

You must enable javascript to see captcha here!