java学习面向对象之接口

上一节当中我们说道抽象类,抽象类当中的方法可以是抽象的也可以是非抽象的,那么当抽象类中所有方法都是抽象的时候,我们就可以把它重新定义为接口。代码示例:

1 abstract class Animal
2 {
3 
4     abstract void eat();
5     abstract void goToBed();
6 
7 }

上述代码写成接口的形式为:

1 interface Animals
2 {
3 
4     public void eat();
5     public void goToBed();
6 
7 }

在接口当中的方法,也只能是方法名称,没有方法体。

类与类当中存在的关系是继承关系,那么类与接口之间是什么关系呢?类与接口之间的关系就是实现的关系,类实现接口。类与类当中只能够单继承,作为体系的扩展延伸,但是类与接口之间的实现,就对类做了功能上的扩展。同时接口的出现,也是java支持多继承的表现。虽然java不支持类的多继承,但是java中接口是可以多实现的,也就是说可以做到跟多继承相像的多功能上的扩展。并且,实现和继承是可以同时实现的。

类与接口的实现:

比如我们现在有个笔记本电脑,笔记本电脑是继承自电脑的,具有电脑所有的功能,但是笔记本本身又特有,折叠这个特殊的功能。我们可以这样定义,笔记本电脑继承自电脑这个抽象类,同时实现了折叠的这个功能。我们用代码来体现下:

 1 /**
 2 *抽象类Pc抽取出了Pc当中必须实现的
 3 *一些功能
 4 */
 5 abstract class Pc
 6 {
 7     /**
 8     *抽象类方法connectNet(),实现连接网络的方法
 9     *没有返回值
10     */
11     abstract void connectNet();
12     /**
13     *抽象类方法hasMouse(),实现电脑都有鼠标的功能
14     *没有返回值
15     */
16     abstract void hasMouse();
17 
18 }
19 
20 
21 /**
22 *接口ZheDie,提供折叠的功能
23 */
24 interface ZheDie
25 {
26     /**
27     *zheDie()提供折叠的方法
28     *没有返回值
29     */
30     public abstract void zheDie();
31 
32 }
33 
34 /**
35 *笔记本电脑继承了电脑这个体系,证明笔记本电脑是属于电脑的
36 *,具有电脑的所有功能以及属性。同时他又实现了折叠,这个特殊的
37 *功能。
38 */
39 class BookPc extends Pc implements ZheDie
40 {
41     /**
42     *实现了连网功能
43     */
44     public void connectNet()
45     {
46 
47         System.out.println("BookPc can connectNet");
48 
49     }
50     /**
51     *实现了鼠标功能
52     */
53     public void hasMouse()
54     {
55 
56         System.out.println("BookPc has a Mouse");
57 
58     }
59     /**
60     *实现了折叠的功能
61     */
62     public void zheDie()
63     {
64 
65         System.out.println("BookPc can be zheDie");
66 
67     }
68 
69 }
70 
71 class InterDemo2
72 {
73 
74     public static void main(String[] args) {
75         
76         new BookPc().zheDie();
77 
78     }
79     
80 
81 }

接口当中除了有方法之外,还可以有属性:

1 interface Demo
2 {
3 
4     public static final int a = 10;
5 
6 }

那么接口都有哪些细节需要注意呢?

1、接口不同于类,接口的定义方法是interface关键字不是class关键字。

2、接口只能被类用implement关键字实现不能被类继承,但是接口与接口之间可以被继承

3、接口可以实现类的多继承,也就是多实现。

4、接口中的方法都是抽象的并且是公开的,包括属性,也就是接口当中的属性还有方法前面的修饰词都是固定的。如果不加修饰词的情况下,在编译的时候会给加上的。

5、接口当中的方法还有成员变量都是公开的。成员函数用public abstract修饰,成员变量用public static final 修饰。因为类是实现接口,而不是继承接口。

6、接口当中的方法,必须被要实现这个接口的类全部实现。否则,这个实现了这个接口的类就是一个抽象类。

7、接口的出现,避免了单继承的局限性

java利用接口实现多继承:

在java当中不支持多继承,因为这样存在不确定性方法一样,不知道调用哪个?。但是java当中为了支持多继承,就用了“多实现”。多继承的意义存在于:提供功能,提高扩展性。

