JAVA编程思想(3) - 复用类(二)

2024-06-05 21:38
文章标签 java 复用 编程 思想

本文主要是介绍JAVA编程思想(3) - 复用类(二),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

组合和继承之间选择

  • 组合和继承都允许在新的类中放置子对象,组合是显示地这么做的,而继承是隐式地做。
  • 组合技术通常用于想在新类中使用现有类的功能而非它的接口这种情况。在新类中我们只能看到的是为新类所定义的接口,而非所嵌入对象的接口。为取得这个效果,需要在新类中嵌入一个现有类的private对象。
  • 有时,允许类的用户直接访问新类中组合成分是极据意义的,声明为public(一般情况下是private);如果成员对象自身都隐藏了具体实现,那么这种做法是安全的
  • “is-a”(是一个)的关系是用继承来表达,而has-a(有一个)的关系是用组合来表达的。

protected关键字

  • 在实际项目中,经常会想要将某些事物尽可能对这个世界隐藏起来,但仍然允许导出类的成员访问它们。
  • 关键字protected它指明“就类用户而言,这是private的,但对于任何继承此类的导出类或其他任何位于同一个包内的类来说,它却是可以访问的。”(protected也提供了包内访问权限。)
  • 尽管可以创建protected域,但是最好的方式还是将域保持为private;你应当一直保留“修改底层实现”的权力,然后通过protected方法来控制类的继承者的访问权限。

向上转型

  • “新类是现有类的一种类型”这句话用来概括新类和基类之间的关系。
//: reusing/Wind.java
// Inheritance & upcasting.class Instrument {public void play() {}static void tune(Instrument i) {// ...i.play();}
}// Wind objects are instruments
// because they have the same interface:
public class Wind extends Instrument {public static void main(String[] args) {Wind flute = new Wind();Instrument.tune(flute); // Upcasting}
} ///:~
  • 在此例中,tune()方法可以接受Instrument的引用。但在Wind.main()中,传递的给tune()方法的是一个Wind引用,java会严格检查类型,除非你认识到Wind对象同样是一种Instrument对象,而且也不存在任何tune()方法是可以通过Instrument来调用的,同时又不存在与Wind之中。将导出类的引用转换为基类的引用的动作,叫向上转型
  • 由导出类转型为基类,在继承图上是向上移动的,因此一般称为向上转型。由于向上转型是从一个较专用类型向比较通用的类型转换,所以总是安全的。也就是说导出类是基类的一个超集。它可能会有跟多的方法,但是至少具备基类中所含的方法。在向上转型的过程中,类接口唯一可能发生的事情是丢失方法,而不是获取它们,这就是为什么编译器在“未曾”明确表示转型未曾指定特殊标记的情况下,仍然支持这种行为。
再论组合与继承
  • 尽管在教授OOP的过程中我们多次强调继承,但这并不意味着尽可能使用它。相反,应当慎用这一技术,其使用场合仅限于你确信使用该技术确实有效的情况下。使用组合还是继承取决于你:你是否需要向上转型

final关键字

  • 根据上下文环境,Java的关键字final的含义存在着细微的区别,但通常它指的是“这是无法改变的”。不想做改变的可能出于两个理由:设计或效率,由于这两个原因相差很远,所以关键字final有可能被误用。
  • final可能使用到的三种情况:数据、方法和类。
final数据
  • 有时数据的恒定不变是很有用的,比如:
    1. 一个永不改变的编译时常量
    2. 一个在运行时被初始化的值,而你不希望它被改变。
  • 对于编译期常量这种情况,编译器可以将该常量值带入任何可能用到它的计算式中,减轻了一些运行时的负担。在Java中,这类常量必须是基本数据类型,并且以关键字final表示。在对这个常量进行定义的时候,必须对其进行赋值
  • 一个既是static又是final的域只占据一段不能改变的存储空间。
  • 当对对象引用而不是基本类型运行final时,与对于基本类型不同,基本类型是使数值不变,而引用是使引用恒定不变。一旦引用被初始化指向一个对象,就无法再把它改为指向另一个对象。然而,对象其本身是可以修改的。
空白final
  • Java允许生成“空白的final”,所谓空白的final是指被声明为final但又未给定初值的域。无论什么情况,编译器都会确保空白的final在使用前必须被初始化。但是空白final在关键字final的使用上提供个更大的灵活性。
//: reusing/BlankFinal.java
// "Blank" final fields.class Poppet {private int i;Poppet(int ii) { i = ii; }
}public class BlankFinal {private final int i = 0; // Initialized finalprivate final int j; // Blank finalprivate final Poppet p; // Blank final reference// Blank finals MUST be initialized in the constructor:public BlankFinal() {j = 1; // Initialize blank finalp = new Poppet(1); // Initialize blank final reference}public BlankFinal(int x) {j = x; // Initialize blank finalp = new Poppet(x); // Initialize blank final reference}public static void main(String[] args) {new BlankFinal();new BlankFinal(47);}
} ///:~
  • 必须在域的定义处或者每个构造器中用表达式对final进行赋值,这正是final域在使用前总是被初始化的原因。
