当前位置:首页 > 工业控制 > 电子设计自动化
[导读]有时候,我们可能想要构造一个很抽象的父类对象,它可能仅仅代表一个分类或抽象概念,它的实例没有任何意义,因此不希望它能被实例化。例如:有一个父类“ 水果(Fruit)”,它有几个子类“苹果(Apple)”、“橘子(

有时候,我们可能想要构造一个很抽象的父类对象,它可能仅仅代表一个分类或抽象概念,它的实例没有任何意义,因此不希望它能被实例化。例如:有一个父类“ 水果(Fruit)”,它有几个子类“苹果(Apple)”、“橘子(Orange)”、“香蕉(Banana)”等。水果在这里仅仅只是作为一个分类,显然水果的实例没有什么意义(就好像一个人如果告诉你他买了一些水果但是却不告诉你是苹果还是橘子,你很难想象他到底买的是什么。)。而水果类又要能被子类化,这就要求我们使用抽象类(abstract class)来解决这个问题。

在java中,通过在class关键字前增加abstract修饰符,就可以将一个类定义成抽象类。抽象类不能被实例化。例如:

定义抽象类水果(Fruit)

public abstract class Fruit {

……

}

如果我们试图用以下语句来获得一个实例,将无法编译成功。

Fruit fruit = new Fruit();

而我们仍然可以构造水果类的子类,如:

子类“苹果(Apple)”

public class Apple extends Fruit {

……

}

子类“橘子(Orange)”

public class Orange extends Fruit {

……

}

这样就达到我们的目的了。

抽象类除了能象普通类一样可以拥有一般的属性和方法,也可以拥有抽象方法(abstract method)。例如:

抽象类“形状(Shape)”拥有抽象方法draw()。

public abstract class Shape {

……

public abstract void draw();

……

}

抽象方法与抽象的行为相对应,通常是这个行为对父对象没有意义,而子对象有具体动作。例如方法draw()对于类Shape没有意义,而类Shape的子类矩形(Rectangle)的方法draw()可以有实际的动作(根据矩形的四个顶点画出矩形的四个边),子类圆(Circle)的方法draw()也可以有实际的动作(根据圆心和半径画出圆周)。

抽象类可以有抽象方法也可以没有抽象方法;但是如果一个类有抽象方法,那这个类只能定义为抽象类。

如果按照以下代码类“形状(Shape)”仍然拥有抽象方法draw(),但没有定义为抽象类,将会编译失败。

public class Shape {

……

public abstract void draw();

……

}

抽象方法还有一个特点是,它强迫子类要么仍然保持抽象性(即不具体实现该方法并仍然定义为抽象类),要么具体表现出这个方法的行为(实现具体的动作或者通过抛出UnsupportedOperationException异常来表明不支持该行为)。这样也可以强化多态性。

上面简要分析了抽象类,下面谈谈接口(interface)。java语言使用关键字interface定义一个接口。接口也是抽象对象,它甚至比抽象类更抽象。接口中的方法都是抽象方法。

一个接口可以继承其他接口;一个类通过关键字implements声明要实现一个接口,并具体实现接口的方法。

例如:有一个接口InterfaceA,

Java代码

public interface InterfaceA {

void methodA();

}

类ClassA实现接口InterfaceA。

Java代码

public class ClassA implements InterfaceA {

public void methodA() {

System.out.println( "methodA of ClassA implements InterfaceA" );

}

}

如果是抽象类实现一个接口,那么抽象类中可以不具体实现接口的方法(保持其抽象性),而由其子类去实现。

抽象类ClassB实现接口InterfaceA,但是没有具体实现方法methodA(),

Java代码

public abstract class ClassB { }

子类ClassBSub实现接口InterfaceA,但是没有具体实现方法methodA(),

Java代码

public class ClassBSub {

public void methodA() {

System.out.println( "methodA of ClassBSub the subclass of ClassB" );

}

}

接口和抽象类显著的共同点是接口和抽象类都可以有抽象方法。

接口和抽象类的不同点有:

(1)抽象类可以有实例变量,而接口不能拥有实例变量,接口中的变量都是静态(static)的常量(final)。

(2)抽象类可以有非抽象方法,而接口只能有抽象方法。

java中,类与类之间是不能多继承的。java之所以禁止类与类之间的多继承是因为多继承有很大的缺点。

多继承虽然能使子类同时拥有多个父类的特征,但是其缺点也是很显著的,主要有两方面:

(1)如果在一个子类继承的多个父类中拥有相同名字的实例变量,子类在引用该变量时将产生歧义,无法判断应该使用哪个父类的变量。例如:

类ClassA:

Java代码

public class ClassA {

protected int varSame = 0 ;

}

类ClassB:

Java代码

public class ClassB {

protected int varSame = 1 ;

}

子类ClassC:(假设允许类与类之间多继承)

Java代码

public class ClassC extends ClassA, ClassB {

public void printOut() {

System.out.println( super .varSame);

}

public static void main(String[] args) {

ClassC classC = new ClassC();

classC.printOut();

}

}

上面程序的运行结果会是什么呢?输出0还是1?

(2)如果在一个子类继承的多个父类中拥有相同方法,子类中有没有覆盖该方法,那么调用该方法时将产生歧义,无法判断应该调用哪个父类的方法。例如:

类ClassA:

Java代码

public class ClassA {

public void printOut() {

System.out.println( 0 );

}

}

类ClassB:

Java代码

public class ClassB {

public void printOut() {

System.out.println( 1 );

}

}

子类ClassC:(假设允许类与类之间多继承)

Java代码

public class ClassC extends ClassA, ClassB {

public static void main(String[] args) {

ClassA classA = new ClassC();

classA.printOut(); // ------------------------- A行

ClassB classB = new ClassC();

classB.printOut(); // ------------------------- B行

ClassC classC = new ClassC();

classC.printOut(); //------------------------- C行

}

}

上面程序的运行结果会是什么呢?A、B、C三行的输出是0还是1?

正因为有以上的致命缺点,所以java中禁止一个类继承多个父类;但是幸运的是java提供了接口,并能通过接口的功能获得多继承的许多优点而又摒弃了类与类多继承的缺点。

java允许一个接口继承多个父接口,也允许一个类实现多个接口,而这样的多继承有上面提到的缺点马?

答案是没有,这是由接口的抽象性决定的。

正如前面介绍的,在接口中不能有实例变量,只能有静态的常量,不能有具体的方法(包含方法体),只能有抽象方法,因此也就摒弃了多继承的缺点。

对于一个类实现多个接口的情况,因为接口只有抽象方法,具体方法只能由实现接口的类实现,在调用的时候始终只会调用实现类的方法(不存在歧义),因此不存在多继承的第二个缺点;而又因为接口只有静态的常量,但是由于静态变量是在编译期决定调用关系的,即使存在一定的冲突也会在编译时提示出错;而引用静态变量一般直接使用类名或接口名,从而避免产生歧义,因此也不存在多继承的第一个缺点。

对于一个接口继承多个父接口的情况也一样不存在这些缺点。

请看以下示例。

接口A:

Java代码

public interface InterfaceA {

int len = 1 ;

void output();

}

接口B:

Java代码

public interface InterfaceB {

int len = 2 ;

void output();

}

接口Sub继承接口A和接口B:

Java代码

public interface InterfaceSub extends InterfaceA, interfaceB { }

类Xyz实现接口Sub:

Java代码

public class Xyz implements InterfaceSub {

public void output() {

System.out.println( "output in class Xyz." );

}

public void outputLen( int type) {

switch (type) {

case InterfaceA.len:

System.out.println( "len of InterfaceA=." +type);

break ;

case InterfaceB.len:

System.out.println( "len of InterfaceB=." +type);

break ;

}

}

public static void main(String[] args) {

Xyz xyz= new Xyz ();

xyz .output();

xyz .outputLen();

}

以上代码不存在什么问题,但是如果试图编写以下存在冲突的代码,则会编译失败。

Java代码

Xyz xyz = new Xyz();

int len = xyz.len;

System.out.println(len);

由于引入了接口,java显得非常灵活,也使得java中的多态性更加富有魔力。



来源:xinxin0次

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