一个类可以实现多个接口。

 1 interface A
 2 {
 3 
 4     public void show1();
 5 
 6 }
 7 
 8 interface B
 9 {
10 
11     public void show2();
12 
13 }
14 
15 class Test implements    A,B //这个地方就体现了java通过多实现的方式,实现了多继承
16 {
17 
18     public void show1()
19     {
20 
21         System.out.print("Show1()");
22 
23     }
24 
25     public void show2()
26     {
27 
28         System.out.print("Show2()");
29 
30     }
31 
32 }
33 
34 class InterDemo3
35 {
36 
37     public static void main(String[] args) {
38         
39         Test t = new Test();
40         t.show1();
41         t.show2();
42 
43     }
44 
45 }

多实现需要注意的地方:

1、多实现的时候,由于没有方法体,就没有了两个方法,同时具有不同的方法体,且当实现的时候不知道该实现哪个方法这个不确定性的隐患。因为即使一个类实现了多个接口的时候,多个接口当中又具有相同方法的时候,这个时候相同的方法都是抽象方法,并且这些个抽象的方法在被类实现的时候,就被一个方法覆盖了,所以并不存在不确定性。

2、在抽象方法当中,返回值可以定义,方法接收的参数也是可以定义的。

 1 interface A
 2 {
 3 
 4     public void show();
 5 
 6 }
 7 
 8 interface B
 9 {
10 
11     public void show(int a,int b);
12 
13 }
14 
15 class Test implements    A,B 
16 {
17 
18     public void show()
19     {
20 
21         System.out.print("Show1()");
22 
23     }
24 
25     public void show(int a , int b)
26     {
27 
28         System.out.print("a + b = "+(a+b));
29 
30     }
31 
32 }
33 
34 class InterDemo3
35 {
36 
37     public static void main(String[] args) {
38         
39         Test t = new Test();
40         t.show();
41         System.out.println();
42         t.show(7,8);
43 
44     }
45 
46 }

这个样子写是正确的,但是这个样子写就是错误的:

 1 interface A
 2 {
 3 
 4     public void show();
 5 
 6 }
 7 
 8 interface B
 9 {
10 
11     public int show();//这样子写是错误的
12 
13 }
14 
15 class Test implements    A,B 
16 {
17 
18     public void show()
19     {
20 
21         System.out.print("Show1()");
22 
23     }
24 
25     public int show()
26     {
27 
28         return 1;
29 
30     }
31 
32 }
33 
34 class InterDemo3
35 {
36 
37     public static void main(String[] args) {
38         
39         Test t = new Test();
40         t.show();
41         System.out.println();
42         t.show();
43 
44     }
45 
46 }

这个就叫做,调用的不确定性。

接口与接口之间的关系,接口与接口之间是继承关系,并且是多继承的关系。因为接口当中的抽象方法,就避免了不确定性,比如:

 1 interface AA
 2 {
 3 
 4     public void show1();
 5 
 6 }
 7 
 8 interface BB
 9 {
10 
11     public void show2();
12 
13 }
14 
15 interface CC extends AA,BB
16 {
17 
18 
19 
20 }

这个就是接口与接口之间的继承关系,并且是多继承关系。

这样,当一个类实现CC这个接口的时候,就同时必须实现两个方法,即:

1 class Test implements CC
2 {
3 
4     public void show1(){}
5     public void show2(){}
6 
7 }

接口的特点:

1、接口是对外曝露的规则;

2、接口是对程序的扩展;

3、接口能够降低程序的耦合性;

4、接口可以用来多实现。

5、类与接口之间是实现的关系,一个类可以继承自一个类的同时,实现多个接口。而接口与接口之间是继承且是多继承的关系。

为了更形象的说明这个特点,我们来举一个例子来说一下,就拿笔记本电脑的这个例子来说:

早期的笔记本电脑是没有usb接口的,因为早期的笔记本电脑是把所需要的功能直接嵌入到笔记本当中,这样我们就可以用了。但是这个时候就出现了一个问题,就是如果我们不习惯笔记本的触摸板,想把触摸板替换成鼠标的话,我们就需要把笔记本拆开,并且把触摸板上面的两个线接入到鼠标上面,这样鼠标就可以使用了。同时如果我们这个时候笔记本键盘坏掉了,我们应该怎么办呢,我们也需要把笔记本拆开把键盘上的线引出来 ,接到我们完好的键盘上面。在做这些事情的时候,我们就会发现一个问题,这样来回的接线,会显得我们的笔记本电脑不够人性化,我们就在想,如果我们在生产笔记本的时候,可以预先保留几个特定规格的且有特定规则的接口,来供我们使用,这样就大大提高了笔记本的扩展性,和使用性。比如说,笔记本的生产厂家,在生产笔记本的时候,提前预留了这个接口。同时,鼠标、键盘厂家也按照这个接口做了一些鼠标和键盘出来,这样当我们不想用原装的时候,就可以利用这个usb接口直接接外接的键盘或者鼠标。这就就显得方便了许多,同样我们电脑主板上面也有许多这样的接口,不如PCI接口啊,内存接口啊,等等之类的。

