复用代码是Java众多引人注目的功能之一。这句话很通顺,没什么问题,但问题在于不少人并不清楚“复用”是什么。就好像我说“沉默王二是一个不止会写代码的程序员”,唉,沉默王二是谁?程序员
若是想学习Java工程化、高性能及分布式、深刻浅出。微服务、Spring,MyBatis,Netty源码分析的朋友能够加个人Java高级交流:854630135,里面有阿里大牛直播讲解技术,以及Java大型互联网技术的视频免费分享给你们。分布式
咱们须要来给“复用”下一个定义。复用,说白了就是重复使用。微服务
举个例子,不少名人说了不少名言,咱们在说话、写做的时候,就常常有意无心的重复这些名言。好比说我,就特别喜欢重复使用王小波的那句名言:“从话语中,你不多能学到人性,从沉默中却能。假如还想学得更多,那就要继续一声不吭 。”源码分析
上面这个例子,只能说是“复用”的一种低级的应用,其实就是复制粘贴了。还有高级的复用方式吗?性能
有,固然有。Java做为一种优秀的面向对象设计的语言,在复用的应用上就高级得多了。学习
最多见的复用方法就是继承——使用extends
关键字在基类的基础上建立新类,新类能够直接复用基类的非private
的属性和方法;就像程序清单1-1那样。this
程序清单1-1:spa
public class Wangxiaosan extends Wangsan { public Wangxiaosan() { System.out.println("我是新类王小三"); setName("王老三"); System.out.println(getName()); } public static void main(String[] args) { new Wangxiaosan(); } } class Wangsan { private String name; Wangsan() { System.out.println("我是基类王三"); } public String getName() { return name; } public void setName(String name) { this.name = name; } }
从程序清单1-1中咱们能够看得出,getName()和setName()方法虽然是在基类Wangsan中建立的,但能够在新类Wangxiaosan中使用,代码的复用工做就这样轻松地完成了。设计
另一种常见的复用方法就是组合——在新类中建立已有类的对象,经过该对象来调用已有类中的非private的属性和方法;就像程序清单2-1那样。代理
程序清单2-1:
public class Tongxiangyu { private Baizhantang boyFriend = new Baizhantang(); public Tongxiangyu() { System.out.println("我是同福客栈的掌柜佟湘玉"); boyFriend.pointHand("郭芙蓉"); } public static void main(String[] args) { new Tongxiangyu(); } } class Baizhantang { Baizhantang() { System.out.println("我是退隐江湖的盗圣白展堂"); } public void pointHand(String name) { System.out.println("那谁" + name + ",准备一下——葵花点穴手"); } }
从程序清单2-1中咱们能够看得出,葵花点穴手虽然是白展堂的绝技,但做为佟掌柜的男友,佟掌柜要展堂点个穴,展堂也是不敢辞让的。你看,佟掌柜虽然是个弱女子,但自从有了展堂这个武功首屈一指的男友,再没有谁敢不听话啊——厉害的组合啊。
须要注意的是,如何在继承和组合之间作出选择呢?
若是新类和已有类须要具备一些类似的方法和属性时,就采用继承的形式;若是新类只是为了借用已有类的一些方法和属性时,而二者没有不少类似之处时就须要采用组合的形式。
还有一种复用方法是代理——在新类中建立代理,经过代理来操做已有类的非private的属性和方法;就像程序清单3-1那样。
程序清单3-1:
public class Member { public static void main(String[] args) { Proxy proxy = new Proxy(); System.out.println("代理说一个药丸十五块"); proxy.buy(15); } } class Proxy { private Shop shop = new Shop(); public void buy(int money) { System.out.println("一个药丸十五块"); shop.sale(money - 5); } } class Shop { public void sale(int money) { System.out.println("一个药丸十块钱"); } }
从程序清单3-1中咱们能够看得出,代理的模式和组合有点相似,但又有差异——代理成功的隔开了新类(会员)和已有类(店铺)的直接关系,使得已有类的方法不直接暴露在新类面前(组合的方式会将已有类的非private的方法和属性直接暴露在新类中);与此同时,代理拿到了足够的好处。
做为代码的生产者来讲,咱们有时候但愿代码被复用,有的时候又但愿代码不被复用。当咱们不想代码被复用时,final关键字就派上用场了。final这个关键字很形象,它自己就说明了一切——最后的,最终的;决定性的;不可更改的。
使用final的场景有三种,分别是数据、方法和类。咱们来稍做说明。
1)final 数据
最多见的final数据就是常量了,例如:
public class Consts { public static final String CMOWER = "沉默王二"; }
对于常量来讲,它对于整个应用内的全部类都是可见的,所以是public的;它能够直接经过类名.常量名访问,因此是static的;它是不可修改的,所以是final的。
另一种常见的final数据就是参数了,参照程序清单4-1。
程序清单4-1:
public class Cmower { public void write(final String content) { // content += "犹未雪"; // final修饰的参数是没法在方法内部被再次修改的 System.out.println(content); } public void write1(String content) { content += "犹未雪"; System.out.println(content); } public static void main(String[] args) { Cmower cmower = new Cmower(); cmower.write("精忠报国"); cmower.write1("靖康耻"); } }
2)final 方法
在Java类中,全部的private方法都隐式地指定为final的(也就是说,若是你在private方法上加上final修饰符,实际上是没啥意义的)。在介绍继承的时候,你应该注意到我强调的一句话,就是新类能够直接复用基类的非private的属性和方法,也就是说private方法是没法被继承者修改的,由于private方法是final的。
来看程序清单4-2,你会发现Wangsan类型的san引用是不能调用say(String words)方法的,由于private方法是没法被继承者修改的,尽管Wangxiaosan中从新定义了say(String words)方法。
程序清单4-2:
public class Wangxiaosan extends Wangsan { public Wangxiaosan() { say("吃中饭没"); } public void say(String words) { System.out.println("王小三在说:" + words); } public static void main(String[] args) { Wangsan san = new Wangxiaosan(); // san.say("吃晚餐没"); // 没法访问,并不会被覆盖 } } class Wangsan { public Wangsan() { say("吃早饭没"); } private void say(String words) { System.out.println("王三在说:" + words); } }
3)final 类
当咱们认为某个类就是最终的形态了,它很完美,不该该被继承,就可使用final关键字来修饰;参照程序清单4-3。
程序清单4-3:
// 没法继承 public class Wangxiaosan extends Wangsan { } final class Wangsan { public Wangsan() { System.out.println("我就是最终形态,别继承我!"); } }