• 接口和抽象类比较 (干货)


    接口

    1.接口不能被实例化
    2.接口只能包含方法声明
    3.接口的成员包括方法、属性、索引器、事件
    4.接口中不能包含常量、字段(域)、构造函数、析构函数、静态成员。
     

    抽象类:

    1.抽象方法只作声明,而不包含实现,可以看成是没有实现体的虚方法
    2.抽象类不能被实例化
    3.抽象类可以但不是必须有抽象属性和抽象方法,但是一旦有了抽象方法,就一定要把这个类声明为抽象类
    4.具体派生类必须覆盖基类的抽象方法
    5.抽象派生类可以覆盖基类的抽象方法,也可以不覆盖。如果不覆盖,则其具体派生类必须覆盖它们。
     

    接口除了可以包含方法之外,还可以包含属性、索引器、事件,而且这些成员都被定义为公有的。除此之外,不能包含任何其他的成员,例如:常量、域、构造函数、析构函数、静态成员。一个类可以直接继承多个接口,但只能直接继承一个类(包括抽象类)。

    抽象类是特殊的类,只是不能被实例化;除此以外,具有类的其他特性;重要的是抽象类可以包括抽象方法,这是普通类所不能的。抽象方法只能声明于抽象类中,且不包含任何实现,派生类必须覆盖它们。另外,抽象类可以派生自一个抽象类,可以覆盖基类的抽象方法也可以不覆盖,如果不覆盖,则其派生类必须覆盖它们。

    接口和抽象类相同点:

        (1) 都可以被继承

        (2) 都不能被实例化

        (3) 都可以包含方法声明

        (4) 派生类必须实现未实现的方法

    接口和抽象类的区别:

    1.类是对对象的抽象,可以把抽象类理解为把类当作对象,抽象成的类叫做抽象类.而接口只是一个行为的规范或规定,

    微软的自定义接口总是后带able字段,证明其是表述一类类“我能做。。。”。抽象类更多的是定义在一系列紧密相关的类

    间,而接口大多数是关系疏松但都实现某一功能的类中;
         

    2. 接口基本上不具备继承的任何具体特点,它仅仅承诺了能够调用的方法;
        

    3. 一个类一次可以实现若干个接口,但是只能扩展一个父类;
         

    4. 接口可以用于支持回调,而继承并不具备这个特点;
         

     5. 抽象类不能被密封;
         

    6. 抽象类实现的具体方法默认为虚的,但实现接口的类中的接口方法却默认为非虚的,当然您也可以声明为虚的;
         

    7.(接口)与非抽象类类似,抽象类也必须为在该类的基类列表中列出的接口的所有成员提供它自己的实现。但是,允许抽象类将接口方法映射到抽象方法上;
        

     8. 抽象类实现了oop中的一个原则,把可变的与不可变的分离。抽象类和接口就是定义为不可变的,而把可变的座位子类去实现;
       

    9. 好的接口定义应该是具有专一功能性的,而不是多功能的,否则造成接口污染。如果一个类只是实现了这个接口的中一个功能,而不得不去实现接口中的其他方法,

    就叫接口污染;
        

    10. 尽量避免使用继承来实现组建功能,而是使用黑箱复用,即对象组合。因为继承的层次增多,造成最直接的后果就是当你调用这个类群中某一类,就必须把他们全部

    加载到栈中!后果可想而知。(结合堆栈原理理解)。同时,有心的朋友可以留意到微软在构建一个类时,很多时候用到了对象组合的方法。比如 asp.net中,Page类,

    有Server Request等属性,但其实他们都是某个类的对象。使用Page类的这个对象来调用另外的类的方法和属性,这个是非常基本的一个设计原则;
       

     11.如果抽象类实现接口,则可以把接口中方法映射到抽象类中作为抽象方法而不必实现,而在抽象类的子类中实现接口中方法。

    举个例子:

    •飞机会飞,鸟会飞,他们都继承了同一个接口“飞”;但是F22属于飞机抽象类,鸽子属于鸟抽象类。
    •就像铁门木门都是门(抽象类),你想要个门我给不了(不能实例化),但我可以给你个具体的铁门或木门(多态);而且只能是门,你不能说它是窗(单继承);一个门可以有锁(接口)也可以有门铃(多实现)。 门(抽象类)定义了你是什么,接口(锁)规定了你能做什么(一个接口最好只能做一件事,你不能要求锁也能发出声音吧(接口污染)。

    简单的接口程序:

    using System;
    
    namespace Interfaces
    {
        interface IEquatable<T>
        {
            bool Equals(T obj);
        }
    
        public class Car : IEquatable<Car>
        {
            public string Make { get; set; }
            public string Model { get; set; }
            public string Year { get; set; }
    
            // Implementation of IEquatable<T> interface
            public bool Equals(Car car)
            {
                if (this.Make == car.Make &&
                    this.Model == car.Model &&
                    this.Year == car.Year)
                {
                    return true;
                }
                else
                    return false;
            }
    
        }
    
        public class  testInterface
        {
            
            public static void Main(string[] args)
            {
                Car Audi = new Car();
                Audi.Make = "Audi";
                Audi.Model = "A8";
                Audi.Year = "2015";
    
                Car BMW = new Car();
                BMW.Make = "BMW";
                BMW.Model = "X5";
                BMW.Year = "2015";
    
                if (Audi.Equals(BMW))
                {
                    Console.WriteLine("Audi is the same as BMW!");
                    Console.ReadKey();
                }
                else
                {
                    Console.WriteLine("Audi is different from BMW!");
                    Console.ReadKey();
                }
         
            }
        }
    
    }
    View Code


    The out put is Audi is different from BMW!


    You can refer to below websites for some basic examples:

    http://www.tutorialspoint.com/csharp/csharp_interfaces.htm

    http://www.runoob.com/csharp/csharp-interface.html

  • 相关阅读:
    Eclipse中的快捷键
    Eclipse配置
    Java语言概述
    视口及媒体查询
    弹性盒flex
    less——css预处理语言
    过渡/动画/变形
    列表/表格/表单
    渐变
    背景background
  • 原文地址:https://www.cnblogs.com/I-am-Betty/p/5063531.html
Copyright © 2020-2023  润新知