我们用代码来体现的话就是:

 1 interface Usb
 2 {
 3 
 4     public void open();
 5     public void close();
 6 
 7 }
 8 
 9 
10 class UMouse implements Usb
11 {
12 
13     public void open()
14     {
15 
16         System.out.println("Mouse has opened");
17 
18     }
19 
20     public void close()
21     {
22 
23         System.out.println("Mouse has closed");
24 
25     }
26 
27 }
28 
29 class UKeypbord implements Usb
30 {
31 
32     public void open()
33     {
34 
35         System.out.println("Keybord has opened");
36 
37     }
38 
39     public void close()
40     {
41 
42         System.out.println("Keybord has closed");
43 
44     }
45 
46 }
47 
48 class BookPc
49 {
50 
51     public static void main(String[] args) {
52         
53         BookPc b = new BookPc();
54         b.method(null);
55         b.method(new UMouse());
56         b.method(new UKeypbord());
57     }
58 
59     private void method(Usb u)
60     {
61         if(u!=null){
62             u.open();
63             u.close();
64         }
65     }
66 
67 }

这里来说一下 这个例子是如何体现接口的特点的:

1、定义接口,这个时候提供了给外界使用,并且鼠标、键盘厂商必须实现的两个方法,设备的关闭和开启。这个就是对外曝露的规则。

2、因为在笔记本出厂之前就实现了对这个功能的扩展,所以这个也是对笔记本电脑功能上的扩展。

3、同时在利用接口的同时,就不用把所有的代码都写在笔记本这个类当中,这样就降低了程序的耦合性,这样当我们要增加外加设备的时候,比如说要增加一个usb摄像头,这个时候,我们 就可以在外边定义一个摄像头的类,然后实现usb这个接口中的所有方法,之后,在pc当中直接调用就好了。

以上就是接口的好处还有优点。

接口和抽象类的异同点总结:

1、共性:都是不断抽取出来的抽象概念。

2、异性一:抽象类体现继承关系,一个类只能单继承。接口体现实现的关系,一个类可以多实现。

3、异性二:抽象类是继承,是is a关系,在定义该体系当中基本共性的内容。接口是实现,是like a关系,在定义体系额外功能。举个例子来说,学生都具有学习的功能,但是打架的功能就是额外的,有的孩子不打架,有的就是天天打架。在这个地方我们用一个例子来形容下:

在我们生活当中犬又分为导盲犬、搜救犬等等,这个是按照功能来划分的,有的狗没有,有的狗就有。但是他们有一个共性就是都是犬,犬就是一个体系基本的功能,比如有叫,而导盲、搜救这些都属于额外功能,我们用代码来体现下。

 1 abstract class Dog
 2 {
 3 
 4     abstract public void houJiao();
 5 
 6 }
 7 
 8 interface DaoMang
 9 {
10 
11     public void lookWay();
12 
13 }
14 
15 interface SouJiu
16 {
17 
18     public void giveHelp();
19 
20 }
21 
22 class DaoMangDog extends Dog implements DaoMang
23 {
24 
25     public void houJiao()
26     {
27 
28         System.out.println("Flow me!");
29 
30     }
31 
32     public void lookWay()
33     {
34 
35         System.out.println("I can give you right way to home");
36 
37     }
38 
39 }
40 
41 class SouJiuDog extends Dog implements SouJiu
42 {
43 
44     public void houJiao()
45     {
46 
47         System.out.println("Is someone need Help?");
48 
49     }
50 
51     public void giveHelp()
52     {
53 
54         System.out.println("Ok I can give you a hand");
55 
56     }
57 
58 }
59 
60 class InterDemo6
61 {
62 
63     public static void main(String[] args) {
64         
65         new SouJiuDog().houJiao();
66         new DaoMangDog().houJiao();
67 
68     }
69 
70 }

4、异性三:抽象类当中有构造函数还有非抽象方法,可以直接供子类来使用。而接口当中只能存在抽象方法,并且都是公开属性,且有固定的修饰符。

原文地址:https://www.cnblogs.com/sunchuanzhen/p/3348249.html