C#.NET 中的类型转换(二)

 


(该文于 2002 年 8 月 28 日发表于计算机世界知识中心)
C#.NET 中的类型转换(续)
边城狂人
===================


6. 字符串和字节数组之间的转换

    如果还想从 System.String 类中找到方法进行字符串和字节数组之间的转换,恐怕你会失望了。为了进行这样的转换,我们不得不借助另一个类: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 则会将它们都转换成两个字节。我们可以通过下列简单的了解一下转换的方法,以及使用 Encoding.Default 和 Encodeing.Unicode 的区别:

    private void TestStringBytes() {
        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 还是由编码决定。在 TestStringBytes() 函数中添加如下语句作为实例:

        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. 各种数值类型和字节数组之间的转换

    在第 1 条中我们可以查到各种数值型需要使用多少字节的空间来保存数据。将某种数值类型的数据转换成字节数组的时候,得到的一定是相应大小的字节数组;同样,需要把字节数组转换成数值类型,也需要这个字节数组大于相应数值类型的字节数。
    现在介绍此类转换的主角:System.BitConverter。该类提供了 byte[] GetBytes(...) 方法将各种数值类型转换成字节数组,也提供了 ToInt32、ToInt16、ToInt64、ToUInt32、ToSignle、ToBoolean 等方法将字节数组转换成相应的数值类型。

    由于这类转换通常只是在需要进行较细微的编码/解码操作时才会用到,所以这里就不详细叙述了,仅把 System.BitConverter 类介绍给大家。

8. 转换成十六进制

    任何数据在计算机内部都是以二进制保存的,所以进制与数据的存储无关,只与输入输出有关。所以,对于进制转换,我们只关心字符串中的结果。
    在上面的第 4 条中提到了 ToString() 方法可以将数值转换成字符串,不过在字符串中,结果是以十进制显示的。现在我们带给它加一些参数,就可以将其转换成十六进制——使用 ToString(string) 方法。
    这里需要一个 string 类型的参数,这就是格式说明符。十六进制的格式说明符是 "x" 或者 "X",使用这两种格式说明符的区别主要在于 A-F 六个数字:"x" 代表 a-f 使用小写字母表示,而 "X" 而表示 A-F 使用大字字母表示。如下例:

    private void TestHex() {
        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

    这时候,我们可能有另一种需求,即为了显示结果的整齐,我们需要控制十六进制表示的长度,如果长度不够,用前导的 0 填补。解决这个问题,我们只需要在格式说明符“x”或者“X”后写上表示长度的数字就行了。比如,要限制在 4 个字符的长度,可以写成“X4”。在上例中追加一句:

        this.textBox1.AppendText("a(16) = " + a.ToString("X4") + "/n");

    其结果将输出 a(16) = 00BC。
    现在,我们还要说一说如何将一个表示十六进制数的字符串转换成整型。这一转换,同样需要借助于 Parse() 方法。这里,我需要 Parse(string, System.Globalization.NumberStyles) 方法。第一个参数是表示十六进制数的字符串,如“AB”、“20”(表示十进制的 32) 等。第二个参数 System.Globalization.NumberStyles 是一个枚举类型,用来表示十六进制的枚举值是 HexNumber。因此,如果我们要将“AB”转换成整型,就应该这样写:int b = int.Parse("AB", System.Globalization.NumberStyles.HexNumber),最后得到的 b 的值是 171。

9. 日期型数据和长整型数据之间的转换

    为什么要将日期型数据转换为长整型数据呢?原因很多,但就我个人来说,经常将它用于数据库的日期存储。由于各种数据库对日期型的定义和处理是不一样的,各种语言对日期型数据的定义的处理也各不相同,因为,我宁愿将日期型数据转换成长整型再保存到数据库中。虽然也可以使用字符串来保存,但使用字符串也会涉及到许多问题,如区域等问题,而且,它需要比保存长整型数据更多的空间。
    日期型数据,在 C# 中的参与运算的时候,应该也是转换为长整型数据来运算的。它的长整型值是自 0001 年 1 月 1 日午夜 12:00 以来所经过时间以 100 毫微秒为间隔表示时的数字。这个数在 C# 的 DateTime 中被称为 Ticks(刻度)。DateTime 类型有一个名为 Ticks 的长整型只读属性,就保存着这个值。如此,要从一个 DataTime 型数据得到 long 型值就非常简单了,只需要读出 DataTime 对象的 Ticks 值即可,如:

    long longDate = DateTime.Now.Ticks;

    DateTime 的构造函数中也提供了相应的,从长整型数据构造 DateTime 型数据的函数:DateTime(long)。如:

    DateTime theDate = new DateTime(longDate);

    但这样对于很多 VB6 程序员来说,是给他们出了一道难题,因为 VB6 中的日期型数据内部是以 Double 型表示的,将其转换为长整型后得到的仅仅是日期,而没有时间。如何协调这两种日期类型呢?
     System.DateTime 提供了 double ToOADate() 和 static DateTime FromOADate(double) 两个函数来解决这个问题。前者将当前对象按原来的 double 值输出,后者则从一个 double 值获得一个 System.DateTime 对象。举例如下:

    private void TestDateTimeLong() {
        double doubleDate = DateTime.Now.ToOADate();
        DateTime theDate = DateTime.FromOADate(doubleDate);
        this.textBox1.Text = "";
        this.textBox1.AppendText("Double value of now: " + doubleDate.ToString() + "/n");
        this.textBox1.AppendText("DateTime from double value: " + theDate.ToString() + "/n");
    }

    运行结果:

    Double value of now: 37494.661541713
    DateTime from double value: 2002-8-26 15:52:37

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      秒。一位数的秒数有一个前导零。

    为了便于大家的理解,不妨试试下面的程序:

    private void TestDateTimeToString() {
        DateTime now = DateTime.Now;
        string format;
        this.textBox1.Text = "";
        format = "yyyy-MM-dd HH:mm:ss";
        this.textBox1.AppendText(format + ": " + now.ToString(format) + "/n");
        format = "yy年M日d日";
        this.textBox1.AppendText(format + ": " + now.ToString(format) + "/n");
    }

    这段程序将输出结果:

    yyyy-MM-dd HH:mm:ss: 2002-08-26 17:03:04
    yy年M日d日: 02年8日26日

    这时候,又出现一个问题,如果要输出的文本信息中包含格式字符怎么办?如

        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

    可以看出,只需要使用单引号或者双引号将文本信息括起来就好。
    如果文本信息中包含双引号或者单引号又怎么办呢?这个问题,请读者们动动脑筋吧!

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值