文章目錄
- 一、抽象類
- 1. 定義
- 2. 示例代碼
- 3. 特性
- 二、接口初識
- 1. 定義
- 2. 命名與語法
- 3. 示例代碼
- 4. 常見特性
- 5. 多接口實現
- 6. 接口的繼承
- 三、Object類初識
- 1. equals方法
- 2. hascode方法
一、抽象類
1. 定義
- 請你假設這樣一個場景,我們定義一個人的類,這個類中我們定義一個成員方法
void sleep()
,然后我們派生出兩個子類Student()
類和Teacher()
類,然后再在子類中寫出各自的sleep()
方法,也能實現多態 - 但是,今天我們要講的是抽象類和抽象方法,因此在語法上,抽象類指的是被
abstract
關鍵字修飾的類,進而抽象方法就指的是被abstract
關鍵字修飾的方法 - 在抽象類和抽象方法中,沒有具體的實現,在人的類中,我們描述這個人的類很虛幻,很寬泛,因而我們的
abstract void sleep();
方法中并無具體的內容,就單純是個方法的展示,說明人具有睡覺的習性,而未定義具體怎么睡的 - 那你可能就會問了,那我在此處不定義,難道在子類中再重新定義嗎(即重寫),是的,我們創建抽象方法的意義就是為了在子類中重寫
- 那聰明的你也會問了,我都有繼承了我為什么還要定義抽象類呢,其實這樣主要是為了在子類繼承父類的時候,檢查你是否有重寫父類中的方法,便于校驗,以免造成遺漏,而且這樣規范性也更高了,進一步提高了代碼的穩定性
2. 示例代碼
//抽象類Person
public abstract class Person {String name;public Person(String name) {this.name = name;}abstract void sleep();
}
//Student類
public class Student extends Person{String classes;String stage;public Student(String name, String classes,String stage) {super(name);this.classes = classes;this.stage = stage;}@Overridevoid sleep() {System.out.println(name+classes+"學生在睡覺");}
}
//Teacher類
public class Teacher extends Person{String department;String subject;public Teacher(String name,String department, String subject) {super(name);this.department = department;this.subject = subject;}@Overridevoid sleep() {System.out.println(name+department+"老師在睡覺");}
}
//Test測試類
public class Test {public static void func(Person person){person.sleep();}public static void main(String[] args) {Person per1 = new Student("張三","計科二班","大一");Person per2 = new Teacher("王五","教務處","計算機科學與技術");func(per1);func(per2);}
}
3. 特性
- 可以和普通的類一樣擁有成員變量和成員方法,不一定要有抽象方法,但是反之一個類中有抽象方法那么這個類必須為抽象類
- 抽象類不可以被實例化,但是它可以被繼承,而且它的主要目的就是繼承
- 當一個普通的類繼承抽象類之后,這個普通類需要去重寫抽象類中所有的抽象方法,不過若你不想自己手動寫,直接
Alt+回車
查看,然后點擊重寫選項即可
- 可以有構造方法,跟普通方法一樣,通過子類去初始化抽象類(父類)中的成員變量,因此抽象類中成員變量和構造方法功能與普通的父類近乎沒有差別,剛剛的示例代碼中就有
- 抽象類也可以發生動態綁定、向上轉型、多態等,比如示例代碼的Test測試類中的func方法
- 因抽象類之間的可繼承性,若抽象類B繼承了抽象類A,那么在抽象類B中可以不用重寫抽象類A中的抽象方法,但是當一個普通類C去繼承抽象類B的時候,不光要重寫抽象類B中的抽象方法,還要重寫抽象類A中的抽象方法,因為普通類C間接繼承了抽象類A
- 抽象方法不能被
private,final,static
等關鍵字修飾,因為這樣會導致不可被繼承,進而就不能在子類中重寫抽象方法了
Tips:如果你細心,你會發現抽象類的圖標和普通類中的圖標不一樣
二、接口初識
1. 定義
- 其實我也不好描述,你就把它理解為多個類的公共準則,即讓類具有某些屬性,一種引用的數據類型的規范,因此它可被多個類引用,就決定類其高度公有性與多個類之間共性
- 跟抽象類一樣,接口類指的是被interface關鍵字修飾的
- 成員變量默認都是被
public static final
修飾,當你寫入權限限定符時會顯示是冗余的,并且在定義的時候還需要初始化,這一點跟類不一樣
- 成員方法默認都是被
public abstract
,即都是抽象方法,因此接口類方法中不能有方法的具體定義,但是,凡事都有特殊情況,如果被static
關鍵字或defeault
關鍵字修飾后就可以
- 接口跟抽象類一樣,它不可以實例化
2. 命名與語法
- 接口內部有一套新的語法體系,命名一般是大寫字母I+形容詞
- 內部方法與成員變量不能添加任何修飾符比如public、private等
3. 示例代碼
- 我們這里就以我們之前寫過的學生類的代碼做個演示,整體思路就是借助
implement
關鍵字去實現類,并且去重寫接口中所有抽象方法
//首先我們定義一個接口IPerosn
public interface IPerson {void Attendclass();//定義上課void Classisover();//定義下課
}
//其次我們再定義兩個子類Teacher和Student
//重寫接口中的抽象方法,使其上課和下課的方式各不同
public class Student implements IPerson{@Overridepublic void Attendclass() {System.out.println("學生在上課");}@Overridepublic void Classisover() {System.out.println("學生下課離開了");}void note(){//定義學生特有方法System.out.println("記筆記");}
}public class Teacher implements IPerson{@Overridepublic void Attendclass() {System.out.println("老師在上課");}@Overridepublic void Classisover() {System.out.println("老師下課離開了");}void blackboard(){//定義老師特有方法System.out.println("粉筆寫字");}
}
//再定義一個教室類Classroom用于把兩個子類集中起來,一邊協作
public class Classroom {//這個大類把兩個子類集中起來void openDoor(){System.out.println("打開教室門");}void closeDoor(){System.out.println("關機教室門");}void learn(IPerson person){//定義協作方法person.Attendclass();if(person instanceof Student){Student stu = (Student) person;stu.note();}else if(person instanceof Teacher){Teacher ter = (Teacher) person;ter.blackboard();}person.Classisover();}
}
//我們再定義一個測試類,分別實例化教室類Classroom、學生類Student和老師類Teacher
public class Test {public static void main(String[] args) {Classroom classroom = new Classroom();Student stu = new Student();Teacher ter = new Teacher();classroom.openDoor();classroom.learn(stu);//調用協作方法classroom.learn(ter);//調用協作方法classroom.closeDoor();}
}
我來解釋下這些代碼,以便讓你更加明白
- 首先我們在接口中定義兩個需要在實現類中重寫的兩個抽象方法,僅僅聲明,并無具體實現方法
- 我們再定義兩個類
Student
和Teacher
,來實現接口,再重寫接口中的抽象方法,做后我們分別定義了當前類中特有方法note()
和blackboard()
- 我們再定義一個教室類
Classroom
去整合兩個類Student
和Teacher
,在教室類Classroom
中先定義兩個特有方法openDoor()
和closeDoor()
,再定義一個方法實現當前類Classroom
和其他兩個類Student
和Teacher
的協作方法learn
,雖然接口不可實例化,但是它也可以引用傳參,因此參數列表我們寫IPerson person
- 此時你想,如果我用向上轉型,是不是只能調用被重寫的方法,不能調用類中特有的方法(因為你形參是
IPerson person
,接口引用去引用類對象),那怎么辦呢,還記得向下轉型嗎,是的,我們就用向下轉型去訪問,但是不是說向下轉型有風險嗎,你可以看到在轉型前我們先是會檢查這個類實例化的是誰,即這個接口類引用到底引用的是哪個類,是Student
還是Teacher
呢,檢查完之后我們就可以用向下轉型去訪問類中特有的方法了- 最后我們再定義個測試類
Test
,去new了三個對象,再把Student
類和Teacher
類的對象傳給Classroom
類中的協作方法,這樣我們就完成了接口的實現,是不是很神奇呢,別介,后面還有終極大招等著
4. 常見特性
- 在重寫接口方法時候,不可使用默認的權限,要滿足重寫權限>=原抽象方法,但是呢又因為接口中方法默認是public修飾的,因此你重寫后只能用public修飾了,這也能理解,接口畢竟就是為了對外嘛
- 之前也說過,在接口中也可以定義變量,一般被隱式指定為
public static final
- 接口中不可以有代碼塊類型,譬如靜態代碼塊、構造代碼塊、普通代碼塊等
- 接口中不可以有構造方法,畢竟你接口中的變量都要定義的時候就初始化,還要構造方法干嘛
- 如果你的類沒有重寫接口中的所有抽象方法,那你當前的類就要設置為抽象類
- 即使接口并不是class類,但是編譯后還是能產生class文件
- 在JDK8之后,可以在接口中定義被default修飾的方法了
5. 多接口實現
- 原理:接口間也滿足繼承的關系
- 痛點分析:繼承有個什么痛點,就拿我們之前寫過的Person類來舉例,我們把Person定義為抽象類,我們寫了個抽象方法睡覺,也寫了成員變量name,好,我們讓兩個類
Student
和Teacher
類繼承了這個抽象類,重寫了sleep()
抽象方法,之后再在測試類中去調用即可 - 但是你想,我們可以把
Teacher
類和Student
類中的特有方法寫在抽象類中嗎,不可以,學生上課記筆記老師上課一般會記筆記嗎,不會,一般是教書,那你如果寫進抽象類中,就會導致在Student
類和Tracher
類中去重寫這兩個方法,雖然語法上可以,但是邏輯上不可 - 誒,你是不是想到了把各個特有的方法定義成一個類,再讓
Teacher
類和Student
類去繼承唄,很遺憾,Java不支持多繼承,怎么辦呢!!!
解決方法:創建多個接口去實現
- 我們把特有方法定義在接口中,再讓
Teacher
類和Student
類去實現 - 因此我們可以把記筆記這個特有方法“封裝”在一個個的接口,如果想實現多個接口,那么各個接口之間用逗號隔開即可,此時我們實現特定接口,我們再重寫接口中的方法即可
- 同時再在測試類中定義兩個特有方法的接口
public static void noting(INoteable noteable)
和public static void blackborading(IBlackboardable blackboardable)
,此時我們再把對應的對象傳給特定的方法,此時你會驚奇的發現,我們無需關心對象到底是什么類型,我們只關心他能做什么,能干什么就好,只要他具備了這些特定方法的功能,我們都可以去調用,不論你是什么類型,比如不管你是學生還是老師,只要你會記筆記,你直接調用特定方法即可,這樣就相當于給外界創建了個接口,只要你能連接上,甭管你是什么設備,都可以調用這個接口而無需知道其具體實現內容 - 以下是示例代碼,我們新定義了兩個具有特有抽象類方法的接口
INoteable
和IBlackboardable
,在Test測試類中,我們定義了兩個配合接口的特有方法noting
,blackborading
,我們直接傳具有各個特有方法的能力的對象就好noting(stu);
和
blackborading(ter);
//其他代碼都沒變,這里只展示變了的代碼
//Noteable接口
public interface INoteable {void note();
}
//IBlackboardable接口
public interface IBlackboardable {void blackboard();
}
//Test測試類中
public class Test {public static void noting(INoteable noteable){noteable.note();}public static void blackborading(IBlackboardable blackboardable){blackboardable.blackboard();}public static void main(String[] args) {//究極大招Student stu = new Student();Teacher ter = new Teacher();noting(stu);blackborading(ter);}
}
6. 接口的繼承
之前我們就講過接口是可以繼承的,具體是怎么樣的呢?說白了就是接口的擴展
- 比如有一名優秀的學生,他既會講課又會記筆記,那我們是不是可以再定義一個接口讓其具有這兩個接口的功能,因此我們用手
extends
關鍵字,在剛剛代碼中我們要寫兩個接口的實現,此時我們只需要寫一個接口的實現就好了,之后就是跟之前實現接口一樣 - 此時你會發現因為這個接口兼并了其他接口,因此在子類中實現的時候需要重寫所有被繼承接口的抽象方法,但是若這個兼并的接口里面還是有其他抽象方法,你還是要在接口實現的類中去重寫這個兼并接口中的抽象方法
- 代碼展示
//定義了IExcellent接口,暫時不寫IExcellen類中特有抽象方法
public interface IExcellent extends INoteable,IBlackboardable{
}
//定義了ExcleentStudent方法,可以發現重寫了繼承來的所有抽象方法
public class ExcleentStudent implements IExcellent {@Overridepublic void blackboard() {System.out.println("優秀學生在講課");}@Overridepublic void note() {System.out.println("優秀學生在記筆記");}
}
//我們再在Test測試類中定義一個特殊方法來調用這個接口
public static void excleenting(IExcellent iexcellent){iexcellent.blackboard();iexcellent.note();}
//再在main方法中創建對象,再傳對象,大功告成
public static void main(String[] args) {//究極大招ExcleentStudent stus = new ExcleentStudent();excleenting(stus);}
三、Object類初識
-
我們之前說過,object類是所有類的父類,因此類中有些方法默認繼承了Object中的方法
-
我們可以通過雙擊shift鍵輸入
Object
來查看,點擊左上角結構(快捷鍵Alt+7)查看我們Object
類中所有方法
-
那既然它是所有類的父類,那么說明其是萬能的類型,比如之前的代碼
Student student = new Student();
我們就可以改寫為Object student = new Student(......);
,甚至是int a = 10;
改寫成object a = 10
,既然都這樣了為什么不寫成object類型呢? -
因為其太寬泛了,不好易于區分不同類型數據
1. equals方法
- 我們給個示例代碼,拿我們剛剛演示的代碼為例
public static void main(String[] args) {Person per1 = new Student("張三","計科二班","大一");Person per3 = new Student("張三","計科二班","大一");System.out.println(per1==per3);//結果為false
}
為什么結果是false,因為你不同對象即使內容一樣,地址卻是不一樣的,比較的主要是地址,你想我用equals不就行了嗎,System.out.println(per1.equals(per3));
結果還是false,為什么,你轉到源碼去看看
你發現返回的就過是當前對象,這不還是跟System.out.println(per1==per3);
沒有半毛錢區別嗎,所以我們需要對equals方法進行重寫,我們以后也會經常碰到需要重寫的方法
-
我們可以右鍵讓編譯器重寫
-
也可我們自己重寫,這邊演示我們自己重寫,代碼有注釋,應該可以看懂
@Overridepublic boolean equals(Object obj){if(obj == null){return false;//如果你傳的是空對象直接返回}if(this == obj){//如果你傳的另外個對象是你本身的話return true;}if(!(obj instanceof Person)){//實例化的不是同個類return false;}//此時的情況就是同一個類的情況了,只需要比較內容是否一致就好Person temp = (Person) obj;//為什么此時向下轉型安全,因為之前已經檢查過了if(temp.name.equals(this.name)){//假設名字一樣就整體相等return true;}return false;}
當然也可以簡化下代碼
@Overridepublic boolean equals(Object obj){if(obj == null){return false;//如果你傳的是空對象直接返回}if(this == obj){//如果你傳的另外個對象是你本身的話return true;}if(!(obj instanceof Person)){//實例化的不是同個類return false;}//此時的情況就是同一個類的情況了,只需要比較內容是否一致就好Person temp = (Person) obj;//為什么此時向下轉型安全,因為之前已經檢查過了return temp.name.equals(this.name);//簡化了這里}
2. hascode方法
用于在散列表中獲取位置,底層是用C/C++寫的,看不到具體實現方法,但是如果你直接用object類的會有問題,通過per1.hashcode()
和per3.hashcode()
不一樣,因此我們需要對hashcode進行重寫操作,我們直接右鍵生成,此時返回的是散列碼進行比較了,發現其散列碼就一樣了
- 你是否會好奇為啥我通過對象直接使用
.
去訪問equals和hashcode方法呢,還記不記得之前說object類是所有類的父類,因此我們默認繼承了object類這些方法,難道我們不能訪問嗎,顯然可以訪問,畢竟每個類對位object的子類
這里補充一點:我們傳對象時候可以直接傳new的對象或者是定義一個傳對象的數組去傳
func(new Dog());
func(new horse());
Person [] person = {new Student(),new Teacher()};