Type Conversion
文章著作权归作者所有。转载请联系作者,并在文中注明出处,给出原文链接。
本文原更新于作者的github博客,这里给出链接。
什么是转换
转换是接受一个类型的值并使用它作为另一个类型的等价值的过程。转换后值等价,类型为目标类型。
转换的分类
一般转换
从转换的方式来看,可以分为显式转换和隐式转换。显示和隐式最主要的区分点在于,这种转换是否让这个“值”的描述更加“精确”,如果是,那么则是显示转换,否则为隐式转换。例如:玫瑰是植物。如果我们需要更精确地指出玫瑰是一种花,那么我们就需要显式地强调“玫瑰是花的一种”。反过来,玫瑰是花,而花是植物是已知的,这时候我们便不需要特别说明玫瑰是植物了,因为此时它已经包含了这一信息。
一般来说,在不丢失数据的情况下,语言会自动进行预定义对象类型的转换,这种转换称为预定义的隐式转换。对于数字类型雷说,例子即为把int类型的数字10转换为double类型;相对地,在可能发生数据丢失的场合,语言并不会为我们自动进行转换,这时,我们需要做显式转换。
// 显式转换例子
ushort a = 100;
ushort b = 600;
byte c = (byte)a;
byte d = (byte)b;
ushort类型的表示范围是02^16^-1,而`byte`类型的表示范围只有028-1。此时则需要进行显式转换。在上面的例子中,a可以被安全地转换为byte类型的c,但是对于d来说,由于b的值超过了byte的表示范围,在转换过程中则会进行高位截断,也就是说,ushort超出byte的八个高位数值会被忽略,转换后,d的值为600 % 128 = 88。
对于引用类型,由于所有的类都继承了Object,所以都可以被隐式转换为Object类型,类似地,类引用也可以转换为继承链中的任意类引用以及它实现的所有接口引用。引用的显式转换则发生在从基类到派生类的过程中,前提是这个转换不会编译错误,或者说抛出异常。
装箱、拆箱
基于面向对象,还有一种特殊的转换,装箱和拆箱转换。
在C#中,所有的类型都派生自Object类型,包括值类型。值类型是高效轻量的类型,他们在堆上存在时并不会包括它所属的对象组件。当我们需要使用这个对象时,我们就会根据这个值组装为一个完整的对象,也就是装箱。装箱也是一种隐式转换,它接受值类型,并根据这个值在堆上创建一个完整对象并返回引用。装箱操作返回的并不是原值的引用,而是一个根据原值组装的对象的副本。
int number = 0;
object obj = number; // 这里隐式地进行了装箱操作
number = 1;
obj = 2; // 此时number的值仍是1;obj中的值只是一个副本,在它身上做的修改不会反馈到原对象上
对应地,拆箱则是把装箱后的对象转换回值类型的过程。拆箱是显式转换。
int number = 0;
object obj = number; // 先把number装箱
int number2 = (int)obj; // 显式进行拆箱
// 此时,number、obj、number2对应的值均为0
拆箱的操作结果只能是原始类型,否则会抛出异常。
自定义转换
从转换的定义者来看,还可以分为预定义转换和自定义转换。这个比较容易理解,语言本身帮我们进行的称为预定义转换,其余则是自定义转换。
// 一个例子
public class A {
public int num;
public A(int _num) {
num = _num;
}
// 在C#中,转换是静态方法
// implicit/explicit分别代表隐/显式
public static implicit operator int(A a) {
return a.num;
}
public static implicit operator A(int _num) {
return new A(_num);
}
}
自定义的类型转换有以下规则:
- 不能重定义标准的转换
- 转换必须以类或结构为单位
- 转换源和目标类型必须不同
- 对于同源同目标的转换,不可以同时声明其显隐式
- 转换操作符必须是源或目标的成员
- 源类型和目标类型不能互为继承关系(标准一的细化)
- 源类型和目标类型均不能为接口或者
Object类型(标准一的细化)
转换的保障
is运算符
通过前面的学习,我们知道了有些转换请求时不成功的,并在运行时抛出异常,这是我们不希望看到的。is运算符可以帮我们确认转换的成功与否。
if (ExpressionA is ExpressionB) {
// Do type conversion
}
is运算符返回值为bool,当ExpressionA的返回值可以被成功转换为ExpressionB的返回值时返回true,但是is的检查只适用于引用转换、装箱、拆箱,并不支持自定义转换。
as运算符
as运算符则是强化版强制转换,并且不会抛出异常,而是在转换失败时返回null。
destinationTypeInstance = sourceTypeInstance as DestinationType;
if (destinationTypeInstance != null) {
// Do something
}
注意到,as运算符返回的是引用类型,也就是说,我们可以将其作为赋值运算的右值,而DestinationType则必须为引用类型。
Type Conversion的更多相关文章
- JavaScript Type Conversion
Data Types 5 Data Types string, number, boolean, object, function 3 Object Types object, array, date ...
- error: expected constructor, destructor, or type conversion before '.' token
今天写代码是遇到这样一个问题error: expected constructor, destructor, or type conversion before '.' token:立马网上查,原来是 ...
- 【错误】expected constructor, destructor, or type conversion before '.' token - 第八个游侠的日志 - 网易博客
[错误]expected constructor, destructor, or type conversion before '.' token - 第八个游侠的日志 - 网易博客 [错误]expe ...
- Spring Framework 官方文档学习(四)之Validation、Data Binding、Type Conversion(一)
题外话:本篇是对之前那篇的重排版.并拆分成两篇,免得没了看的兴趣. 前言 在Spring Framework官方文档中,这三者是放到一起讲的,但没有解释为什么放到一起.大概是默认了读者都是有相关经验的 ...
- Spring Framework 官方文档学习(四)之Validation、Data Binding、Type Conversion(二)
接前一篇 Spring Framework 官方文档学习(四)之Validation.Data Binding.Type Conversion(一) 本篇主要内容:Spring Type Conver ...
- delphi 10.1 berlin datasnap提交clientdataset.delta报:invalid variant type conversion(类型转换错误)问题的解决
delphi 10.1 berlin datasnap提交clientdataset.delta报:invalid variant type conversion(类型转换错误)问题的解决,需要打这个 ...
- java 反射 报错:Attempt to get java.lang.Integer field "..." with illegal data type conversion to int
类: Integer id; 反射时: Field f = User.class.getDeclaredField("id"); f.setAccessible(true); in ...
- Spring Framework 官方文档学习(四)之Validation、Data Binding、Type Conversion
本篇太乱,请移步: Spring Framework 官方文档学习(四)之Validation.Data Binding.Type Conversion(一) 写了删删了写,反复几次,对自己的描述很不 ...
- [C++] Type Conversion(类型转换)
Type Conversion(类型转换) Two kinds of type conversion explict type conversion(显式类型转换) impict type conve ...
随机推荐
- 严重: StandardWrapper.Throwable org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'goodsController' defined in file [D:\eclipse\eclipse-space\pinyougou_parent\pinyou
由于错误太宽,没法截取完整的,所以不怎么连贯,但是不影响错误的解决. 这个错误是因为service无法自动注入.显示嵌套状态异常. 我就查看了一下我的坐标和配置文件,配置文件的路径和访问地址都是正确的 ...
- 洛谷P3385 【模板】负环
题目描述 暴力枚举/SPFA/Bellman-ford/奇怪的贪心/超神搜索 寻找一个从顶点1所能到达的负环,负环定义为:一个边权之和为负的环. 输入输出格式 输入格式: 第一行一个正整数T表示数据组 ...
- 配置 Tomcat 以服务方式自动运行
接手一台用 Tomcat 跑着 Java Web 应用的 Linux CentOS 服务器,通过镜像的方式更换服务器后站点无法访问,发现是因为 Tomcat 不能在服务器启动后自动启动,于是基于 in ...
- AVL树的Java实现
AVL树:平衡的二叉搜索树,其子树也是AVL树. 以下是我实现AVL树的源码(使用了泛型): import java.util.Comparator; public class AVLTree< ...
- Vue.js中使用select选择下拉框
在Vue.js中使用select选择下拉框有两种方法: 第一种: Add.html: <select v-model="sysNotice.noticeType" id=&q ...
- Rabbit MQ 消息确认和持久化机制
一:确认种类 RabbitMQ的消息确认有两种.一种是消息发送确认,用来确认生产者将消息发送给交换器,交换器传递给队列的过程中消息是否成功投递.发送确认分为两步,一是确认是否到达交换器,二是确认是否到 ...
- linux获取网络信息函数
获取IP地址 int sys_getIP(char *ip_addr) { ] = {"ifconfig eth0 | grep inet | cut -d: -f2 | cut -d' ' ...
- 解决nohup中不写入日志的问题
(一)问题描述: nohup 你的程序命令 如: nohup python manage.py runserver 0.0.0.0:6789 (此shell窗口1不要关,另外开一个shell窗口2 ...
- JS 变量和函数提升 全局变量和局部变量
变量提升 1. var a = 10; function test() { a = 100; console.log(a); console.log(this.a); var a; console.l ...
- [LeetCode] 120. Triangle _Medium tag: Dynamic Programming
Given a triangle, find the minimum path sum from top to bottom. Each step you may move to adjacent n ...