Java代码复用的三种常用方式:继承、组合和代理(2)

简介: Java代码复用的三种常用方式:继承、组合和代理

03、 代理


还有一种复用方法是代理——在新类中创建代理,通过代理来操作已有类的非 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的方法和属性直接暴露在新类中);与此同时,代理拿到了足够的好处。


04、final


作为代码的生产者来说,我们有时候希望代码被复用,有的时候又希望代码不被复用。当我们不想代码被复用时,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("我就是最终形态,别继承我!");
  }
}


相关文章
|
1月前
|
Java
在Java中,接口之间可以继承吗?
接口继承是一种重要的机制,它允许一个接口从另一个或多个接口继承方法和常量。
96 1
|
2月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
32 3
|
2月前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
56 2
|
2月前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
41 2
|
1月前
|
Java
JAVA 静态代理 & 动态代理
【11月更文挑战第14天】静态代理是一种简单的代理模式实现,其中代理类和被代理类的关系在编译时已确定。代理类实现与被代理类相同的接口,并持有被代理类的实例,通过调用其方法实现功能增强。优点包括代码结构清晰,易于理解和实现;缺点是对于多个被代理类,需为每个类编写相应的代理类,导致代码量大增,维护成本高。动态代理则在运行时动态生成代理类,更加灵活,减少了代码冗余,但可能引入性能损耗和兼容性问题。
|
2月前
|
设计模式 Java API
[Java]静态代理与动态代理(基于JDK1.8)
本文介绍了代理模式及其分类,包括静态代理和动态代理。静态代理分为面向接口和面向继承两种形式,分别通过手动创建代理类实现;动态代理则利用反射技术,在运行时动态创建代理对象,分为JDK动态代理和Cglib动态代理。文中通过具体代码示例详细讲解了各种代理模式的实现方式和应用场景。
38 0
[Java]静态代理与动态代理(基于JDK1.8)
|
2月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
47 1
|
2月前
|
Java
Java访问外网图片地址时,如何添加代理?
【10月更文挑战第14天】Java访问外网图片地址时,如何添加代理?
49 2
|
2月前
|
Java 测试技术 编译器
Java零基础-继承详解!
【10月更文挑战第4天】Java零基础教学篇,手把手实践教学!
49 2
|
2月前
|
Java 编译器
在Java中,关于final、static关键字与方法的重写和继承【易错点】
在Java中,关于final、static关键字与方法的重写和继承【易错点】
31 5