继承
1 继承extends
1.1 什么是继承,有什么用?
继承:在现实世界当中也是存在的,例如:父亲很有钱,儿子不用很努力也会很有钱。
继承的作用:
基本作用:子类继承父类,代码可以得到复用。(这个不是最重要的,是基本作用。)
主要(重要)作用:因为有了继承关系,有了后期的方法覆盖和多态机制。
1.2 继承的相关特性
1 B类继承A类,则称A类为超类(superclass)、父类、基类,B类则称为子类(subclass)、派生类、扩展类。
class A{}
class B extends A{}
我们平时聊天说的是比较多的是:父类和子类。
superclass 父类
subclass 子类
2 java中的继承只支持单继承,不支持多继承,C++中支持多继承,这也是java体现简单性的一点,换句话说。
java中不允许这样写代码:class B extends A,C{ }。这是错误的。
3 虽然java中不支持多继承,但有的时候会产生间接继承的效果。
例如:class C extends B,class B extends A,也就是说,C直接继承B,其实C还间接继承A。
4 java中规定,子类继承父类,除构造方法不能继承之外,剩下都可以继承。
但是私有的属性无法在子类中直接访问。(父类中private修饰的不能在子类中直接访问,可以通过间接的手段访问。)
5 java中的类没有显示的继承任何类,则默认继承Object类,Object类是java语言提供的根类(老祖宗类),也就是说,一个对象与生俱来就有Object类型中所有的特征。
6 继承也存在一些缺点,例如:CreditAccount类继承Account类会导致它们之间的耦合度非常高,Account类发生改变之后会马上影响到CreditAccount类。
7 测试:子类继承父类之后,能使用子类对象调用父类方法吗?
可以,因为子类继承了父类之后,这个方法就属于子类了。
当然可以使用子类对象来调用。
8 在实际开发中,满足什么条件的时候,我可以使用继承?
凡是采用“is a”能描述的,都可以继承
例如:
Cat is a Animal: 猫是一个动物
Dog is a Animal: 狗是一个动物
CreditAccount is a Accnount: 信用卡账户是一个银行账户。
....
假设以后的开发中有一个A类,有一个B类,A类和B类确实也有重复的代码,那么他们两个之间就可以使用继承吗?
不一定,还是要看一看他们之间是否能使用is a来描述。
class Customer{ // 客户
String name;//名字
// setter and getter
}
class product{ // 商品
String name;//名字
// setter and getter
}
class Product extends Customer{
}
以上的继承就属于很失败的。因为:Product is a Customer, 是有违伦理的。
1.3 任何一个类,没有显示继承任何类,默认继承Object,那么Object类当中有哪些方法呢?老祖宗为我们提供了哪些方法?
以后慢慢的大家一定要适应看JDK的源代码(多看看牛人写的程序自己才会变成牛人。)
先模仿后超越。
java为什么比较好学呢?
是因为Java内置了一套庞大的类库,程序员不需要从0开始写代码,程序员可以基于这套庞大的类库进行“二次”开发。(开发速度较快,
因为JDK内置的这套库实现了很多基础的功能。)
例如:String是SUN编写的字符串类、System是SUN编写的系统类。
这些类都可以拿来直接使用。
JDK源代码在什么位置?
F:\Java\jdk-13.0.2(你自己安装JDK的路径下)\lib\src.zip
xxx.java
xxx.class
String[] args
System.out.println();
你现在能看懂以下代码了吗?
System.out.println("Hello World");
System.out 中,out后面没有小括号,说明out是变量名。
另外System是一个类名,直接使用类名System.out,说明out是一个静态变量。
System.out 返回一个对象,然后采用“对象.”的方式访问println方法。
我们研究了以下Object类当中有很多方法,大部分看不懂,其中有一个叫做toString()的,我们进行了测试,发现:
System.out.println(引用);
当直接输出一个“引用”的时候,println()方法会先自动调用“引用.toString()”,然后输出toString()方法的执行结果。
没有继承的案例:
/* 1 思考一下 一下代码是不是有点臃肿了,是不是重复的很多。 分析一下一下程序有什么问题?代码臃肿。代码没有得到他该有的复用性。 */ public class ExtendsTest01{ public static void main(String[] args){ // 创建普通账户 Account a = new Account(); a.setActno("11111"); a.setBalance(1000000); System.out.println("用户:" + a.getActno() + "的余额是:" + a.getBalance()); // 创建信用账户 CreditAccount a1 = new CreditAccount(); a1.setActno("22222"); a1.setBalance(-1000000); a1.setCredit(100); System.out.println("用户:" + a1.getActno() + "的余额是:" + a1.getBalance() + "信誉度为:" + a1.getCredit()); } } // 银行账户类 // 银行的属性: 账号、余额 class Account{ // 属性 private String actno; private double balance; // 构造方法 // 无参 public Account(){ } // 有参 public Account(String actno,double balance){ this.actno = actno; this.balance = balance; } // getter and setter 方法 实例方法哦 public void setActno(String actno){ this.actno = actno; } public String getActno(){ return actno; } public void setBalance(double balance){ this.balance = balance; } public double getBalance(){ return balance; } } // 其他类型的账户:信用卡账户 // 账号、余额、信誉度 class CreditAccount{ // 属性: private String actno; private double balance; private double credit; // 构造方法 public CreditAccount(){ } public CreditAccount(String actno,double balance,double credit){ this.actno = actno; this.balance = balance; this.credit = credit; } // getter and setter 方法 实例方法哦 public void setActno(String actno){ this.actno = actno; } public String getActno(){ return actno; } public void setBalance(double balance){ this.balance = balance; } public double getBalance(){ return balance; } public void setCredit(double credit){ this.credit = credit; } public double getCredit(){ return credit; } }
继承之后的案例:
// 使用继承机制来解决代码问题 // 继承也是存在缺点的:耦合度高,父类修改,子类受牵连。 public class ExtendsTest02{ public static void main(String[] args){ // 创建普通账户 Account a = new Account(); a.setActno("11111"); a.setBalance(1000000); System.out.println("用户:" + a.getActno() + "的余额是:" + a.getBalance()); // 创建信用账户 CreditAccount a1 = new CreditAccount(); a1.setActno("22222"); a1.setBalance(-1000000); a1.setCredit(100); System.out.println("用户:" + a1.getActno() + "的余额是:" + a1.getBalance() + "信誉度为:" + a1.getCredit()); } } // 银行账户类 // 银行的属性: 账号、余额 class Account{ // 父类 // 属性 private String actno; private double balance; // 构造方法 // 无参 public Account(){ } // 有参 public Account(String actno,double balance){ this.actno = actno; this.balance = balance; } // getter and setter 方法 实例方法哦 public void setActno(String actno){ this.actno = actno; } public String getActno(){ return actno; } public void setBalance(double balance){ this.balance = balance; } public double getBalance(){ return balance; } } // 其他类型的账户:信用卡账户 // 账号、余额、信誉度 class CreditAccount extends Account{ // 子类 // 属性: private double credit; // 构造方法 public CreditAccount(){ } public void doSome(){ // 错误: actno 在 Account 中是 private 访问控制 // System.out.println(actno); // 间接访问 // System.out.println(this.getActno()); System.out.println(getActno()); } // getter and setter 方法 实例方法哦 public void setCredit(double credit){ this.credit = credit; } public double getCredit(){ return credit; } }
java可以实现多继承吗?的案例:
class A{ } class B{ } class C extends A{ } class D extends B{ } /* 语法错误 java只允许单继承。不允许多继承。java是简单的。c++支持多重继承。 c++更接近现实一点。因为在现实世界中儿子同时继承父母两方特征。 ExtendsTest03.java:17: 错误: 需要'{' class E extends A,B{ */ //class E extends A,B{ //} class X{ } class Y extends X{ } class M extends X{ } // 其实这说明了Z是继承了X和Y的。 // 这样描述:Z直接继承了Y,Z间接继承了X class Z extends Y{ }
子类继承了父类,可以直接调用父类中的方法吗?案例:
/* Z继承了Y Y继承了X X继承了Object Z对象具有Object对象的特征(基因)。 Object是所有类的超类。老祖宗。类体系结构中的根。 java这么庞大的一个继承结构,最顶点是:Object *//* 测试:子类继承父类之后,能使用子类对象调用父类方法吗? 实际上以上的这个问题问的有点蹊跷!!!!! 哪里蹊跷?“能使用子类对象调用父类方法” 本质上,子类继承父类之后,是将父类继承过来的方法归自己所有。 实际上调用的也不是父类的方法,是他子类自己的方法。(因为继承了,所以就属于他自己的了。) */ public class ExtendsTest04{ public static void main(String[] args){ // 创建子类对象 Cat c = new Cat(); //调用方法 c.move(); // 通过子类对象访问name可以吗? System.out.println(c.name); } } // 父类 // class Animal extends Object { class Animal{ // 名字(先不封装) String name = "xiaoHua";//默认值不是null,给一个xiaoHua // 提供一个动物移动的方法 public void move(){ System.out.println(name + "正在移动!"); } } // Cat子类 class Cat extends Animal{ }
默认继承Object,Object类中有哪些方法呢?(附Object在JDK中的源代码)案例:
// 默认继承Object,Object类中有哪些方法呢? /* package java.lang; import jdk.internal.HotSpotIntrinsicCandidate; public class Object { // 注意:当原码当中一个方法以“;”结尾,并且修饰符列表中有“native”关键字 // 表示底层调用C++写的dll程序(dll动态链接库文件) private static native void registerNatives(); // 静态代码块(类加载时执行,而且只执行一次。) static { // 调用registerNatives()方法。 registerNatives(); } // 无参数构造方法 @HotSpotIntrinsicCandidate public Object() {} // 底层也是调用C++ @HotSpotIntrinsicCandidate public final native Class<?> getClass(); // 底层也是调用C++ @HotSpotIntrinsicCandidate public native int hashCode(); // equals方法你应该能看懂。、 // public 公开的 // boolean 返回值 // equals 是一个方法名:相等 // obj 形参 public boolean equals(Object obj) { // 方法体 return (this == obj); } // 已有对象a,想创建一个和a一模一样的对象,你可以调用这个克隆方法。 // 底层也是调用C++ @HotSpotIntrinsicCandidate protected native Object clone() throws CloneNotSupportedException; // 一会我们可以测试一下toString() 方法 // public 表示公共的。 // String 是一个返回值类型,toString()方法执行结束后返回一个字符串。 // toString 这是方法名。 // () 表示形参个数为0 public String toString() { return getClass().getName() + "@" + Integer.toHexString(hashCode()); } @HotSpotIntrinsicCandidate public final native void notify(); @HotSpotIntrinsicCandidate public final native void notifyAll(); public final void wait() throws InterruptedException { wait(0L); } public final native void wait(long timeoutMillis) throws InterruptedException; public final void wait(long timeoutMillis, int nanos) throws InterruptedException { if (timeoutMillis < 0) { throw new IllegalArgumentException("timeoutMillis value is negative"); } if (nanos < 0 || nanos > 999999) { throw new IllegalArgumentException( "nanosecond timeout value out of range"); } if (nanos > 0 && timeoutMillis < Long.MAX_VALUE) { timeoutMillis++; } wait(timeoutMillis); } @Deprecated(since="9") protected void finalize() throws Throwable { } } */ public class ExtendsTest05{ // ExtendsTest05 默认继承Object // ExtendsTest05 类当中是有toString()方法的 // 不过toString()方法是一个实例对象,需要创建对象才能调用。 /* public String toString() { return getClass().getName() + "@" + Integer.toHexString(hashCode()); } */ public static void main(String[] args){ // 分析这个代码可以执行吗? //ExtendsTest05.toString(); //错误: 无法从静态上下文中引用非静态 方法 toString() // 先new对象。 ExtendsTest05 et = new ExtendsTest05(); String retValue = et.toString(); // 2ff4acd0 可以“等同”看做对象在对内存当中的内存地址。 // 实际上是内存地址经过你“哈希算法”得出的十六进制的结果。 System.out.println(retValue);//ExtendsTest05@2ff4acd0 // 创建对象 Product prd = new Product(); String retValue2 =prd.toString(); System.out.println(retValue2);//Product@5caf905d // 以上两行代码能否合并成一行!!!可以 System.out.println(prd.toString()); // 如果直接输出“引用”呢????? // 输出 prd 默认调用prd.toSting()方法 System.out.println(prd);//Product@5caf905d } } class Product{ /* public String toString() { return getClass().getName() + "@" + Integer.toHexString(hashCode()); } */ }
测试案例:
// 在editplus中的红色字体,表示这个类是SUN的JDK写好的一个类。 public class Test{ // 静态变量 static Student stu = new Student(); // 入口 public static void main(String[] args){ Test.stu.exam(); System.out.println("Hello World!"); } } class Student{ // 实例方法 public void exam(){ System.out.println("考试。。。。。。"); } }