首页 > 代码库 > 【转】C#的数据类型转换
【转】C#的数据类型转换
装箱/拆箱/别名、数值类型间相互转换、ASCII码和Unicode码、数值字符串和数值之间的转换、字符串和字符数组/字节数组之间的转换、各种数值 类型和字节数组之间的转换、十六进制数输出以及日期型数据的一些转换处理。
1. 装箱、拆箱、别名
int->Int32是一个装箱的过程,反之则是拆箱的过程,其他例 如:short<->Int16,long<->Int64等。装箱和拆箱的动作由编译器自动完成,不需要人工干预。为了记住这 些类型之间的关系,我们使用了“别名”。C#是全面向对象的语言,它把简单数据类型通过默认的装箱动作封装成了类。Int32、Int16、Int64等 就是相应的类名,而那些熟悉的简单易记的名称如int、short、long等,我们称之为Int32、Int16、Int64等类型的别名。除了这三种 类型之外,还有常用的“别名”有如下一些:
bool -> System.Boolean (布尔型,其值为 true 或者 false)
char -> System.Char (字符型,占有两个字节,表示 1 个 Unicode 字符)
byte -> System.Byte (字节型,占 1 字节,范围 0 ~ 255)
sbyte -> System.SByte (带符号字节型,表示 8 位整数,范围 -128 ~ 127)
ushort -> System.UInt16 (无符号短整型,表示 16 位正整数,范围 0 ~ 65,535)
uint -> System.UInt32 (无符号整型,表示 32 位正整数,范围 0 ~ 4,294,967,295)
ulong -> System.UInt64 (无符号长整型,表示 64 位正整数,范围 0 ~ 大约 10 的 20 次方)
short -> System.Int16 (短整型,表示 16 位整数,范围 -32,768 ~ 32,767)
int -> System.Int32 (整型,表示 32 位整数,范围 -2,147,483,648 到 2,147,483,647)
long -> System.Int64 (长整型,表示 64 位整数,范围大约 -10的19次方 到 10的19次方)
float -> System.Single (单精度浮点型,占 4 个字节)
double -> System.Double (双精度浮点型,占 8 个字节)
例: byte a = 1; char b = ‘a‘; short c = 1;
int d = 2; long e = 3; uint f = 4; bool g = true;
this.textBox1.Text = "";
this.textBox1.AppendText("byte -> " + a.GetType().FullName + "\n");
this.textBox1.AppendText("char -> " + b.GetType().FullName + "\n");
this.textBox1.AppendText("short -> " + c.GetType().FullName + "\n");
this.textBox1.AppendText("int -> " + d.GetType().FullName + "\n");
this.textBox1.AppendText("long -> " + e.GetType().FullName + "\n");
this.textBox1.AppendText("uint -> " + f.GetType().FullName + "\n");
this.textBox1.AppendText("bool -> " + g.GetType().FullName + "\n");
运行结果如下:
byte -> System.Byte
char -> System.Char
short -> System.Int16
int -> System.Int32
long -> System.Int64
uint -> System.UInt32
bool -> System.Boolean
2. 数值类型之间的相互转换
这里所说的数值类型包括 byte, short, int, long, fload, double 等,根据这个排列顺序,各种类型的值依次可以向后自动进行转换。例如:
byte a = 1; short b = a; int c = b;
long d = c; float e = d; double f = e;
this.textBox1.Text = "";
this.textBox1.AppendText("byte a = " + a.ToString() + "\n");
this.textBox1.AppendText("short b = " + b.ToString() + "\n");
this.textBox1.AppendText("int c = " + c.ToString() + "\n");
this.textBox1.AppendText("long d = " + d.ToString() + "\n");
this.textBox1.AppendText("float e = " + e.ToString() + "\n");
this.textBox1.AppendText("double f = " + f.ToString() + "\n");
译顺利通过,运行结果是各变量的值均为 1;当然,它们的类型不会发生变化。当赋值的顺序反过来时无法通过编译,例如:
int g = 1;
short h = g;
this.textBox1.AppendText("h = " + h.ToString() + "\n");
结果编译报错:“...Form1.cs(118): 无法将类型“int”隐式转换为“short””
这时如果坚持要进行转换,就应该使用强制类型转换,与C语言的方式相同,就是使用“(类型名)变量名”形式的语句来对数据进行强制转换。如上例修改如下:
short g = 1;
byte h = (byte) g;
this.textBox1.AppendText("h = " + h.ToString() + "\n");
编译通过,运行结果输出了 h = 1,转换成功。
但是使用强制转换中必须考虑数据类型的取值范围,如下例:
short g = 265; //265 = 255 + 10
byte h = (byte) g;
this.textBox1.AppendText("h = " + h.ToString() + "\n");
编译没有出错,运行结果却不是 h = 265,而是 h = 9(g的低字节值)。
3. 字符的ASCII码和Unicode码
C#在字符和编码之间的转换仍延用了C语言的做法——强制转换,例如:
char ch = ‘a‘; short ii = 65;
this.textBox1.Text = "";
this.textBox1.AppendText("The ASCII code of \‘" + ch + "\‘ is: " + (short) ch + "\n");
this.textBox1.AppendText("ASCII is " + ii.ToString() + ", the char is: " + (char) ii + "\n");
char cn = ‘中‘; short uc = 22478;
this.textBox1.AppendText("The Unicode of \‘" + cn + "\‘ is: " + (short) cn + "\n");
this.textBox1.AppendText("Unicode is " + uc.ToString() + ", the char is: " + (char) uc + "\n");
它的运行结果是
The ASCII code of ‘a‘ is: 97
ASCII is 65, the char is: A
The Unicode of ‘中‘ is: 20013
Unicode is 22478, the char is: 城
4. 数值字符串和数值之间的转换
所有数值型的void ToString()方法都能将数据转换为数值字符串。如123.ToSting()就将得到字符串"123"。
将数值字符串转换成数值可以利用int,float等数值类型的static Parse()静态函数。例:float f=float.Parse("3.21"); 其结果f的值为3.21F。当然,其它的数值类型也可以使用同样的方法进行转换。
5. 字符串和字符数组之间的转换
字符串类System.String的void ToCharArray()方法可以实现字符串到字符数组的转换。如下例:
字符数组转换成字符串:可以使用System.String类的构造函数
char[] tcs = {‘t‘, ‘e‘, ‘s‘, ‘t‘, ‘ ‘, ‘m‘, ‘e‘};
string tstr = new String(tcs);
this.textBox1.AppendText("tstr = \"" + tstr + "\"\n");
运行结果输入 tstr = "test me",测试说明转换成功。
System.String的[]运算符可以得到该字符串中的某个字符。如下例
char ch = tstr[3];
this.textBox1.AppendText("\"" + tstr + "\"[3] = " + ch.ToString());
正确的输出是 "test me"[3] = t,经测试,输出正确。
6. 字符串和字节数组之间的转换
字符串和字节数组之间的转换需要借助另一个类:System.Text.Encoding。该类提供了bye[] GetBytes(string)方法将字符串转换成字节数组,还提供了string GetString(byte[])方法将字节数组转换成字符串。
System.Text.Encoding类有几个默认的 Encoding:Encoding.Default(获取系统的当前 ANSI 代码页的编码)、Encoding.ASCII(获取 7 位 ASCII 字符集的编码)、Encoding.Unicode(获取采用 Little-Endian 字节顺序的 Unicode 格式的编码)、Encoding.UTF7(获取 UTF-7 格式的编码)、Encoding.UTF8(获取 UTF-8 格式的编码) 等。
Encoding.Default和Encoding.Unicode用于转换的区别:在字符串转换到字节数组的过程 中,Encoding.Default 会将每个单字节字符,如半角英文,转换成 1 个字节,而把每个双字节字符,如汉字,转换成 2 个字节。而 Encoding.Unicode 则会将它们都转换成两个字节。
例如:
string s = "C#语言";
byte[] b1 = System.Text.Encoding.Default.GetBytes(s);
byte[] b2 = System.Text.Encoding.Unicode.GetBytes(s);
string t1 = "", t2 = "";
foreach (byte b in b1) {
t1 += b.ToString("") + " ";
}
foreach (byte b in b2) {
t2 += b.ToString("") + " ";
}
this.textBox1.Text = "";
this.textBox1.AppendText("b1.Length = " + b1.Length + "\n");
this.textBox1.AppendText(t1 + "\n");
this.textBox1.AppendText("b2.Length = " + b2.Length + "\n");
this.textBox1.AppendText(t2 + "\n");
运行结果如下:
b1.Length = 6
67 35 211 239 209 212
b2.Length = 8
67 0 35 0 237 139 0 138
将字节数组转换成字符串,使用Encoding类的string GetString(byte[])或string GetString(byte[], int, int)方法,具体使用何种Encoding还是由编码决定。实例:
byte[] bs = {97, 98, 99, 100, 101, 102};
string ss = System.Text.Encoding.ASCII.GetString(bs);
this.textBox1.AppendText("The string is: " + ss + "\n");
运行结果为:The string is: abcdef
7. 各种数值类型和字节数组之间的转换
将某种数值类型的数据转换成字节数组的时候,得到的一定是相应大小的字节数组;同样,需要把字节数组转换成数值类型,也需要这个字节数组大于相应数值类型 的字节数。这种转换用到类System.BitConverter。该类提供了byte[] GetBytes(...)方法将各种数值类型转换成字节数组,也提供了ToInt32、ToInt16、ToInt64、ToUInt32、 ToSignle、ToBoolean 等方法将字节数组转换成相应的数值类型。
8. 转换成十六进制
数值类型的ToString()方法可以将数值转换成字符串,不过结果是十进制的。要转换成十六进制的话可以使用ToString(string)方法。 string类型的参数是格式说明符。十六进制的格式说明符是"x"或者"X",分别对应小写大写。如下例:
int a = 188;
this.textBox1.Text = "";
this.textBox1.AppendText("a(10) = " + a.ToString() + "\n");
this.textBox1.AppendText("a(16) = " + a.ToString("x") + "\n");
this.textBox1.AppendText("a(16) = " + a.ToString("X") + "\n");
运行结果如下:
a(10) = 188
a(16) = bc
a(16) = BC
为了控制十六进制表示的长度,只需要在格式说明符“x”或者“X”后写上表示长度的数字就行了。比如a.ToString("X4")
将一个表示十六进制数的字符串转换成整型,需要借助于Parse()方 法:Parse(string,System.Globalization.NumberStyles)方法。第二个参数 System.Globalization.NumberStyles是一个枚举类型,用来表示十六进制的枚举值是HexNumber。int b = int.Parse("AB", System.Globalization.NumberStyles.HexNumber),b=171。
9. 日期型数据和长整型数据之间的转换
由于各种数据库对日期型的定义和处理是不一样的,各种语言对日期型数据的定义的处理也各不相同,为了方便将日期型数据转换成长整型再保存到数据库中。
日期型数据DateTime,在C#中的参与运算的时候是转换为长整型数据来运算的。它的长整型值是自0001年1月1日午夜12:00以来所经过时间以 100毫微秒为间隔表示时的数字。这个数在C#的DateTime中被称为Ticks(刻度)对应DateTime类型的Ticks的只读属性。long longDate = DateTime.Now.Ticks; DateTime的构造函数中也提供从长整型数据构造。如:DateTime theDate = new DateTime(longDate);
10. 格式化日期型数据
将日期型数据按照一定的格式输出需要使用System.DateTime类的ToString()方法,并为其指定格式字符串。MSDN 中,System.Globalization.DateTimeFormatInfo类的概述里对格式字符串有详细的说明。常用有:
d 月中的某一天 一位数的日期没有前导零
dd 月中的某一天 一位数的日期有一个前导零
ddd 周中某天的缩写名称 在 AbbreviatedDayNames 中定义
dddd 周中某天的完整名称 在 DayNames 中定义
M 月份数字 一位数的月份没有前导零
MM 月份数字 一位数的月份有一个前导零
MMM 月份的缩写名称 在 AbbreviatedMonthNames 中定义
MMMM 月份的完整名称 在 MonthNames 中定义
y 不包含纪元的年份 如果不包含纪元的年份小于 10,则显示不具有前导零的年份
yy 不包含纪元的年份 如果不包含纪元的年份小于 10,则显示具有前导零的年份
yyyy 包括纪元的四位数的年份
h 12 小时制的小时 一位数的小时数没有前导零
hh 12 小时制的小时 一位数的小时数有前导零
H 24 小时制的小时 一位数的小时数没有前导零
HH 24 小时制的小时 一位数的小时数有前导零
m 分钟 一位数的分钟数没有前导零
mm 分钟 一位数的分钟数有一个前导零
s 秒 一位数的秒数没有前导零
ss 秒 一位数的秒数有一个前导零
这时候又出现一个问题,如果要输出的文本信息中包含格式字符怎么办?如
format = "year: yyyy, month: MM, day: dd";
this.textBox1.AppendText(now.ToString(format) + "\n");
将输出:
2ear: 2002, 4on下5: 08, 26a2: 26
这并不是我想要的结果,怎么办呢?有办法
format = "\"year\": yyyy, \‘month\‘: MM, \‘day\‘: dd";
this.textBox1.AppendText(now.ToString(format) + "\n");
看,这次运行结果对了:
year: 2002, month: 08, day: 26
可以看出,只需要使用单引号或者双引号将文本信息括起来就好。
如果文本信息中包含双引号或者单引号又怎么办呢?这个问题,请读者们动动脑筋吧!