女皇武则天:我不愿被 extends

01、
利用继承,我们可以基于已存在的类构造一个新类。继承的好处在于,子类可以复用父类的非 private 的方法和非 private 成员变量。
is-a 是继承的一个明显特征,就是说子类的对象引用类型可以是一个父类。我们可以将通用的方法和成员变量放在父类中,达到代码复用的目的;然后将特殊的方法和成员变量放在子类中,除此之外,子类还可以覆盖父类的方法。这样,子类也就焕发出了新的生命力。
一个对象变量可以引用多种类型的现象被称为多态。多态发生的前提条件就是继承。也就是说,先有继承,后有多态。
class Wanger {
public void write() {
System.out.println("我为自己活着");
}
}
class Wangxiaoer extends Wanger {
public void write() {
System.out.println("我也为自己活着");
}
}
class Test {
public static void main(String [] args) {
Wanger wanger;
wanger = new Wanger();
wanger = new Wangxiaoer();
Wangxiaoer wangxiaoer;
//wangxiaoer = new Wanger(); // 不可以
wangxiaoer = new Wangxiaoer(); // 只能这样
}
}
wanger 这个对象变量既可以引用 Wanger 对象,也可以引用 Wangxiaoer对象。但 wangxiaoer 就只能引用 Wangxiaoer 对象,不能引用 Wanger 对象。根本的原因在于 Wangxiaoer 是 Wanger 的继承者。
当使用 wanger 调用 write() 方法时,程序会在运行时自动识别其引用的对象类型,然后选择调用哪个方法——这种现象称为动态绑定。
动态绑定有一个非常重要的特性:无需对现有的代码进行修改,就能对程序进行扩展。假如 Wangdaer 也继承了 Wanger,并且 wanger 引用了Wangdaer 的对象,那么 wanger.write() 仍然可以正常运行。
当然了,有些类不愿意被继承,也没法被继承。谁不愿意被继承呢?比如武则天,亲手弄死自己的亲儿子。谁没法被继承呢,每朝每代最后的那位倒霉皇帝。
类怎么做到不被继承呢?可以使用 final 关键字。final 关键字修饰的类不能被继承,final 修饰的方法不能被覆盖。
final class Wanger {
public final void write() {
System.out.println("你们谁都别想继承我");
}
}
继承是面向对象编程当中举足轻重的一个概念,与多态、封装共为面向对象的三个基本特征。 继承可以使得子类具有父类的成员变量和方法,还可以重新定义、追加成员变量和方法等。
在设计继承的时候,可以将通用的方法和成员变量放在父类中。但不建议随心所欲地将成员变量以 protected 的形式放在父类当中;尽管允许这样做,并且子类可以在需要的时候直接访问,但这样做会破坏类的封装性(封装要求成员变量以 private 的形式出现,并且提供对应 getter / setter 用来访问)。
Java 是不允许多继承的,为什么呢?
如果有两个类共同继承一个有特定方法的父类,那么该方法会被两个子类重写。然后,如果你决定同时继承这两个子类,那么在你调用该重写方法时,编译器不能识别你要调用哪个子类的方法。
这也正是著名的菱形问题,见下图。ClassC 同时继承了 ClassA 和 ClassB,ClassC 的对象在调用 ClassA 和 ClassB 中重载的方法时,就不知道该调用 ClassA 的方法,还是 ClassB 的方法。

