黑马程序员技术交流社区

标题: 抽象类中有构造函数,接口中为什么没有? [打印本页]

作者: minliang    时间: 2015-3-29 09:39
标题: 抽象类中有构造函数,接口中为什么没有?
抽象类有构造函数,接口中为什么没有?
作者: 奋斗的黑马    时间: 2015-3-29 10:03
从语法角度谈抽象类与接口区别
        在另一博文中 ‍http://hi.baidu.com/gabe2008/blog/item/a3f201ce9412df37f9dc61d4.html 以举例的方式说到如何区分抽象类和接口,这里我们从Java语法谈起,使我们更加了解这两者的内在区别。它们的语法区别:1)接口不能有构造方法,抽象类可以有。2)接口不能有方法体,抽象类可以有。3)接口不能有静态方法,抽象类可以有。4)在接口中凡是变量必须是public static final,而在抽象类中没有要求。
       忽然有此一文,是因为同学疑惑道:抽象类居然还有构造方法,又不能直接用来new。我的解释是平时在抽象类中对于构造方法的存在,没有什么印象,是因为IDE默认为你生成了一个无参构造方法,也可以显式地写出构造方法,这个构造方法,是用来被子类调用的,因为任何子类都必须调用从Object开始的所有父亲的构造方法,才算完成初始化工作。那么我引申一下,问他们,接口有构造方法吗?他们的理解,抽象可以有,为什么接口不可以有?!那么在接口里写入构造方法时,编译器提示:Interfaces cannot have constructors。这又何解?
        从语法的角度来说,抽象类必须有构造方法,而接口严禁有构造方法,这本身也说明了它们性质的不同。抽象类是一个类,别的类是用关键字 extends 来继承下来,并扩展的,有非常强的is-a的关系,这种关系一般来说符合里氏代换原则。而接口,是被其他类用关键字 implements 来实现接口定义的方法的。如果没什么区别,何必整出两个不同的关键字。   接口只是定义功能和行为规范,如果一个类实现了一个接口,那么这个类必须遵守这个接口的方法约定,但没有is-a的关系。把墙壁上的“小学生行为规范”想象成一个接口,那么是小学生必须遵守这个约定,但小学生不是“行为规范”。
       构造方法是用来在对象初始化前对对象进行一些预处理的,提供了实例化一个具体东西的入口。接口只是声明而已,不一定要进行什么初始化,就算要进行初始化,也可以到实现接口的那一些类里面去初始化。接口只是用来表述动作,表述规范来的,可以new一台computer,但我们无法new一个IDE、SATA、PCI、PS-2。因此,接口要构造方法何用?接口是一种规范,被调用时,主要关注的是里边的方法,而方法是不需要初始化的,类可以实现多个接口,若多个接口都有自己的构造器,则不好决定构造器的调用次序,构造器是属于类自己的,不能继承,因为是纯虚的,接口不需要构造方法。而抽象类是具体类的祖先,即使是石器时代,也总要干些初始化的工作,抽象类虽然是不能直接实例化,但实例化子类的时候,就会初始化父类,不管父类是不是抽象类都,都会调用父类的构造方法,初始化一个类,先初始化父类,有没有说初始化接口。
        再拿汽车的例子来说明两者的区别,Car, Track, Van 是 Vehicle 抽象类的子类,那么你可以说小娇车是车子,拖拉机是车子,货车是车子。而做为他们共同的父类,会做一些初始化工作,如加汽油、机油、冷却液。另外所有车子具有共同属性,轮子,方向盘,油门等。Brake 是一个刹车动作接口,这个规范要求车子实现了它,必须有能力把速度减到零,只是实现途径不一样,或者是鼓刹,或者是碟刹。或者是效果不一样,有的刹车灵一点,有的刹车差一点。
        在抽象类和接口均可以被考虑的地方,接口首先是被提倡使用的,在语法上没有is-a的关系,使用起来更加灵活,另外可以多实现毕竟是一种难得的资源。而抽象类当存在大家共同的实现方法,或者有很多属性操作时,才是首选,当考虑使用抽象类时,在可预见的未来,它要体现出被继承时非常强烈的is-a关系。
       接口不能有方法体,就是强制接口定义者不能给接口增加一些详细的实现,指定的必须是纯虚的接口,对于架构者来说,它需要做的就是定义一个可以理解的接口名,参数列表,以及返回类型。而抽象类是可以有自己的实现方法,这个方法可以被用来被执行。有时候,我们会误以为抽象类中的方法不能用super显示调用,因为super是指代父类对象,而抽象类MS不能有对象。但事实上可以,任何子类实际的对象,都可以理解成父类的对象。
        接口的所有方法都是抽象的,而抽象方法是没有static,有static的方法是不能override的,不能实现多态,所以这样定义接口才有意义,接口中定义的方法目的很明确:就是给实现类去实现,如果你在接口中将方法声明为静态的(与具体的实例无关),但接口中的方法又要求必须被实现类去实现(可能会有多个实现类),这岂不是自相矛盾!如果硬要加上static上去,编译器会友好提示:Illegal modifier for the interface method FF.f(); only public & abstract are permitted。说到底,不能再加任何除public 和 abstract 的任何修饰符了,如:private, protected, final, static.
        数据成员没有多态的概念,只有可不可以被访问的说法,如果是公有的数据成员是在任何地方都可以被访问和修改的,那么接口定义的数据成员如果是私有的,又何必定义,因为它没有方法,就是说没有任何方法对这个数据成员可以操作。如果是公有的,谁都可以修改,那么,多重实现多个接口,这个数据不是改得乱了套?所以接口一般不允许有数据成员,但如果真的有,明确默认即为public final static, 严格保证对于所有实现者来说只有一份原始的数据。
        那么在架构设计时,如何使用抽象类和接口来解决问题,是一个非常复杂的问题,抽象类更侧重于归纳同一父类的子类的共同特征,如果属性,方法;接口更侧重于定义任意的类有没有相同语义的方法,它是一个一经定义不轻易更改的规范,它的修改在项目中,往往是动一发而牵全身,即使有考虑不周到的地方,也会使用新增接口的形式去弥补。其实abstract class表示的是"is-a"关系,interface表示的是"like-a"关系。最顶级的是接口,然后是抽象类实现接口,最后才到具体类实现。
