黑马程序员技术交流社区

标题: Java接口详解 [打印本页]

作者: hihi    时间: 2015-6-12 21:27
标题: Java接口详解
  1. 对初学者来说,接口不是很好理解。现将某高手的一篇文章贴出来,共大家分享!
  2. 我们来看一个类
  3. class A {
  4. private int a;
  5. public int getA() {
  6. return a;
  7. }
  8. }
  9. 这个类的属性是私有的,外界不能访问,而外界可以通过公有方法来访问这个类。我们说一个类的公有方法就是这个类的对外接口。通常一个类的属性都是私有的,方法大多是公有的。外界只能过个这些公有方法来访问类。这也是Java封装性的体现。如果一个类没有公有属性, 也没有公有方法,这个类就是无法使用的类了。所以我们需要为一个类提供对外接口。

  10. 一个类中的方法,不只是说明了它要“做什么”,而且方法的内容也说明了“怎么做”。打个不太恰当的例子,一个杀人方法。从名字上看,你知道了“做什么”,但没有办法看到“怎么做”。而方法的内容说明了“怎么做”。
  11. class killer {
  12. private String name;
  13. private int age;
  14. private String phone;
  15. private String addr;
  16. ......
  17. public void kill(Person p) {
  18. Qiang qiang = new Qiang("ak47");
  19. qiang.fire(p);
  20. }
  21. }
  22. 这个类的kill方法内容说明了杀人的过程。如果你不想用这种方式杀人。因为你很BT,你想用毒药杀人。那么这个类的内容就需要改。但是,还有很多其它的“客户”,需要用不同的方式杀人。怎么办呢?一个很好的办法就是,我们只定义“做什么”,而不定义“怎么做”。
  23. interface Killer {
  24. public void kill(Person p);
  25. }
  26. 接口说明了“做什么”,而实现这个接口的类,也就是实现类需要说明“怎么做”。
  27. class Killer1 implements Killer {
  28. public void kill(Person p) {
  29. Qiang qiang = new Qiang("ak47");
  30. qiang.fire(p);
  31. }
  32. }
  33. class Killer2 implements Killer {
  34. public void kill(Person p) {
  35. Bane bane = new Bane();
  36. p.eat(bane);
  37. }
  38. }
  39. public class Test {
  40. public static void main(String[] args) {
  41. Killer jingKe = new Killer1();
  42. Person yingZheng = new Person();
  43. jingKe.kill(yingZheng);
  44. }
  45. }
  46. 接口可以把“做什么”和“怎么做”分离开来。这给Java带来了很多好处。虽然代码量增加了,可我们的程序的可维护性加强了。我们的程序是可以拆分的。就象电脑一样,可以拆分成很多组件。我一直在想,如果我的MP3耳机可以拆分就好了,那样在耳机只坏掉一个的时候就不用重新买一个了。

  47. 不过上面的例子看不到什么太大的好处。你可能会说,如果你的目的是为了不修改代码,那么,如果我想使用Killer2来完成任务,还是需要修改main方法为:Killer jingKe = new Killer2();。没有错,不过你可以通过一个工厂来完成上面的任务。也就是说,不通过new语句来获得Killer对象,而是通过工厂来获得Killer对象。
  48. public class KillerFactory {
  49. public static killer getKiller() {
  50. return new Killer1();
  51. }
  52. public static Killer getKiller(String path) throws Exception {
  53. Properties prop = new Properties();
  54. prop.load(new FileInputStream(path));
  55. String className = prop.getProperty("killer");
  56. Class clazz = Class.forName(className);
  57. return (Killer)clazz.newInstance();
  58. }
  59. }
  60. 代码确实增加了很多,可是这对后期的系统维修和系统升级带来了很多好处。
  61. 水龙头与出水管的关系,我们需要把水龙头安装到出水管上。如果有一天我们需要更换水龙头时,只需要把老的水龙头拆卸下来,把新的水龙头安装到出水管上即可。如果水龙头与出水管是一体的,就是无法拆卸的怎么办呢?或是说出水管只能安装水龙头,而不能安装淋浴器,这就使我们生活很不方便。我们可以理解为出水管的连接方法,连接的对象是“出水设备”,而这个“出水设备”是一个接口。而水龙头与淋 浴器都是这个接口的实现类。但是接口在哪里呢?它长什么样子?我们没看到。它是一个标准,连接处的内径与外径。螺丝抠的密度等。这就和你的电脑上为什么可以连接USB设备一样。如果电脑和某一个USB设
  62. 备电焊到一起,那么其它的USB设备就无法使用了。电脑使用的是实现了USB接口的电子设备,而我们的U盘、MP3、移动硬盘及鼠标都是USB接口的实现类。

  63. 用Java写出来的程序也和我们现实生活中的设备一样。如电脑,我们希望电脑的所有部件都可以更换,如果主板上的内存插槽与内存条不附。我们说,内存条没有实现某某接口。Java是完全面向对象的,而面向对象是我们最为熟悉的东东。面向对象并不简单,而是我们太熟悉它了。所以我们学习Java会很方便。在现实社会中,所有的标准到了Java中都是接口。一盒香烟多少支烟,烟的长度等都是标准。一个光盘的大 小。Java中的JDBC就是一个标准,而各大数据库厂商及第三方厂商实现了这一标准。JDBC只说明了目的,没有说明怎么完成的目的。

  64. 面向对象就在我们眼前,不过我们有时不太注意它。希望你在今后学习Java时,多与现实社会联系。这样可以有利与你的理解。代码量加大了,但对后期的维护与升级提供了方便。软件公司卖给客户的是class文件,而不是java文件。如果你的客户需要更换Killer对象,只需修改资源文件既可。

  65. 下面我们来看一个定时器类。现在什么都是自动化的,如空调、电视、洗衣机等等。都要用到定时器这个类。那我们可不可以只写一个定时器类,应用到所有的需要定时器的设备上呢?答案是肯定的,我们需要这个类。

  66. 好了,我们来分析一下定时器需要完成什么工作吧。定时器应该有启动、停止方法。定时器启动之后,每过一定时间就执行某个动作。其中时间间隔为long型,而要执行的动作可能是输出一个字符串,也可能是打印作业。具体要干什么由使用定时器的用户来完成。而定义定时器时,根本就不知道要干什么。
  67. public class Timmer {
  68. private long time;
  69. private Action action;
  70. public Timmer() {}
  71. public Timmer(long time, Action action) {
  72. this.time = time;
  73. this.action = action;
  74. }
  75. public void start() {
  76. state = START;
  77. if(th == null) {
  78. th = new Thread() {
  79. public void run() {
  80. while(state == START) {
  81. try {
  82. Thread.sleep(time);
  83. action.action();
  84. } catch(Exception e) {
  85. }
  86. }
  87. }
  88. };
  89. }
  90. th.start();
  91. }
  92. public void stop() {
  93. state = STOP;
  94. }
  95. public void setTime(long time) {
  96. this.time = time;
  97. }
  98. public void setAction(Action action) {
  99. this.action = action;
  100. }
  101. public long getTime() {
  102. return (this.time);
  103. }
  104. public Action getAction() {
  105. return (this.action);
  106. }
  107. }
  108. Action是一个接口,它只有一个方法,就是要完成的任务。我们在定时器启动时调用这个接口的方法。而这个Action接口的对象,代表一个动作,这个动作就是用户要完成的动作。
  109. public interface Action {
  110. public void action();
  111. }
  112. public static void main(String[] args) {
  113. Timer t = new Timer(2000, new Action() {
  114. public void action() {
  115. System.out.println("Hello World!");
  116. }
  117. });
  118. t.start();
  119. javax.swing.JOptionPane.showMessageDialog(null, "点击确定按钮停止定时器");
  120. t.stop();
  121. System.exit(0);
  122. }
  123. 这是一个典型的回调方法例子。在AWT中,java使用了大量的监听器。这些监听器都是回调方法。在XML解析器SAX中,也使用了回调方法来解析XML文档。

  124. 接口要比抽象类还要抽象。抽象类需要子类继承,而Java是单继承,所以抽象类被限制了。而接口不同,一个类可以实现多个接口。好比人类与程序员类之间的关系。可以说程序员是人类的子类,如果程序员是一个接口。用人类的子类来实现它就会更好。这个子类还可以去实现会计接口、音乐家接口等等。

  125. 在struts2.0、spring、hibernate等框架中,都大量使用接口。我们关心的是某个接口与另一个接口之间的关系。而不关心某个实现类与另一个接口实现类的关系。在客观世界中,我们交谈时都大量使用接口,只是我们没有注意罢了。如:我公司需要一个程序员(一个实现了程序员接口的对象)。上课时,讲师有一台计算机,用白板笔在白板上写字(计算机是接口,白板及白板笔也是接口)。讲师希望学生能学会所有的知识(讲师及学生都是接口)。
复制代码



作者: 一休    时间: 2015-6-12 21:55
这些东西挺抽象的  需要多写些例子来体会
作者: zhao_HHH    时间: 2015-6-12 21:58
受益了,谢谢
作者: 痞子刘忙    时间: 2015-6-12 22:42
恩,java面向对象的体现操作就是体现在接口上,效果还是很好的。后期容易维护和跟进。
作者: 嘎路的米    时间: 2015-6-12 22:49
嗯,学到了
作者: TOBESKY    时间: 2015-6-12 23:02
还是要多练才行
作者: 蓝菊蚊香2    时间: 2015-6-12 23:16
不错不错~
作者: 流水0215    时间: 2015-6-13 08:13
学习学习




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