02、
在 Java 中,所有类都由 Object 类继承而来。Object 这个单词的英文意思是对象,是不是突然感觉顿悟了——万物皆对象?没错,Java 的设计者真是良苦用心了啊!现在,你一定明白了为什么 Java 是面向对象编程语言的原因。
你可能会疑惑地反问道:“我的类明明没有继承 Object 类啊?”如果一个类没用显式地继承某一个类,那么它就会隐式地继承 Object 类。换句话说,不管是鸡生了蛋,还是蛋孵出了鸡,总有一只 Object 鸡或者一个 Object 蛋。
在面试的时候,你可能会被问到这么一个问题:“Object 类包含了哪些方法呢?”
1)protected Object clone() throws CloneNotSupportedException 创建并返回此对象的副本。
不过,《阿里巴巴 Java 开发手册》上建议:慎用 Object 的 clone 方法来拷贝对象。因为 Object 的 clone 方法默认是浅拷贝,如果想实现深拷贝需要重写 clone 方法实现属性对象的拷贝。
什么是浅拷贝,什么是深拷贝呢?
浅拷贝是指在拷贝对象时,会对基本数据类型的变量重新复制一份,而对于引用类型的变量只拷贝了引用,并没有对引用指向的对象进行拷贝。
深拷贝是指在拷贝对象时,同时对引用指向的对象进行拷贝。
浅拷贝和深拷贝的区别就在于是否拷贝了对象中的引用变量所指向的对象。
2)public boolean equals(Object obj) 判断另一对象与此对象是否「相等」。
该方法使用的区分度最高的“==”操作符进行判断,所以只要两个对象不是同一个对象,那么 equals() 方法一定返回 false。
《阿里巴巴 Java 开发手册》上强调:由于 Object 的 equals 方法容易抛出空指针异常,所以应该使用常量或者确定不为 null 的对象来调用 equals。
正例:"test".equals(object);
反例:object.equals("test");
在正式的开发项目当中,最经常使用该方法进行判断的就是字符串。不过,建议使用org.apache.commons.lang3.StringUtils,不用担心出现空指针异常。具体使用情况如下所示:
StringUtils.equals(null, null) = true
StringUtils.equals(null, "abc") = false
StringUtils.equals("abc", null) = false
StringUtils.equals("abc", "abc") = true
StringUtils.equals("abc", "ABC") = false
3)public native int hashCode() 返回此对象的哈希码。hashCode() 是一个 native 方法,而且返回值类型是整形;实际上,该方法将对象在内存中的地址作为哈希码返回,可以保证不同对象的返回值不同。
A native method is a Java method whose implementation is provided by non-java code.
native方法是一个Java调用非Java代码的接口。该方法的实现由非Java语言实现,比如 C。这个特征并非Java所特有,其它的编程语言也有这个机制,比如C++。
hashCode() 通常在哈希表中起作用,比如 HashMap。
向哈希表中添加 Object 时,首先调用 hashCode() 方法计算 Object 的哈希码,通过哈希码可以直接定位 Object 在哈希表中的位置。如果该位置没有对象,可以直接将 Object 插入该位置;如果该位置有对象,则调用 equals() 方法比较这个对象与 Object 是否相等,如果相等,则不需要保存 Object;如果不相等,则将该 Object 加入到哈希表中。
4)protected void finalize() throws Throwable 当垃圾回收机制确定该对象不再被调用时,垃圾回收器会调用此方法。不过,fnalize 机制现在已经不被推荐使用,并且在 JDK 9 开始被标记为 deprecated(过时的)。
5)public final Class getClass() 返回此对象的运行时类。
当我们想知道一个类本身的一些信息(比如说类名),该怎么办呢?这时候就需要用到 Class 类,该类包含了与类有关的信息。请看以下代码:
Wanger wanger = new Wanger();
Class c1 = wanger.getClass();
System.out.println(c1.getName());
// 输出 Wanger
6)public String toString() 返回此对象的字符串表示形式。
《阿里巴巴 Java 开发手册》强制规定:POJO 类必须重写 toString 方法;可以使用 Eclipse 直接生成,点击 「Source」→「Generate toString」。示例如下:
class Wanger {
private Integer age;
@Override
public String toString() {
return "Wanger [age=" + age + "]";
}
}
重写 toString() 有什么好处呢?当方法在执行过程中抛出异常时,可以直接调用 POJO 的 toString() 方法打印其属性值,便于排查问题。
POJO(Plain Ordinary Java Object)指简单的 Java 对象,也就是普通的
JavaBeans,包含一些成员变量及其getter / setter,没有业务逻辑。有时叫做 VO (value - object),有时叫做 DAO (Data Transform Object)。
03、
Java 语言虽然号称“万物皆对象”,但基本类型是例外的——有 8 个基本类型不是 Object,包括 boolean、byte 、short、char、int、foat、double、long。
我们以 int 为例,来谈一谈 Integer 和 int 之间的差别。Integer 是 int 的包装器,int 是 Integer 的基本类型。Java 5 的时候,引入了自动装箱和自动拆箱功能(boxing / unboxing),Integer 和 int 可以根据上下文自动进行转换。
比如说:
Integer a = 100;
实际的字节码是(装箱和拆箱发生在 javac 阶段而不是运行时):
Integer a = Integer.valueOf(100);
Java 为什么要这么做呢?因为在实际的应用当中,Java 的设计者发现大部分数据操作都集中在有限的、较小的数值范围内。为了提升性能,Java 5 新增了静态工厂方法 valueOf():
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
那么 IntegerCache.low 和 IntegerCache.high 的值是多少呢?来看源码:
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[];
static {
// high value may be configured by property
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h;
cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++);
// range [-128, 127] must be interned (JLS7 5.1.7)
assert IntegerCache.high >= 127;
}
private IntegerCache() {}
}
这说明最小值是 -128,最大值是 127。现在,我们来看一个有意思的代码,看看它们会输出什么。
Integer a = 127, b = 127;
System.out.println(a == b); // 输出 true
Integer a1 = 128, b1 = 128;
System.out.println(a1 == b1); // 输出 false
Integer a2 = 127;
int b2 = 127;
System.out.println(a2 == b2); // 输出 true
Integer a3 = 128;
int b3 = 128;
System.out.println(a3 == b3); // 输出 true
我是怎么判断出来结果的呢?这需要借助 class 字节码,如下:
Integer a = Integer.valueOf(127);
Integer b = Integer.valueOf(127);
System.out.println(a == b);
Integer a1 = Integer.valueOf(128);
Integer b1 = Integer.valueOf(128);
System.out.println(a1 == b1);
Integer a2 = Integer.valueOf(127);
int b2 = 127;
System.out.println(a2.intValue() == b2);
Integer a3 = Integer.valueOf(128);
int b3 = 128;
System.out.println(a3.intValue() == b3);
1)a == b 为 true,是因为 127 刚好在 -128 到 127 的边界,会从 IntegerCache 中产生。
2)a1 == b1 为 false,是因为 128 刚好不在 -128 到 127 之间,a1 和 b1 都是重新 new 出来的 Integer 对象,两个对象之间是不会 == 的。
3)a2 == b2 和 a3 == b3 为 true是因为两个 int 值在比较。尽管源码是 Integer == int,但编译后的字节码实际是 Integer.intValue() == int(intValue() 返回的类型为 int),也就是两个基本类型在比较,它们的值是相等的,所以返回 true。
Integer a = int 是自动装箱;int a = Integer.intValue() 是自动拆箱。
《阿里巴巴 Java 开发手册》强制规定:所有相同类型的包装器对象之间值的比较,应该使用 equals() 方法比较。
04、
本篇,我们先谈了面向对象的重要特征继承;然后谈到了继承的终极父类 Object;最后谈到了“万物皆对象”的漏网之鱼基本类型。这些知识点都相当的重要,请务必深入理解!
女皇武则天:我不愿被 extends的更多相关文章
- Java:并发不易,先学会用
我从事Java编程已经11年了,绝对是个老兵:但对于Java并发编程,我只能算是个新兵蛋子.我说这话估计要遭到某些高手的冷嘲热讽,但我并不感到害怕. 因为我知道,每年都会有很多很多的新人要加入Java ...
- iBatis.net 类的继承extends和懒加载
<resultMaps> <resultMap id="FullResultMap" class="t_c_team_member_permission ...
- java关键字extends(继承)、Supe(父类引用空间)、 This(方法调用者对象)、Instanceof(实例类型-判断对象是否属于某个类)、final(最终)、abstract(抽象) 、interface(接口)0
java 继承使用关键字extends 继承的作用:减少代码量,优化代码 继承的使用注意点: 1子类不能继承父类的私有变量 2.子类不能继承父类的构造方法 3.子类在调用自己的构造方法时 会默认调 ...
- Java,extends,继承
1. 继承的概念: 继承在本职上是特殊--一般的关系.子类继承父类,表明子类是一种特殊的父类,并且具有父类所不具有的一些属性或方法. 2. 继承中的初始化顺序: 从类的结构上而言,其内部可以有如下四种 ...
- java里面interface,implement和extends的作用和用法
今天阅读<设计模式示例>,看到一段代码涉及到了interface,implements和extends,其实在C++中经常用到.今天特百度,比较了一下: interface是一个接口,类似 ...
- extends 和 implements
extends是继承类与类的 implements是实现接口的 类与类之间只支持单继承 接口与接口之间支持多继承
- es6 Class的继承extends & super
Class之间可以通过extends关键字,实现继承. 子类会继承父类的属性和方法. class Point { constructor(x, y) { this.x = x; this.y = y; ...
- Java 中extends与implements使用方法
Java 中extends与implements使用方法 标签: javaclassinterfacestring语言c 2011-04-14 14:57 33314人阅读 评论(7) 收藏 举报 分 ...
- Atitit.java图片图像处理attilax总结 BufferedImage extends java.awt.Image获取图像像素点image.getRGB(i, lineIndex); 图片剪辑/AtiPlatf_cms/src/com/attilax/img/imgx.javacutImage图片处理titit 判断判断一张图片是否包含另一张小图片 atitit 图片去噪算法的原理与
Atitit.java图片图像处理attilax总结 BufferedImage extends java.awt.Image 获取图像像素点 image.getRGB(i, lineIndex); ...
随机推荐
- es6(二):解构赋值
ES中允许按照一定格式从数组,对象值提取值,对变量进行赋值,这就是解构(Destructuring) let [a,b,c]=[1,10,100] console.log(a,b,c)//1 10 1 ...
- 用calc()绘制手机图案解锁的九宫格样式
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Spark学习笔记
Map-Reduce 我认为上图代表着MapReduce不仅仅包括Map和Reduce两个步骤这么简单,还有两个隐含步骤没有明确,全部步骤包括:切片.转换.聚合.叠加,按照实际的运算场景上述步骤可以简 ...
- Leetcode_删除排序数组中的重复项
Leetcode 删除排序数组中的重复项 题目: 给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度. 不要使用 额外的数组空间,你必须在原地修改输入数 ...
- C# 中Datatime类用法总结
C# 中Datatime类用法总结 收集了一些记录下来,这些有的是从网上找的,有些是自己使用到的: DateTime dt = DateTime.Now; dt.ToString();//2005-1 ...
- Flask入门之Pycharm写Hello Word
在这里记录整理Flask入门的笔记! 今天讲讲使用Pycharm写一个Hello World 前提是已经安装好Pycharm,安装过程很简单,这里不赘述. 只放两个可以提供注册码的网站: Intell ...
- spring cloud 入门系列七:基于Git存储的分布式配置中心
我们前面接触到的spring cloud组件都是基于Netflix的组件进行实现的,这次我们来看下spring cloud 团队自己创建的一个全新项目:Spring Cloud Config.它用来为 ...
- Scrapy爬虫框架补充内容三(代理及其基本原理介绍)
前言:(本文参考维基百科及百度百科所写) 当我们使用爬虫抓取数据时,有时会产生错误比如:突然跳出来了403 Forbidden 或者网页上出现以下提示:您的ip访问频率太高 或者时不时跳出一个验证码需 ...
- city-picker插件使用-移动h5三级联动
首先访问该链接:http://www.jq22.com/jquery-info12914 看看是否是你要找的三级联动插件,(主要看注释的部分!) 好了,不知道是不是我傻,没有找到初始化数据的方法,本人 ...
- springboot数据库连接池使用策略
springboot官方文档介绍数据库连接池的使用策略如下: Production database connections can also be auto-configured using a p ...