繼承可使用 extends 和 implements 這兩個關鍵字來實現繼承,並且全部的類都是繼承於 java.lang.Object,當一個類沒有繼承的兩個關鍵字,則默認繼承object(這個類在 java.lang 包中,因此不須要 import)祖先類。java
final 關鍵字聲明類能夠把類定義爲不能繼承的,即最終類;或者用於修飾方法,該方法不能被子類重寫。編程
子類是不繼承父類的構造器(構造方法或者構造函數)的,它只是調用(隱式或顯式)。安全
若是父類的構造器帶有參數,則必須在子類的構造器中顯式地經過 super 關鍵字調用父類的構造器並配以適當的參數列表。編程語言
若是父類構造器沒有參數,則在子類的構造器中不須要使用 super 關鍵字調用父類構造器,系統會自動調用父類的無參構造器。 ide
重寫是子類對父類的容許訪問的方法的實現過程進行從新編寫, 返回值和形參都不能改變。即外殼不變,核心重寫!函數
重寫的好處在於子類能夠根據須要,定義特定於本身的行爲。 也就是說子類可以根據須要實現父類的方法。測試
重寫方法不能拋出新的檢查異常或者比被重寫方法申明更加寬泛的異常。例如: 父類的一個方法申明瞭一個檢查異常 IOException,可是在重寫這個方法的時候不能拋出 Exception 異常,由於 Exception 是 IOException 的父類,只能拋出 IOException 的子類異常。this
在面向對象原則裏,重寫意味着能夠重寫任何現有方法。spa
方法的重寫規則設計
Super關鍵字的使用
當須要在子類中調用父類的被重寫方法時,要使用super關鍵字。
class Animal{ public void move(){ System.out.println("動物能夠移動"); } } class Dog extends Animal{ public void move(){ super.move(); // 應用super類的方法 System.out.println("狗能夠跑和走"); } } public class TestDog{ public static void main(String args[]){ Animal b = new Dog(); // Dog 對象 b.move(); //執行 Dog類的方法 } }
重載(overloading) 是在一個類裏面,方法名字相同,而參數不一樣。返回類型能夠相同也能夠不一樣。
每一個重載的方法(或者構造函數)都必須有一個獨一無二的參數類型列表。
最經常使用的地方就是構造器的重載。
重載規則:
方法的重寫(Overriding)和重載(Overloading)是java多態性的不一樣表現,重寫是父類與子類之間多態性的一種表現,重載能夠理解成多態的具體表現形式。
多態性是對象多種表現形式的體現。
如,同一個事件發生在不一樣的對象上會產生不一樣的結果。
當使用多態方式調用方法時,首先檢查父類中是否有該方法,若是沒有,則編譯錯誤;若是有,再去調用子類的同名方法。
多態的好處:可使程序有良好的擴展,並能夠對全部類的對象進行通用處理。
public class Test { public static void main(String[] args) { show(new Cat()); // 以 Cat 對象調用 show 方法 show(new Dog()); // 以 Dog 對象調用 show 方法 Animal a = new Cat(); // 向上轉型 a.eat(); // 調用的是 Cat 的 eat Cat c = (Cat)a; // 向下轉型 c.work(); // 調用的是 Cat 的 work } public static void show(Animal a) { a.eat(); // 類型判斷 if (a instanceof Cat) { // 貓作的事情 Cat c = (Cat)a; c.work(); } else if (a instanceof Dog) { // 狗作的事情 Dog c = (Dog)a; c.work(); } } } abstract class Animal { abstract void eat(); } class Cat extends Animal { public void eat() { System.out.println("吃魚"); } public void work() { System.out.println("抓老鼠"); } } class Dog extends Animal { public void eat() { System.out.println("吃骨頭"); } public void work() { System.out.println("看家"); } }
虛函數的存在是爲了多態。
Java 中其實沒有虛函數的概念,它的普通函數就至關於 C++ 的虛函數,動態綁定是Java的默認行爲。若是 Java 中不但願某個函數具備虛函數特性,能夠加上 final 關鍵字變成非虛函數。
咱們將介紹在 Java 中,當設計類時,被重寫的方法的行爲怎樣影響多態性。
咱們已經討論了方法的重寫(子類可以重寫父類的方法)。當子類對象調用重寫的方法時,調用的是子類的方法,而不是父類中被重寫的方法。
要想調用父類中被重寫的方法,則必須使用關鍵字 super。
例
父類
public class Employee { private String name; private String address; private int number; public Employee(String name, String address, int number) { System.out.println("Employee 構造函數"); this.name = name; this.address = address; this.number = number; } public void mailCheck() { System.out.println("郵寄支票給: " + this.name + " " + this.address); } public String toString() { return name + " " + address + " " + number; } public String getName() { return name; } public String getAddress() { return address; } public void setAddress(String newAddress) { address = newAddress; } public int getNumber() { return number; } }
子類
public class Salary extends Employee { private double salary; // 整年工資 public Salary(String name, String address, int number, double salary) { super(name, address, number); setSalary(salary); } public void mailCheck() { System.out.println("Salary 類的 mailCheck 方法 "); System.out.println("郵寄支票給:" + getName() + " ,工資爲:" + salary); } public double getSalary() { return salary; } public void setSalary(double newSalary) { if(newSalary >= 0.0) { salary = newSalary; } } public double computePay() { System.out.println("計算工資,付給:" + getName()); return salary/52; } }
測試
/* 文件名 : VirtualDemo.java */ public class VirtualDemo { public static void main(String [] args) { Salary s = new Salary("員工 A", "北京", 3, 3600.00); Employee e = new Salary("員工 B", "上海", 2, 2400.00); System.out.println("使用 Salary 的引用調用 mailCheck -- "); s.mailCheck(); System.out.println("\n使用 Employee 的引用調用 mailCheck--"); e.mailCheck(); } }
以上實例編譯運行結果以下:
Employee 構造函數 Employee 構造函數 使用 Salary 的引用調用 mailCheck -- Salary 類的 mailCheck 方法 郵寄支票給:員工 A ,工資爲:3600.0 使用 Employee 的引用調用 mailCheck-- Salary 類的 mailCheck 方法 郵寄支票給:員工 B ,工資爲:2400.0
1. 生活中的接口最具表明性的就是插座,例如一個三接頭的插頭都能接在三孔插座中,由於這個是每一個國家都有各自規定的接口規則,有可能到國外就不行,那是由於國外本身定義的接口類型。
2. java中的接口相似於生活中的接口,就是一些方法特徵的集合,但沒有方法的實現。
接口(英文:Interface),在JAVA編程語言中是一個抽象類型,是抽象方法的集合,接口一般以interface來聲明。一個類經過繼承接口的方式,從而來繼承接口的抽象方法。
接口並非類,編寫接口的方式和類很類似,可是它們屬於不一樣的概念。類描述對象的屬性和方法。接口則包含類要實現的方法。
除非實現接口的類是抽象類,不然該類要定義接口中的全部方法。
接口沒法被實例化,可是能夠被實現。一個實現接口的類,必須實現接口內所描述的全部方法,不然就必須聲明爲抽象類。另外,在 Java 中,接口類型可用來聲明一個變量,他們能夠成爲一個空指針,或是被綁定在一個以此接口實現的對象。
重寫接口中聲明的方法時,須要注意如下規則:
在實現接口的時候,也要注意一些規則:
一個接口能繼承另外一個接口,和類之間的繼承方式比較類似。接口的繼承使用extends關鍵字,子接口繼承父接口的方法。
在Java中,類的多繼承是不合法,但接口容許多繼承。
在接口的多繼承中extends關鍵字只須要使用一次,在其後跟着繼承接口。
在面向對象程式設計方法中,封裝(英語:Encapsulation)是指一種將抽象性函式接口的實現細節部份包裝、隱藏起來的方法。
封裝能夠被認爲是一個保護屏障,防止該類的代碼和數據被外部類定義的代碼隨機訪問。
要訪問該類的代碼和數據,必須經過嚴格的接口控制。
封裝最主要的功能在於咱們能修改本身的實現代碼,而不用修改那些調用咱們代碼的程序片斷。
適當的封裝可讓程式碼更容易理解與維護,也增強了程式碼的安全性。
1. 良好的封裝可以減小耦合。
2. 類內部的結構能夠自由修改。
3. 能夠對成員變量進行更精確的控制。
4. 隱藏信息,實現細節。
在面向對象的概念中,全部的對象都是經過類來描繪的,可是反過來,並非全部的類都是用來描繪對象的,若是一個類中沒有包含足夠的信息來描繪一個具體的對象,這樣的類就是抽象類。
抽象類除了不能實例化對象以外,類的其它功能依然存在,成員變量、成員方法和構造方法的訪問方式和普通類同樣。
因爲抽象類不能實例化對象,因此抽象類必須被繼承,才能被使用。也是由於這個緣由,一般在設計階段決定要不要設計抽象類。
父類包含了子類集合的常見的方法,可是因爲父類自己是抽象的,因此不能使用這些方法。
在Java中抽象類表示的是一種繼承關係,一個類只能繼承一個抽象類,而一個類卻能夠實現多個接口。
若是你想設計這樣一個類,該類包含一個特別的成員方法,該方法的具體實現由它的子類肯定,那麼你能夠在父類中聲明該方法爲抽象方法。
Abstract 關鍵字一樣能夠用來聲明抽象方法,抽象方法只包含一個方法名,而沒有方法體。
抽象方法沒有定義,方法名後面直接跟一個分號,而不是花括號。
public abstract class Employee { private String name; private String address; private int number; public abstract double computePay(); //其他代碼 }
聲明抽象方法會形成如下兩個結果:
繼承抽象方法的子類必須重寫該方法。不然,該子類也必須聲明爲抽象類。最終,必須有子類實現該抽象方法,不然,從最初的父類到最終的子類都不能用來實例化對象。
1. 抽象類不能被實例化(初學者很容易犯的錯),若是被實例化,就會報錯,編譯沒法經過。只有抽象類的非抽象子類能夠建立對象。
2. 抽象類中不必定包含抽象方法,可是有抽象方法的類一定是抽象類。
3. 抽象類中的抽象方法只是聲明,不包含方法體,就是不給出方法的具體實現也就是方法的具體功能。
4. 構造方法,類方法(用 static 修飾的方法)不能聲明爲抽象方法。
5. 抽象類的子類必須給出抽象類中的抽象方法的具體實現,除非該子類也是抽象類。