作者: 剑雨飘扬    时间: 2015-3-29 10:17
抽象类中可以有具体的方法,可以创建类对象;接口中的方法全是抽象的,不能创建对象。
作者: leonard    时间: 2015-3-29 13:38
抽象类中包含了非静态的方法和成员变量,在子类初始化时要先对父类初始化,即使父类是抽象类也要初始化。接口中包含的全部都是抽象方法,子类实现接口时不需要对接口初始化,必须全部覆写父类中的抽象方法,如果没有全部覆写,则子类是一个抽象类。
作者: minliang    时间: 2015-3-29 14:33
奋斗的黑马 发表于 2015-3-29 10:03
从语法角度谈抽象类与接口区别
        在另一博文中 ‍http://hi.baidu.com/gabe2008/blog/item/a3f201ce9 ...

说的很详细,谢啦
作者: 大手牵小手    时间: 2015-3-29 21:26
一楼回答好详细,涨知识了!
作者: jiandonn    时间: 2015-3-29 21:51
抽象类中可以有具体的方法,可以创建类对象;接口中的方法全是抽象的,不能创建对象。
作者: 佐佑    时间: 2015-3-29 22:05
好好找书吧
作者: 刘大志    时间: 2015-3-29 22:27
因为接口都是抽象方法,不需要调用成员变量,因而,他不需要成员变量,只有成员常量,进而不需要用构造函数对变量初始化。
作者: wdhm5423    时间: 2015-3-29 22:30
接口如果有构造函数,那构造函数也必须是抽象的。构造函数不能被抽象修饰,只能被访问权限修饰,无返回值,无returen。
作者: 我_杰仔    时间: 2015-3-29 22:33
接口不能被创建对象
作者: 齐小灰同学    时间: 2015-3-29 22:43
肯定是抽象类被实例化了以后,才有了构造函数吧。抽象类怎么可能有构造函数。接口因为不能创建对象,所以也就不存在构造函数了
作者: 一步一个脚印    时间: 2015-3-29 22:45
接口是一个方法列表,抽象类是一个完整的类,也可以生成对象,不过生成的对象基本没什么用。类描述了生成的对象可以处理的消息。而接口表示了对象本身的行为。
作者: minliang    时间: 2015-3-30 19:54
齐小灰同学 发表于 2015-3-29 22:43
肯定是抽象类被实例化了以后,才有了构造函数吧。抽象类怎么可能有构造函数。接口因为不能创建对象,所以也 ...

抽象类是有构造函数的
作者: 齐小灰同学    时间: 2015-3-31 21:54
minliang 发表于 2015-3-30 19:54
抽象类是有构造函数的

我明天学构造函数。。。那我仔细听听
作者: Chenli    时间: 2015-3-31 22:03
来学习下
作者: lf027    时间: 2015-3-31 22:16
分析的真好
作者: luoyu1530    时间: 2015-3-31 22:31
构造函数是用来给类初始化的,你接口中都没东西要初始化,要构造函数干嘛?抽象类中有不是抽象的方法和变量,所以才要构造函数
作者: 高双    时间: 2015-3-31 22:38
多看看视频
作者: yanggz    时间: 2016-9-13 23:31
楼上回答的很好
作者: 江月念华    时间: 2016-9-13 23:52
那么问题来了,抽象类是怎么被调用的?
作者: cat哥    时间: 2016-9-13 23:57
被二楼的长篇幅回帖吓到
作者: 左右.律    时间: 2016-9-14 00:06
看了一楼的回复我就不敢再说了
作者: 辰途27    时间: 2016-9-14 09:18
看了一楼,长知识了,大神啊
作者: icefish    时间: 2016-9-14 11:52
接口中都是抽象方法,没有方法体。构造方法就是用来初始化创建对象的,但是创建一个接口的对象,没有什么用,所以就不给接口社会构造方法了
作者: 杨凯233    时间: 2016-9-14 13:12
如果有,还叫啥接口呢
作者: yanggz    时间: 2016-9-14 19:16
涨知识了!!!!!
作者: Toxicant    时间: 2016-9-14 20:16
接口是功能的的集合,只是提供了要做什么,不解决怎么做的问题,所有没有实现的具体代码,没必要设置构造,而抽象类虽然有抽象方法,但是他可以有自己的东西,有自己要实现的功能,所以要有构造




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2