------- android培訓(xùn)、java培訓(xùn)、期待與您交流! ----------
創(chuàng)新互聯(lián)公司專業(yè)為企業(yè)提供陸河網(wǎng)站建設(shè)、陸河做網(wǎng)站、陸河網(wǎng)站設(shè)計(jì)、陸河網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計(jì)與制作、陸河企業(yè)網(wǎng)站模板建站服務(wù),十年陸河做網(wǎng)站經(jīng)驗(yàn),不只是建網(wǎng)站,更提供有價(jià)值的思路和整體網(wǎng)絡(luò)服務(wù)。
繼承
1繼承的概述
多個(gè)類中存在相同屬性和行為時(shí),將這些內(nèi)容抽取到單獨(dú)一個(gè)類中,那么多個(gè)類無需再定義這些屬性和行為,只要繼承單獨(dú)的那個(gè)類即可。
多個(gè)類可以稱為子類,單獨(dú)這個(gè)類稱為父類或者超類。
子類可以直接訪問父類中的非私有的屬性和行為。
繼承:
1,提高代碼的復(fù)用性。
2,讓類與類之間產(chǎn)生了關(guān)系。有了這個(gè)關(guān)系,才有了多態(tài)的特征。
注意:不要為了獲取其他類的功能,簡化代碼而繼承。必須是類與類之間有所屬關(guān)系才可以繼承,所屬關(guān)系 is a。
java語言中:java只支持單繼承,不支持多繼承。
因?yàn)槎嗬^承容易帶來安全隱患:當(dāng)多個(gè)父類中定義了相同功能,但功能內(nèi)容不同時(shí),子類對(duì)象不確定要運(yùn)行哪一個(gè)。但是java保留了這種機(jī)制,并用另一種體現(xiàn)形式來完成表示:多實(shí)現(xiàn)。
java支持多層繼承。也就是一個(gè)繼承體系。
如何使用一個(gè)繼承體系中的功能呢?
1.想要使用體系,先查閱體系父類的描述,因?yàn)楦割愔卸x的是該體系中的共×××。
2.通過了解共×××,就可以知道該體系的基本功能。
那么這個(gè)體系已經(jīng)可以基本使用了。
在具體調(diào)用時(shí),要?jiǎng)?chuàng)建最子類的對(duì)象,為什么呢?
一是因?yàn)橛锌赡芨割惒荒軇?chuàng)建對(duì)象;
二是創(chuàng)建子類對(duì)象可以使用更多的功能,包括基本的也包括特有的。
簡單一句話:查閱父類功能,創(chuàng)建子類對(duì)象使用功能。
class ExtendsDemo { public static void main(String[] args) { } } class Person { String name; int age; } class Student extends Person { //String name; //int age; void study() { System.out.println("good good study"); } } class Worker extends Person { //String name; //int age; void work() { System.out.println("good good work"); } }
2繼承的特點(diǎn)
子父類出現(xiàn)后,類中成員的特點(diǎn)。
類中成員:1,變量。2,函數(shù)。3,構(gòu)造函數(shù)。
1、變量
如果子類中出現(xiàn)非私有的同名成員變量時(shí),子類要訪問本類中的變量,用this。子類要訪問父類中的同名變量,用super。super的使用和this的使用幾乎一致。this代表的是本類對(duì)象的引用。super代表的是父類對(duì)象的引用。
代碼體現(xiàn):
class Fu { int num = 4; } class Zi extends Fu { int num = 5; void show() { System.out.println(super.num); } } class ExtendsDemo2 { public static void main(String[] args) { Zi z = new Zi(); System.out.println(z.num); } }
2,子類中的函數(shù)。
當(dāng)子類出現(xiàn)和父類一模一樣的函數(shù)時(shí),子類對(duì)象調(diào)用該函數(shù)時(shí),會(huì)運(yùn)行子類函數(shù)的內(nèi)容。如同父類的函數(shù)被覆蓋一樣。這種情況是函數(shù)的另一個(gè)特性:重寫(覆蓋)。當(dāng)子類繼承父類,沿襲了父類的功能到子類中。但是子類雖具備該功能,但是功能的內(nèi)容卻與父類不一致。這時(shí),沒有必要定義新功能,而是使用覆蓋特性,保留父類的功能定義,并重寫功能內(nèi)容。
覆蓋:
1,子類覆蓋父類,必須保證資料權(quán)限大于等于父類權(quán)限,才可以覆蓋,否則編譯失敗。
2,靜態(tài)只能覆蓋靜態(tài)。
重載:只看同名函數(shù)的參數(shù)列表。
重寫:子父類方法要一模一樣。
代碼體現(xiàn):
class Fu { void show() { System.out.println("fu show"); } void speak() { System.out.println("vb"); } } class Zi extends Fu { void show() { System.out.println("zi show"); } void speak() { System.out.println("java"); } } class ExtendsDemo3 { public static void main(String[] args) { Zi z = new Zi(); z.show(); } }
3,子父類中的構(gòu)造函數(shù)。
在對(duì)子類對(duì)象進(jìn)行初始化時(shí),父類的構(gòu)造函數(shù)也會(huì)運(yùn)行。那是因?yàn)樽宇惖臉?gòu)造函數(shù)默認(rèn)第一行有一條隱式的語句 super();
super ():會(huì)訪問父類中空參數(shù)的構(gòu)造函數(shù)。而且子類中所有的構(gòu)造函數(shù)默認(rèn)第一行都是super();
為什么子類一定要訪問父類中的構(gòu)造函數(shù)。
因?yàn)楦割愔械臄?shù)據(jù)子類可以直接獲取,所以子類對(duì)象在建立時(shí),需要先查看父類是如何對(duì)這些數(shù)據(jù)進(jìn)行初始化的。所有子類在對(duì)象初始化時(shí),要先訪問一下父類中的構(gòu)造函數(shù)。
如果要訪問父類中的指定構(gòu)造函數(shù),可以通過手動(dòng)定義super語句的方式來指定。
注意:super語句一定定義在構(gòu)造函數(shù)的第一行。
子類的實(shí)例化過程。
結(jié)論:子類的所有的構(gòu)造函數(shù),默認(rèn)都會(huì)訪問父類中空參數(shù)的構(gòu)造函數(shù)。因?yàn)樽宇惷恳粋€(gè)構(gòu)造函數(shù)內(nèi)的第一行都有一句隱式的super();當(dāng)父類中沒有空參數(shù)的構(gòu)造函數(shù)時(shí),子類必須手動(dòng)通過super語句形式來制定訪問父類中的構(gòu)造函數(shù)。當(dāng)然,子類的構(gòu)造函數(shù)第一行也可以手動(dòng)指定this語句來訪問本類中的構(gòu)造函數(shù)。子類中至少會(huì)有一個(gè)構(gòu)造函數(shù)會(huì)訪問父類中的構(gòu)造函數(shù)。
class ExtendsDemo4 { public static void main(String[] args) { new Zi(); } } class Fu { Fu() { System.out.println("fu run"); } Fu(int x) { System.out.println("fu..."+x); } } class Zi extends Fu { Zi() { //super (); System.out.println("zi run"); } Zi(int x) { this();//沒有隱式的super(); //super (); System.out.println("zi..."+x); } void show(){} }
3 final關(guān)鍵字
final :最終。作為一個(gè)修飾符,
1,可以修飾類,函數(shù),變量。
2,被final修飾的類不可以被繼承。為了避免被繼承,被子類復(fù)寫功能。
3,被final修飾的方法不可以被復(fù)寫。
4,被final修飾的變量時(shí)一個(gè)常量只能賦值一次,既可以修飾成員變量,又可以修飾局部變量。
當(dāng)在描述事物時(shí),一些數(shù)據(jù)的出現(xiàn)值是固定的,那么這時(shí)為了增強(qiáng)閱讀性,都給這些值起個(gè)名字,方便于閱讀。
而這個(gè)值不需要改變,所以加上final修飾。
作為常量:常量的書寫規(guī)范所有字符都大寫,如果由多個(gè)單詞組成。單詞間通過_連接。
5,內(nèi)部類定義在類中的局部位置上時(shí),只能訪問該局部被final修飾的局部變量。
class Demo { public static final double PI = 3.14;//全局常量。 final void show1() {} void show2() {} } class SubDemo extends Demo { void show1() {} } class FinalDemo { public static void main(String[] args) { System.out.println("Hello World!"); } }
4抽象類
抽象類:Java中可以定義沒有方法體的方法,該方法的具體實(shí)現(xiàn)由子類完成,該方法稱為抽象方法,包含抽象方法的類就是抽象類。
抽象方法的由來:
多個(gè)對(duì)象都具備相同的功能,但是功能具體內(nèi)容有所不同,那么在抽取過程中,只抽取了功能定義,并未抽取功能主體,那么只有功能聲明,沒有功能主體的方法稱為抽象方法。
抽象類的特點(diǎn):
1,抽象方法一定在抽象類中。
2,抽象方法和抽象類必須被abstract關(guān)鍵字修飾。
3,抽象類不可以用new創(chuàng)建對(duì)象。因?yàn)檎{(diào)用抽象方法沒意義。
4,抽象類中的抽象方法要被使用,必須有子類復(fù)寫其所有的抽象方法后,建立子類對(duì)象調(diào)用。如果子類值覆蓋了部分抽象方法,那么該子類還是抽象類。
抽象類與一般類沒有太大的不同。
該如何描述事物,就如何描述事物,只不過,該事物中出現(xiàn)了一些看不懂的東西。這些不確定的部分,也是該事物的功能,需要明確出來,但是無法定義主體。通過抽象方法來表示。
抽象類比一般類多了抽象函數(shù)。
抽象類不可以實(shí)例化。
特殊:抽象類中可以不定義抽象方法,這樣做僅僅是不讓該類建立對(duì)象。
抽象類的細(xì)節(jié):
1:抽象類中是否有構(gòu)造函數(shù)?有,用于給子類對(duì)象進(jìn)行初始化。
2:抽象類中是否可以定義非抽象方法?
可以。其實(shí),抽象類和一般類沒有太大的區(qū)別,都是在描述事物,只不過抽象類在描述事物時(shí),有些功能不具體。所以抽象類和一般類在定義上,都是需要定義屬性和行為的。只不過,比一般類多了一個(gè)抽象函數(shù)。而且比一般類少了一個(gè)創(chuàng)建對(duì)象的部分。
3:抽象關(guān)鍵字abstract和哪些不可以共存?final , private , static
4:抽象類中可不可以不定義抽象方法?可以。抽象方法目的僅僅為了不讓該類創(chuàng)建對(duì)象。
雇員示例:
需求:公司中程序員有姓名,工號(hào),薪水,工作內(nèi)容。項(xiàng)目經(jīng)理除了有姓名,工號(hào),薪水,還有獎(jiǎng)金,工作內(nèi)容。對(duì)給出需求進(jìn)行數(shù)據(jù)建模
abstract class Employee { private String name; private String id; private double pay; Employee(String name,String id,double pay) { this.name = name; this.id = id; this.pay = pay; } public abstract void work(); } class Manger extends Employee { private int bonus; Manger(String name,String id,double pay,int bones) { super(name,id,pay); this.bonus= bouns; public void work() { System.out.println("manger work"); } } } class Pro extends Employee { Pro(String name,String id,double pay) { super(name,id,pay); } public void work() { System.out.println("pro work"); } }
模板方法設(shè)計(jì)模式
什么是模板方法?
在定義功能時(shí),功能的一部分是確定的。但是由一部分是不確定,而確定的部分在使用不確定的部分。那么這時(shí)就將不確定的部分暴露出去。由該類的子類去完成。
abstractclass GetTime{ public final void getTime(){ //此功能如果不需要復(fù)寫,可加final限定 long start = System.currentTimeMillis(); code(); //不確定的功能部分,提取出來,通過抽象方法實(shí)現(xiàn) long end =System.currentTimeMillis(); System.out.println("毫秒是:"+(end-start)); } public abstract void code(); //抽象不確定的功能,讓子類復(fù)寫實(shí)現(xiàn) } classSubDemo extends GetTime{ public void code(){ //子類復(fù)寫功能方法 for(int y=0; y<1000; y++){ System.out.println("y"); } } }
5接口
接口:初期理解,可以認(rèn)為是一個(gè)特殊的抽象類。當(dāng)抽象類中的方法都是抽象的,那么該類可以通過接口的形式來表示。
class用于定義類。interface用于定義接口。
接口定義時(shí),格式特點(diǎn):
1,接口中常見定義:常量,抽象方法。
2,接口中的成員都有固定修飾符。
常量:public static final
方法:public abstract
接口中的成員都是public的。
接口:是不可以創(chuàng)建對(duì)象的,因?yàn)橛谐橄蠓椒?。需要被子類?shí)現(xiàn),子類對(duì)接口中的抽象方法全部覆蓋后,子類才可以實(shí)例化。否則子類是一個(gè)抽象類。
interfaceInter { public static final int NUM = 3; public abstract void show(); } interfaceInterA { public abstract void show(); } classDemo { public void function(){} } classTest extends Demo implements Inter,InterA { public void show(){} } 接口可以被類多實(shí)現(xiàn),也是對(duì)多繼承不支持的轉(zhuǎn)換形式,java支出多實(shí)現(xiàn)。 interfaceA { void mehtodA(); } interfaceB //extends A { void mehtodB(); } interfaceC extends B,A//接口可多繼承。 { void mehtodC(); } class Dimplements C { puclic void mehtodA(){} puclic void mehtodB(){} puclic void mehtodC(){} }