C#中的类型转换-自定义隐式转换和显式转换
前言
有时我们会遇到这么一种情况:在json数据里,数组里的数据类型不一致,导致我们不能直接反序列化为目标类型。最终我们只能反序列化为JObject类型,然后通过字符串取值的方式来取出数据。
下面介绍一种新方式:通过自定义隐式转换,把不一样的数据类型反序列化为一样的数据类型。
基础知识
类型转换有2种:隐式转换和显式转换。但是,不管是隐式转换,还是显式转换,都是生成了一个新对象返回的。改变新对象的属性,不会影响老对象!(dynamic对象
除外,详情搜索dynamic动态类型
。)
自定义隐式/显式转换的方法需要用到几个关键字:implicit
(隐式转换)、explicit
(显式转换)、operator(操作符)
。更多的注意点见下:
- 方法必須是static
- 使用
implicit
或explicit
- 搭配
operator
(此也是c#關鍵字,可在類別或結構宣告內多載內建運算子或提供使用者定義的轉換) - 返回值为要转换为的目标类型,但不要在方法上声明,方法名为目标类型。注意:返回值不一定是本类类型。本类型和其他类型之间可以互相转换,只要定义转换方法就行。
- 参数为原始类型,方法名为目标类型
- 类A到类B的类型转换定义不能在类C中进行(即2个类的转换不能在第3个类中定义),否则会报错:
用户定义的转换必须是转换成封闭类型,或者从封闭类型转换
。具体查看后面的用户定义的转换必须是转换成封闭类型,或者从封闭类型转换 - 不能被
virtual
/override
修饰(不能“覆盖”运算符,因为它们是静态的。)Overriding implicit operators in C#
示例代码
//================定义类型和方法================
class Robot
{
public int Id { get; set; }
public string Name { get; set; }
public Robot(int id, string name)
{
Id = id;
Name = name;
}
#region 其他类型->本类
//隐式转换
public static implicit operator Robot(string name)
{
return new Robot(101, name);
}
//显式转换
public static explicit operator Robot(int id)
{
return new Robot(id, "miku");
}
#endregion
#region 本类->其他类型
//隐式转换
public static implicit operator string(Robot robot)
{
return robot.Name;
}
//显式转换
public static explicit operator int(Robot robot)
{
return robot.Id;
}
#endregion
}
//================测试代码================
#region 其他类型->本类
string gumiStr = "gumi";
Robot gumi001 = gumiStr; //隐式转换
Console.WriteLine("隐式转换:gumi001 : {0}", JsonConvert.SerializeObject(gumi001));
int lukaId = 1004;
Robot luka001 = (Robot)lukaId; //显式转换
Console.WriteLine("显式转换:luka001 : {0}", JsonConvert.SerializeObject(luka001));
#endregion
#region 其他类型->本类
Robot miku001 = new Robot(1001, "miku10001");
//隐式转换
string mikuName = miku001;
//显式转换
int mikuId = (int)miku001;
Console.WriteLine("隐式转换:miku001 Name: {0}", mikuName);
Console.WriteLine("显式转换:miku001 Id: {0}", mikuId);
#endregion
输出结果如下:
隐式转换:gumi001 : {"Id":101,"Name":"gumi"}
显式转换:luka001 : {"Id":1004,"Name":"miku"}
隐式转换:miku001 Name: miku10001
显式转换:miku001 Id: 1001
实际应用
问题
[1,[[2,2],[2,2],[2,2],[2,2]]]
这样一个字符串,如何可以反序列化成一个对象?(如何定义这个类?)
答案
using System;
using System.Linq;
using System.Collections.Generic;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
public class Program
{
public static void Main()
{
var json = "[1,[[2,2],[2,2],[2,2],[2,2]]]";
var root = JsonConvert.DeserializeObject<Root>(json);
foreach(var ele in root)
{
if(ele.SingleValue.HasValue)
{//有值,原始数据为 1
Console.WriteLine(ele.SingleValue.Value);
}else
{//原始数据为 二维数组
Console.WriteLine(string.Join(" ",ele.Select(x=>string.Join(",",x))));
}
}
Console.WriteLine(JsonConvert.SerializeObject(root));
}
}
class Root : List<Element> { }
[JsonConverter(typeof(CConverter))]
class Element : List<List<long>>
{
//该属性,存放 1 。后续可以通过判断该属性是否有值来得知原始数据的情况
public long? SingleValue { get; set; }
//遇到 1 ,隐式转换为 该类型,其中 1 被存放到SingleValue属性
public static implicit operator Element(long d)
{
return new Element { SingleValue = d };
}
}
public class CConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return (objectType == typeof(Element));
}
public override bool CanRead { get { return false; } }
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
throw new NotImplementedException();
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
var ele = value as Element;
var token = ele.SingleValue.HasValue ? JToken.FromObject(ele.SingleValue.Value) : JToken.FromObject(ele.ToList());
token.WriteTo(writer);
}
public override bool CanWrite { get { return true; } }
}
报错
用户定义的转换必须是转换成封闭类型,或者从封闭类型转换
这个错误,与封闭类型无关。
是因为有这个限制:类A到类B的类型转换定义不能在类C中进行(即2个类的转换不能在第3个类中定义)
所以对于目标类型是集合类List<T>
,我们无法直接定义到它的转换。不过,有2个迂回的方法:
- 创建个类继承自集合类
List<T>
,定义到这个子类的转换。上面实际应用中的代码就是这样做的:class Element : List<List<long>>
- 创建
T1
到T2
的自定义转换,使用时逐个转换:list.Select(p=>(B)p).ToList()
。
参考
- 隐式转换:用户定义的转换必须是转换成封闭类型,或者从封闭类型转换:https://blog.csdn.net/kamui_shiron/article/details/8807142
其他
应用和设计
在定義類別時,如果有需要,就可以使用這兩個關鍵字來提供類別一些額外的功能
但在使用時也必須考慮設計上是否合理
例如當兩類別有相關性時是否該提取出父类或是接口來使用,而不是為了方便做了一堆轉換,導致程式撰寫與維護上的困難。
读音
- 隐式转换:
implicit
[ɪmˈplɪsɪt] adj.不言明[含蓄]的; 无疑问的,绝对的; 成为一部份的; 内含的; - 显式转换:
explicit
[ɪkˈsplɪsɪt] adj.明确的,清楚的; 直言的; 详述的; 不隐瞒的;
参考
- 【问】这样一个字符串如何反序列化:http://www.newsmth.net/nForum/#!article/DotNET/69817
- 型別轉換關鍵字explicit與implicit的用法:https://dotblogs.com.tw/lastsecret/2011/11/14/57875
- c#关键词implicit和explicit:https://blog.csdn.net/Joyhen/article/details/40110391
C#中的类型转换-自定义隐式转换和显式转换的更多相关文章
- Java基础学习-类型转换之隐式转换
+是一个运算符,我们应该能够看懂,做数据的加法. boolean类型不能转换为其他的数据类型. 默认转换: byte,short,char--int--float--double by ...
- scala自定义隐式转换
Scala自定义隐式转换 一.编写隐式转换类 /** * Author Mr. Guo * Create 2019/4/20 - 17:40 */ object StringImprovments { ...
- 使用自定义隐式转换快速创建失败Result
系统要求方法都返回 Result 结果,通常我们会如此定义一个 Result 1 public class Result<T> 2 { 3 public virtual int Code ...
- VB.Net隐式转换和显式转换的方法(转)
VB.Net隐式转换和显式转换的方法(转) “隐式转换”不需要源代码中的任何特殊语法.在下面的示例中,在将 k 的值赋给 q 之前,Visual Basic 将该值隐式转换成单精度浮点值. Dim ...
- 自己定义隐式转换和显式转换c#简单样例
自己定义隐式转换和显式转换c#简单样例 (出自朱朱家园http://blog.csdn.net/zhgl7688) 样例:对用户user中,usernamefirst name和last name进行 ...
- F#中的自定义隐式转换
我们知道隐式变换在可控情况下会使代码变得简洁.熟悉C#的都知道C#中可以自定义隐式变换,例如 public class A { private int data; public static impl ...
- ahjesus自定义隐式转换和显示转换
implicit 关键字用于声明隐式的用户定义类型转换运算符. 如果可以确保转换过程不会造成数据丢失,则可使用该关键字在用户定义类型和其他类型之间进行隐式转换. 参考戳此 explicit ...
- C++隐式转换与显式转换
普通类型的转换顺序:隐式把char——>int和从short——>double.转换可能会导致数据的丢失. 自定义类型:有两种函数可以进行隐式转换,单参数构造函数 和 隐式类型转换符. ...
- c# 强制转换, 隐式转换, 显式转换
using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...
随机推荐
- 【ybtoj】二分算法例题
[基础算法]第三章 二分算法 例一 数列分段 题目描述 对于给定的一个长度为N的正整数数列A,现在将其分成M段,并要求每段连续,且每段和的最大值最小. 输入格式 第1行包含两个正整数N,M. 第2行包 ...
- C#字符串Base64编解码
C#字符串Base64编解码 首先讲一下什么是Base64编码所谓Base64就是一种基于64个可打印字符来表示二进制数据的方法.Base64编码是从二进制到字符的过程,常用于在网络上传输不可见字符( ...
- Redis小秘密
Redis小秘密 临渊羡鱼,不如退而织网. 一.Redis基本数据类型 想必很多人都能脱口而出String.List.Hash.Sorted Set和Set五种基本数据类型. 以及五大基本数据类型简要 ...
- 换行符号(\r\n)的历史
文章来源:https://cloud.tencent.com/developer/article/1730918 \r\n与\n是有区别的. 如果要通用的则是\r\n,因为有些编辑器它不认\n &qu ...
- Spring源码之六-onRefresh()方法
Spring源码之六-onRefresh()方法 大家好,我是程序员田同学. 今天带大家解读Spirng源码之六的onRefresh()方法,这是refresh()的其中的一个方法,看似是一个空方法, ...
- Java常用--反射
反射的意义 你可能说,平时都是业务的增删查改基本用不到反射.但是如果你学会用反射了,可以减少重复代码,非常的好用. 反射是Java语言的一大特性,允许动态的修改程序行为. 代码说反射 1.反射的三个入 ...
- 白话OAuth2.0
一.OAuth是什么 OAuth简写:Open Authorization(开放授权).百度百科上是这么说的:为用户资源提供了一个安全,开放而又简易的标准.解决了密码的反模式问题(第三方应用不需要知 ...
- linux 提权防护
linux 用户提权 利用系统漏洞或者程序等方面的缺陷使一个低权限用户,获得他们原本不具有的权限或者得到root权限 下面介绍几种可能得到root访问权限的方式 1, 内核开发 防护:及时更新补丁 臭 ...
- kippo: 一款强大的SSH蜜罐工具
Kippo的安装 git clone https://github.com/desaster/kippo.git # 无法运行git命令,请先安装 yum -y install git由于我使用这个命 ...
- Vue之路由的使用
零.传统路由与SPA的区别 传统开发方式下,URL改变后,就会立刻发生请求去请求整个页面,这样可能请求加载的资源过多,可能会让页面出现白屏. 在SPA(Single Page Application) ...