• C#引用类型转换,到底使用is,as还是显式强转?


    在C#中,当引用类型需要转换的时候,经常会用到关键字is、as以及显式强转。本篇来体验这三者的用法。

    先来梳理.NET引用类型转换的"约定俗成",或者叫"惯例":
    ● 子类可以隐式转换为父类/基类,也就是"里氏替换原则"所说的:子类必须能够替换其父类/基类。
    ● 父类/基类转换为子类时,必须进行显式类型强转。

      子类转换成父类

    class Program
        {
            static void Main(string[] args)
            {
                Animal animal = new Dog(){Name = "Dog",Age = 1};
                Console.WriteLine(animal.Name);
                Console.ReadKey();
            }
        }
               
        public class Animal
        {
            public string Name { get; set; }
        }
    
        public class Dog : Animal
        {
            public int Age { get; set; }
        }

    输出结果:Dog   

    以上,体验了子类转换成父类,从中可以看出:子类Dog确实能替换父类Animal,换句话说,子类就是父类,子类的实例可以赋值给父类的变量,不需要as,不需要强转,一切都是隐式发生,很好地体现了"里氏替换原则"。

      父类转换为子类,父类变量通过子类赋值而创建

      父类as成子类:

    Animal animal = new Dog(){Name = "Dog",Age = 1};
                Dog dog = animal as Dog;
                Console.WriteLine(dog.Name + " " + dog.Age);
                Console.ReadKey();

    输出结果:Dog 1

      父类显式强转成子类

    Animal animal = new Dog(){Name = "Dog",Age = 1};
                Dog dog = (Dog)animal;
                Console.WriteLine(dog.Name + " " + dog.Age);
                Console.ReadKey();

    输出结果:Dog 1   

    以上,不管是使用as还是显式强转,父类都可以转换成子类,但是有前提的:把子类实例赋值给父类变量。  

      父类转换为子类,父类变量通过自身构造函数而创建

      父类as成子类失败:

    Animal animal = new Animal(){Name = "Sheep"};
                Dog dog = animal as Dog;
                Console.WriteLine(dog.Name + " " + dog.Age);
                Console.ReadKey();

    输出结果:报"NullReferenceException"错

    可见,当父类变量通过自身构造函数而创建时,无法将父类转换成子类。

    (1)为了避免as的时候报"NullReferenceException"错,引入关键is做类型判断:

    Animal animal = new Animal(){Name = "Sheep"};
                if (animal is Dog)
                {
                    Dog dog = animal as Dog;
                    Console.WriteLine(dog.Name + " " + dog.Age);
                }
                else
                {
                    Console.WriteLine("animal不能转换为Dog");
                }
                Console.ReadKey();

    输出结果:animal不能转换为Dog


    (2)用as做类型转换的时候,如果转换失败,返回null。根据这点,也可以通过判断as转换后的返回值是否为null,来避免报错:

    Animal animal = new Animal(){Name = "Sheep"};
                Dog dog = animal as Dog;
                if (dog != null)
                {
                    Console.WriteLine(dog.Name + " " + dog.Age);
                }
                else
                {
                    Console.WriteLine("animal不能转换为Dog");
                }
                Console.ReadKey();

    输出结果:animal不能转换为Dog   

      父类显式强转成子类失败

    Animal animal = new Animal(){Name = "Sheep"};
                Dog dog = (Dog) animal;
                Console.WriteLine(dog.Name + " " + dog.Age);
                Console.ReadKey();

    输出结果:报"InvalidCastException"错

    可见,当父类变量通过自身构造函数而创建时,无法将父类转换成子类。

    为了避免显示强转时报"InvalidCastException"错,引入关键字is做类型判断:

    Animal animal = new Animal(){Name = "Sheep"};
                if (animal is Dog)
                {
                    Dog dog = (Dog)animal;
                    Console.WriteLine(dog.Name + " " + dog.Age);
                }
                else
                {
                    Console.WriteLine("animal不能转换为Dog");
                }
                Console.ReadKey();

    输出结果:animal不能转换为Dog

      总结

    ● 对于引用类型的转换,应该考虑使用显式强转或as。两者的区别在于:一旦类型无法转换,使用显式强转会报错,而使用as会返回null。
    ● 为了避免显式强转或as所引发的报错,应该考虑使用is来判断类型间是否能转换。
    ● 使用as进行引用类型转换,不仅可以用is来判断类型间是否能转换,还可以通过判断as后的返回值是否为null,然后采取相应的操作。
    ● 对于基本类型的转换:使用Convert, Parse, TryParse等。

    is用于判断类型是否一致,as和显式强转用于类型的转换。

  • 相关阅读:
    linux c/c++ 获取文件大小
    android 打开各种文件(setDataAndType)
    Android framework系统默认设置修改
    android的 root权限
    [Power]待机电流问题,如何查找wakelock
    Android.mk for your own module
    通过adb 发送广播
    ubuntu下minicom的安装及使用
    ubuntu 下使用 putty 调试
    Android平台Overlay机制
  • 原文地址:https://www.cnblogs.com/darrenji/p/3831421.html
Copyright © 2020-2023  润新知