final参数
  • Java允许在参数列表中以声明的方式将参数指明为final。这意味着你无法在方法中更改参数引用所指的对象。
  • 你可以阅读参数,但却无法修改参数。
final方法
  • 使用final方法的原因有两个。第一个原因是把方法锁定,以防任何继承类修改它的含义。这是出于设计的考虑:想要确保在继承中使方法行为保持不变,并且不会被覆盖。
  • 过去建议使用final方法的第二个原因是效率。不过在最近的Java版本中,这种做法正在逐渐的受阻,已经不再需要使用final方法来进行优化了,只有在想要明确禁止覆盖时,才将方法设置为final
final和private关键字
  • 类中所有的private方法都隐式地指定为是final的。由于无法取用private方法,所以也就无法覆盖它。可以对private方法添加final修饰词,但这并不能给该方法增加任何额外的意义。
//: reusing/FinalOverridingIllusion.java
// It only looks like you can override
// a private or private final method.
import static net.mindview.util.Print.*;class WithFinals {// Identical to "private" alone:private final void f() { print("WithFinals.f()"); }// Also automatically "final":private void g() { print("WithFinals.g()"); }
}class OverridingPrivate extends WithFinals {private final void f() {print("OverridingPrivate.f()");}private void g() {print("OverridingPrivate.g()");}
}class OverridingPrivate2 extends OverridingPrivate {public final void f() {print("OverridingPrivate2.f()");}public void g() {print("OverridingPrivate2.g()");}
}public class FinalOverridingIllusion {public static void main(String[] args) {OverridingPrivate2 op2 = new OverridingPrivate2();op2.f();op2.g();// You can upcast:OverridingPrivate op = op2;// But you can't call the methods://! op.f();//! op.g();// Same here:WithFinals wf = op2;//! wf.f();//! wf.g();}
} /* Output:
OverridingPrivate2.f()
OverridingPrivate2.g()
*///:~
  • “覆盖”只有在某方法是基类的接口的一部分时才会出现。即:必须将一个对象向上转型为它的基本类型并调用相同的方法。如果某方法是为private,它就是不是类的接口的一部分,它仅是一些隐藏于类中的程序代码,只不过是具有相同的名称而已。但如果导出类中以相同的名称生成一个publicprotected或包访问权限方法的话,该方法就不会产生在基类中出现的“仅具有相同名称”的情况。此时你并没有覆盖该方法,仅是生成了一个新的方法。由于private无法触及而且能有效隐藏,所以除了把它看成是因为它所属类的组织结构的原因而存在外,其他任何事情不需要考虑到它。
final类
  • 当将某个类的整体定义为final时,就表明了你不打算继承该类,而且也不允许别人这么做。换句话说,出于某种考虑,你对该类的设计永不需要做任何改动,或者出于安全考虑,你不希望它有子类。
  • 无论类是否被定义为final,相同的规则都适用于定义为final的域。然而,由于final类禁止继承,所以final类中所有的方法都隐式指定为是final的,因为无法覆盖它们。在final类中可以给方法添加final修饰词,但这并没有添加任何意义。
初始化及类的加载
  • 在许多语言之中,程序是作为启动过程的一部分立刻被加载的。然后是初始化,紧接着程序开始运行。这些语言的初始化过程必须小心控制,以确保定义为static的东西,其初始化不会造成麻烦。例如C++中,如果某个static期望在另一个static在被初始化之前就能有效地使用它,那么就会出现问题。
  • Java不会出现这个问题,因为它采用的是一种不同的加载方式,因为Java中的所有事物都是对象,每个类的编译代码都存在于它自己的独立文件中,该文件只在需要使用程序代码的时候才会被加载。一般来说,可以说:“类的代码在初次使用时才被加载。”。这通常是指加载发生在创建类的第一个对象之时,但是当访问static域或者static方法时,也会发生加载。
  • 初次使用之处也是static初始化发生之处,所有的static对象和static代码段都会在加载时按程序中的顺序而依次初始化。当然这只会发生一次。
继承和初始化
//: reusing/Beetle.java
// The full process of initialization.
import static net.mindview.util.Print.*;class Insect {private int i = 9;protected int j;Insect() {print("i = " + i + ", j = " + j);j = 39;}private static int x1 =printInit("static Insect.x1 initialized");static int printInit(String s) {print(s);return 47;}
}public class Beetle extends Insect {private int k = printInit("Beetle.k initialized");public Beetle() {print("k = " + k);print("j = " + j);}private static int x2 =printInit("static Beetle.x2 initialized");public static void main(String[] args) {print("Beetle constructor");Beetle b = new Beetle();}
} /* Output:
static Insect.x1 initialized
static Beetle.x2 initialized
Beetle constructor
i = 9, j = 0
Beetle.k initialized
k = 47
j = 39
*///:~
  • 构造器也是static方法,尽管static关键字并没有显示的写出来。 (这个待日后深入理解)
  • Beetle上运行Java时,所发生的第一件事情就是试图访问Beetle.main()(一个static方法)于是加载器开始启动并找出Beetle类的编译代码(其实就是在名为Beetle.class的文件中)。对于这次加载,编译器注意到它有个基类(通过关键字extend得知的),于是它继续进行加载。不管你是否打算产生一个该基类的对象,这都要发生。
  • 如果该基类还有基类就继续加载,如此类推。接下来,根基类中的static初始化即会被执行(这里是Insect),然后是下一个导出类,依次类推。
  • 至此为止,必要的类都被加载完成了,对象可以被创建了。首先,对象中的所有基本类型都会被设为默认值,对象引用被设为null,然后基类的构造器会被调用,在基类的构造器完成之后,实例变量按次序被初始化,最后,构造器的其余部分被执行。

这篇关于JAVA编程思想(3) - 复用类(二)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1034244

相关文章

Java五子棋之坐标校正

上篇针对了Java项目中的解构思维,在这篇内容中我们不妨从整体项目中拆解拿出一个非常重要的五子棋逻辑实现:坐标校正,我们如何使漫无目的鼠标点击变得有序化和可控化呢? 目录 一、从鼠标监听到获取坐标 1.MouseListener和MouseAdapter 2.mousePressed方法 二、坐标校正的具体实现方法 1.关于fillOval方法 2.坐标获取 3.坐标转换 4.坐

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

java8的新特性之一(Java Lambda表达式)

1:Java8的新特性 Lambda 表达式: 允许以更简洁的方式表示匿名函数(或称为闭包)。可以将Lambda表达式作为参数传递给方法或赋值给函数式接口类型的变量。 Stream API: 提供了一种处理集合数据的流式处理方式,支持函数式编程风格。 允许以声明性方式处理数据集合(如List、Set等)。提供了一系列操作,如map、filter、reduce等,以支持复杂的查询和转

Java面试八股之怎么通过Java程序判断JVM是32位还是64位

怎么通过Java程序判断JVM是32位还是64位 可以通过Java程序内部检查系统属性来判断当前运行的JVM是32位还是64位。以下是一个简单的方法: public class JvmBitCheck {public static void main(String[] args) {String arch = System.getProperty("os.arch");String dataM

详细分析Springmvc中的@ModelAttribute基本知识(附Demo)

目录 前言1. 注解用法1.1 方法参数1.2 方法1.3 类 2. 注解场景2.1 表单参数2.2 AJAX请求2.3 文件上传 3. 实战4. 总结 前言 将请求参数绑定到模型对象上,或者在请求处理之前添加模型属性 可以在方法参数、方法或者类上使用 一般适用这几种场景: 表单处理:通过 @ModelAttribute 将表单数据绑定到模型对象上预处理逻辑:在请求处理之前

eclipse运行springboot项目,找不到主类

解决办法尝试了很多种,下载sts压缩包行不通。最后解决办法如图: help--->Eclipse Marketplace--->Popular--->找到Spring Tools 3---->Installed。

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

Java面试题:通过实例说明内连接、左外连接和右外连接的区别

在 SQL 中,连接(JOIN)用于在多个表之间组合行。最常用的连接类型是内连接(INNER JOIN)、左外连接(LEFT OUTER JOIN)和右外连接(RIGHT OUTER JOIN)。它们的主要区别在于它们如何处理表之间的匹配和不匹配行。下面是每种连接的详细说明和示例。 表示例 假设有两个表:Customers 和 Orders。 Customers CustomerIDCus