版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
第5章消息、繼承與多態(tài)5.1消息5.2訪問控制5.3多態(tài)機(jī)制5.4繼承機(jī)制5.5抽象類、接口與包
5.1消息
5.1.1消息的概念
面向?qū)ο蟮南到y(tǒng)中,對象間的相互作用是通過一個(gè)對象向另一個(gè)對象發(fā)送消息來體現(xiàn)的。消息就是向?qū)ο蟀l(fā)出服務(wù)請求,是對數(shù)據(jù)成員和成員方法的引用。它含有信息如下:
提供服務(wù)的對象標(biāo)識——對象名、服務(wù)標(biāo)識——方法名、輸入信息——實(shí)際參數(shù)、回答信息——返回值或操作結(jié)果。
消息具有三個(gè)性質(zhì):
(1)同一對象可接收不同形式的多個(gè)消息,產(chǎn)生不同的響應(yīng)。
(2)相同形式的消息可以發(fā)送給不同的對象,各對象所做出的響應(yīng)可以是截然不同的。
(3)消息的發(fā)送可以不考慮具體的接收者,對象可以響應(yīng)消息,也可以對消息不予理會(huì),對消息的響應(yīng)并不是必須的。5.1.2公有消息和私有消息
(1)私有消息:是對象發(fā)送給自己的消息。私有消息對外是不開放的,外界不必了解它;
(2)公有消息:是外界對象向某對象發(fā)送的消息。外界對象不能向某對象發(fā)送私有消息。
5.1.3特定于對象的消息
特定于對象的消息是指將所有能支持此對象可接收消息的方法集中在一起,形成一個(gè)大消息。這些消息讓對象執(zhí)行這個(gè)方法而不管它可能做什么及怎么做。分為以下三種類型:
(1)可以返回對象內(nèi)部狀態(tài)的消息。
(2)可以改變對象內(nèi)部狀態(tài)的消息。
(3)可以做一些特定操作,改變系統(tǒng)狀態(tài)的消息。
【示例程序C5_1.java】不同類型消息的傳遞示例。給出學(xué)生的學(xué)號、名字、性別及年齡。
classStudent
{publicStringname;public
char
sex;
public
int
no;public
int
age;
Student(int
cno,Stringcname,char
csex,int
cage)
{name=cname;sex=csex;no=cno;age=cage;}
public
voidshowNo(){System.out.println("No:"+no);}
public
voidshowName(){System.out.println("Name:"+name);}
public
voidshowSex(){System.out.println("Sex:"+sex);}
public
voidshowAge(){System.out.println("age:"+age);}
}
classStudentScore
{private
int
no;private
double
score;
public
voidsendScore(int
cno,double
cscore)
{/*下面兩條語句是對象發(fā)送給自身的消息,
要求給自己
的數(shù)據(jù)成員賦值,這是一種私有消息,外界是不知道的*/
no=cno;score=cscore;
}
voidprintScore(){
System.out.println("No:"+no+"score:"+score);}
}public
classC5_1
{
public
static
voidmain(String[]args)
{
int
m;
//下面兩句發(fā)送new消息給類Student,要求創(chuàng)建st1,st2的對象
Studentst1=newStudent(101,"zhangli",'F',18);
Studentst2=newStudent(102,"hongbing",'M',17);
//發(fā)送new消息給類StudentScore,要求創(chuàng)建sc1,sc2的對象
StudentScoresc1=newStudentScore();
StudentScoresc2=newStudentScore();
/*向st1的對象發(fā)送顯示學(xué)號、名字、年齡的消息。這些消息都是公有消息。它們形成了同一對象可接收不同形式的多個(gè)消息,產(chǎn)生不同的響應(yīng)*/
st1.showNo();//這條消息響應(yīng)的結(jié)果是顯示st1的對象的學(xué)號
st1.showName();//顯示對象姓名的消息
st1.showAge();//顯示對象年齡的消息
st1.age=20;//修改對象的數(shù)據(jù)成員的消息,修改st1對象的年齡
m=st1.age;//返回對象的數(shù)據(jù)成員的消息,將返回消息賦給變量m
System.out.println(“m=”+m);
/*向st2的對象發(fā)送兩個(gè)顯示信息的消息,與st1的對象相同,顯示學(xué)號及名字。這些消息都是公有消息,說明了相同形式的消息可以送給不同的對象,各對象所做出的響應(yīng)可以是截然不同的*/
st2.showNo();st2.showName();
/*向sc1、sc2的對象各發(fā)送一個(gè)按學(xué)號輸入成績單的消息,這些消息都是公有消息*/
sc1.sendScore(101,97);sc2.sendScore(102,84);
//向sc1、sc2的對象各發(fā)送一個(gè)打印消息,這些消息都是公有消息
sc1.printScore();sc2.printScore();
}//main
}運(yùn)行結(jié)果:
5.2訪問控制
訪問控制符是一組限定類、數(shù)據(jù)成員或成員方法是否可以被其他類訪問的修飾符。通過聲明類的訪問控制符可以使整個(gè)程序結(jié)構(gòu)清晰、嚴(yán)謹(jǐn),減少可能產(chǎn)生的類間干擾和錯(cuò)誤。5.2.1公共訪問控制符public
Java的類是通過包的概念來組織的。定義在同一個(gè)程序文件中的所有類都屬于同一個(gè)包。處于同一個(gè)包中的類都是可見的,即可以不需任何說明而方便地互相訪問和引用。而對于不同包中的類,一般說來,它們相互之間是不可見的,當(dāng)然也不可能互相引用。
當(dāng)一個(gè)類被聲明為public時(shí),只要在其他包的程序中使用import語句引入這個(gè)public類,就可以訪問和引用這個(gè)類,并創(chuàng)建這個(gè)類的對象,訪問這個(gè)類內(nèi)部可見的數(shù)據(jù)成員和引用它的可見的方法。注意:處于不同包中的public類作為整體對其他類是可見的,并不代表該類的所有數(shù)據(jù)成員和成員方法也同時(shí)對其他包中的類是可見的。只有當(dāng)public類的數(shù)據(jù)成員和成員方法的訪問控制符也被聲明為public時(shí),這個(gè)類的所有用public修飾的數(shù)據(jù)成員和成員方法才同時(shí)對其他包中的類也是可見的。
例如,把Java類庫中的標(biāo)準(zhǔn)數(shù)學(xué)函數(shù)類math和標(biāo)準(zhǔn)數(shù)學(xué)函數(shù)方法都聲明為public,以供其他的類和程序使用。
需要注意的是,數(shù)據(jù)成員和成員方法的訪問控制符被聲明為public時(shí),會(huì)造成安全性和封裝性下降,所以一般應(yīng)盡量少用。
【示例程序A.java和B.java】有無public訪問控制符的不同。
packagech52;importch51.A;public
classB{public
static
voidmain(String[]args)
{double
s1;Ap1=newA();//創(chuàng)建A類p1的對象
//p1.x1=7;x1不是公共數(shù)據(jù)成員不能訪問
p1.x2=5.2;//訪問A類p1的對象的數(shù)據(jù)成員
s1=p1.ar(8);//訪問A類p1的對象的成員方法System.out.println("p1.x2="+p1.x2+"s1="+s1);
}}packagech51;public
classA{double
x1;//友好訪問的數(shù)據(jù)成員
public
double
x2;//公共的數(shù)據(jù)成員
public
doublear(double
x)//公共的成員方法{double
s;
s=x;return
s;}}運(yùn)行結(jié)果:運(yùn)行結(jié)果:5.2.2缺省訪問控制符
如果一個(gè)類沒有訪問控制符,說明它具有缺省的訪問控制特性,這種缺省的訪問控制特性稱為“友好訪問”。友好訪問規(guī)定只有在同一個(gè)包中的對象才能訪問和引用這些類,因此,友好訪問又稱為包訪問性。同樣道理,類內(nèi)的數(shù)據(jù)成員和成員方法如果沒有訪問控制符來限定,那么它們同樣具有“友好訪問”的特性,它們也具有包訪問性,可以被同一個(gè)包中的其他類所訪問和引用。
【示例程序C5_2.java和ClassArea.java】缺省訪問控制符的程序示例。
在ch5包中創(chuàng)建兩個(gè)類:ClassArea類和C5_2類。其中,ClassArea是缺省訪問控制符的類,其功能是計(jì)算矩形面積;C5_2是一個(gè)公共類,在這個(gè)類中創(chuàng)建ClassArea類的實(shí)例對象ss,并訪問該對象的成員方法。public
classC5_2
{public
static
voidmain(String[]args)
{double
a=2.2,b=3.1,z;
/*在類C5_2中創(chuàng)建被訪問ClassArea類的ss的對象,并訪問對象
*的成員方法。這就是說,包中類是可見的,可以互相引用*/
ClassAreass=newClassArea();
z=ss.area(a,b);//ss引用對象的成員方法
System.out.println(“z=”+z);
}
}
classClassArea
{double
lon,wid;//數(shù)據(jù)成員的修飾符為缺省
doublearea(double
x,double
y)//成員方法的修飾符為缺省
{
double
s;
//方法內(nèi)的變量
lon=x;wid=y;s=lon*wid;//求矩形面積
return
s;
//返回面積值
}}//運(yùn)行結(jié)果:z=6.8200000000000015.2.3私有訪問控制符private
用private修飾的數(shù)據(jù)成員或成員方法只能被該類自身訪問和修改,而不能被任何其他類(包括該類的子類)訪問和引用。它提供了最高的保護(hù)級別。當(dāng)其他類希望獲取或修改私有成員時(shí),需要借助于類的方法來實(shí)現(xiàn)。
【示例程序C5_3.java】同一個(gè)包的同一文件中,用private修飾的父類的數(shù)據(jù)成員不能被子類的實(shí)例對象引用,故程序中將該語句變成了注釋語句。classP1
{int
nn;private
int
n=9;//私有數(shù)據(jù)成員n
P1()//構(gòu)造方法
{nn=n++;}//可以被該類的對象自身訪問和修改
voidma()
{System.out.println("n="+n);}//可以被該類的對象自身訪問
}
public
classC5_3extendsP1//C5_3是類P1的子類
{public
static
voidmain(String[]args)
{P1m1=newP1();
System.out.println("m1.nn="+m1.nn);
//System.out.println("m1.n="+m1.n);錯(cuò),不能引用父類的私有成員
m1.ma();//可以引用P1類自身的成員方法
}
}運(yùn)行結(jié)果如下:
m1.nn=9n=105.2.4保護(hù)訪問控制符protected
用protected修飾的成員變量可以被三種類引用:該類自身,與它在同一個(gè)包中的其他類以及在其他包中的該類的子類。使用protected修飾符的主要作用是允許其他包中的它的子類來訪問父類的特定屬性。
【示例程序C41.java和C42.java】當(dāng)父類的數(shù)據(jù)成員用protected修飾時(shí),其他包的子類引用該數(shù)據(jù)成員的情況。具體情況見程序中的注釋。packagech51;
public
classC41{
protected
double
x1;double
x2;
protected
doublear(double
x)
{
double
s;s=x;return
s;}
}
packagech52;
importch51.C41;
public
classC42extendsC41
{public
static
voidmain(String[]args)
{double
s1;C42p1=newC42();
p1.x1=7;//可以訪問,屬性x1是protected修飾的
s1=p1.ar(4);//可以訪問,方法ar()是protected修飾的
//p1.x2=9;錯(cuò),不能訪問,x2沒有訪問控制符,不能被另一包的實(shí)例對象訪問
System.out.println("p1.x1="+p1.x1+"s1="+s1);
}//main
}
運(yùn)行結(jié)果:p1.x1=7.0s1=4.0
5.3多態(tài)機(jī)制
5.3.1多態(tài)的概念
多態(tài)是指一個(gè)程序中具有相同名字而內(nèi)容不同的方法共存的情況。這些方法同名的原因是它們的最終功能和目的都相同,但是由于在完成同一功能時(shí)可能遇到不同的具體情況,因此需要定義包含不同具體內(nèi)容的方法,來代表多種具體實(shí)現(xiàn)形式。
多態(tài)其目的是提高程序的抽象度、封閉性和簡潔性,統(tǒng)一一個(gè)或多個(gè)相關(guān)類對外的接口。
Java中提供了兩種多態(tài)機(jī)制:重載與覆蓋。5.3.2重載
當(dāng)在同一類中定義了多個(gè)同名而不同內(nèi)容的成員方法時(shí),我們稱這些方法是重載(override)的方法。重載的方法主要通過形式參數(shù)列表中參數(shù)的個(gè)數(shù)、參數(shù)的數(shù)據(jù)類型和參數(shù)的順序等方面的不同來區(qū)分。在編譯期間,Java編譯器要檢查每個(gè)方法所用的參數(shù)數(shù)目和類型,然后調(diào)用正確的方法。
【示例程序C5_5.java】加法運(yùn)算重載的例子。
public
classC5_5{
static
intadd(int
a,int
b)//重載的方法1
{return(a+b);}
static
doubleadd(double
x,double
y)//重載的方法2
{return(x+y);}
static
doubleadd(double
x,double
y,double
z)//重載的方法3
{return(x+y+z);}
public
static
voidmain(String[]args)
{
System.out.println("Sumis:"+add(8.5,2.3));
System.out.println("Sumis:"+add(21,38));
System.out.println("Sumis:"+add(8.5,2.3,8.5+2.3));
}//main
}
運(yùn)行結(jié)果如下:Sumis:10.8
Sumis:59
Sumis:21.65.3.3覆蓋
由于面向?qū)ο笙到y(tǒng)中的繼承機(jī)制,子類可以繼承父類的方法。但是,子類的某些特征可能與從父類中繼承來的特征有所不同,為了體現(xiàn)子類的這種個(gè)性,Java允許子類對父類的同名方法重新進(jìn)行定義,即在子類中可以定義與父類中已定義的方法同名而內(nèi)容不同的方法。這種多態(tài)被稱為覆蓋(overload)。
由于覆蓋的同名方法存在于子類對父類的關(guān)系中,因此只需在方法引用時(shí)指明引用的是父類的方法還是子類的方法,就可以很容易地把它們區(qū)分開來。具體應(yīng)用請參閱5.4.3節(jié)。
5.4繼承機(jī)制
5.4.1繼承的概念
繼承所表達(dá)的就是對象類之間的相交關(guān)系,它使得某類對象可以繼承另外一類對象的數(shù)據(jù)成員和成員方法。若類B繼承類A,則屬于類B的對象便具有類A的全部或部分性質(zhì)(數(shù)據(jù)屬性)和功能(操作),我們稱被繼承的類A為基類、父類或超類,而稱繼承類B為A的派生類或子類。父類與子類的層次關(guān)系如果類B具有類A的全部屬性和方法,并又具有自己特有的某些屬性和方法,則把類A稱做一般類,把類B叫做類A的特殊類。
繼承避免了對一般類和特殊類之間共同特征進(jìn)行的重復(fù)描述。同時(shí),通過繼承可以清晰地表達(dá)每一項(xiàng)共同特征所適應(yīng)的概念范圍——在一般類中定義的屬性和操作適用于這個(gè)類本身以及它以下的每一層特殊類的全部對象。運(yùn)用繼承原則使得系統(tǒng)模型更簡潔、清晰。
5.4.2繼承的特征
一般來說,繼承具有下述特征:
(1)繼承關(guān)系是傳遞的。若類C繼承類B,類B繼承類A,則類C既有從類B那里繼承下來的屬性與方法,也有從類A那里繼承下來的屬性與方法,還可以有自己新定義的屬性和方法。繼承來的屬性和方法盡管是隱式的,但仍是類C的屬性和方法。繼承是在一些比較一般的類的基礎(chǔ)上構(gòu)造、建立和擴(kuò)充新類的最有效的手段。
(2)繼承簡化了人們對事物的認(rèn)識和描述,能清晰地體現(xiàn)相關(guān)類間的層次結(jié)構(gòu)關(guān)系。
(3)繼承通過增強(qiáng)一致性來減少模塊間的接口和界面,大大增加了程序的易維護(hù)性。
(4)繼承提供了軟件復(fù)用功能。若類B繼承類A,那么建立類B時(shí)只需要再描述與基類(類A)不同的少量特征(數(shù)據(jù)成員和成員方法)即可。這能減小代碼和數(shù)據(jù)的冗余度,增加程序的重用性。
(5)提供多重繼承機(jī)制。從理論上來說,一個(gè)類可以是多個(gè)一般類的特殊類,它可以從多個(gè)一般類中繼承屬性與方法,這便是多重繼承。Java出于安全性和可靠性的考慮,僅支持單重繼承,而通過使用接口機(jī)制來實(shí)現(xiàn)多重繼承。
在這個(gè)模型中,“本科生”、“研究生”、“脫產(chǎn)研究生”都為單繼承,而“在職研究生”為多重繼承,因?yàn)樗粌H繼承“學(xué)生”/“研究生”的屬性和行為,還繼承“教師”的屬性和行為。單重繼承與多重繼承的例子5.4.3Java用extends指明繼承關(guān)系
在Java程序設(shè)計(jì)中,繼承是通過extends關(guān)鍵字來實(shí)現(xiàn)的。在定義類時(shí)使用extends關(guān)鍵字指明新定義類的父類,新定義的類稱為指定父類的子類,這樣就在兩個(gè)類之間建立了繼承關(guān)系。這個(gè)新定義的子類可以從父類那里繼承所有非private的屬性和方法作為自己的成員。實(shí)際上,在定義一個(gè)類而不給出extends關(guān)鍵字及父類名時(shí),默認(rèn)這個(gè)類是系統(tǒng)類object的子類。
1.?dāng)?shù)據(jù)成員的繼承
子類可以繼承父類的所有非私有的數(shù)據(jù)成員。
【示例程序C5_6.java】數(shù)據(jù)成員的繼承。
classA1
{int
x=25;
private
int
z;
//不能被子類繼承的私有數(shù)據(jù)成員z
}
classC5_6extendsA1//A1是C5_6的父類,C5_6是A1的子類
{public
static
voidmain(String[]argS)
{C5_6p=newC5_6();
System.out.println(“p.x=”+p.x);
//輸出繼承來的數(shù)據(jù)成員的值
/*System.out.println(“p.z=”+p.z);錯(cuò),不能繼承private修飾的z*/
}
}//
運(yùn)行結(jié)果:
p.x=252.?dāng)?shù)據(jù)成員的隱藏
數(shù)據(jù)成員的隱藏是指在子類中重新定義一個(gè)與父類中已定義的數(shù)據(jù)成員名完全相同的數(shù)據(jù)成員,即子類擁有了兩個(gè)相同名字的數(shù)據(jù)成員,一個(gè)是繼承自父類的,另一個(gè)是自己定義的。當(dāng)子類引用這個(gè)同名的數(shù)據(jù)成員時(shí),默認(rèn)操作是引用它自己定義的數(shù)據(jù)成員,而把從父類那里繼承來的數(shù)據(jù)成員“隱藏”起來。當(dāng)子類要引用繼承自父類的同名數(shù)據(jù)成員時(shí),可使用關(guān)鍵字super引導(dǎo)。
【示例程序C5_7.java】數(shù)據(jù)成員的隱藏。
classA11
{int
x=8;}
//父類中定義了數(shù)據(jù)成員x
classC5_7extendsA11
{int
x=24;//子類中也定義了數(shù)據(jù)成員x
public
static
voidmain(String[]argS)
{
int
s1,s2;
A11p=newA11();//創(chuàng)建父類p的對象
C5_7p1=newC5_7();//創(chuàng)建子類p1的對象
s1=p.x;
s2=p1.x;
//子類對象引用自己的數(shù)據(jù)成員,把父類數(shù)據(jù)成員“隱藏”起來
System.out.println("s1="+s1);
System.out.println("s2="+s2);
}//main
}運(yùn)行結(jié)果:s1=8s2=24A11類的p對象的存儲(chǔ)區(qū)C5_7類的p1對象的存儲(chǔ)區(qū)x=8父類數(shù)據(jù)成員
x=8父類數(shù)據(jù)成員,隱藏x=24子類數(shù)據(jù)成員3.成員方法的繼承
子類可以繼承父類的非私有成員方法。
【示例程序C5_8.java】成員方法的繼承。
classA2
{int
x=0,y=1;
voidMyp(){System.out.println("x="+x+"y="+y);}
private
voidPrintme()
{System.out.println("x="+x+"y="+y);}
}
public
classC5_8extendsA2
{public
static
voidmain(Stringarg[])
{
int
z=3;
C5_8p1=newC5_8();p1.Myp();
//p1.Printme();錯(cuò),不能繼承父類的private方法
}}
運(yùn)行結(jié)果:x=0y=14.成員方法的覆蓋
子類可以重新定義與父類同名的成員方法,實(shí)現(xiàn)對父類方法的覆蓋(overload)。方法的覆蓋與數(shù)據(jù)成員的隱藏的不同之處在于:子類隱藏父類的數(shù)據(jù)成員只是使之不可見,父類同名的數(shù)據(jù)成員在子類對象中仍然占有自己獨(dú)立的內(nèi)存空間;子類方法對父類同名方法的覆蓋將清除父類方法占用的內(nèi)存,從而使父類方法在子類對象中不復(fù)存在。
方法的覆蓋中需要注意:子類在重新定義父類已有的方法時(shí),應(yīng)保持與父類完全相同的方法名、返回值類型和參數(shù)列表,否則就不是方法的覆蓋,而是子類定義自己特有的方法,與父類的方法無關(guān)。
【示例程序C5_9.java】成員方法的覆蓋。
classA3
{int
x=10;int
y=31;
public
voidPrintme(){System.out.println("x="+x+"y="+y);}
}
public
classC5_9extendsA3
{int
z=35;
public
voidPrintme()//子類中定義了與父類同名的成員方法,實(shí)現(xiàn)覆蓋
{System.out.println("z="+z);}
public
static
voidmain(Stringarg[])
{A3p2=newA3();
//創(chuàng)建父類p2的對象
C5_9p1=newC5_9();
//創(chuàng)建子類p1的對象
p1.Printme();//子類對象引用子類方法,覆蓋了父類的同名方法
p2.Printme();//父類對象引用父類方法
}
}
運(yùn)行結(jié)果:z=35x=10y=315.4.4this與super
1.this的使用場合
當(dāng)成員方法的形參名與數(shù)據(jù)成員名相同,或者成員方法的局部變量名與數(shù)據(jù)成員名相同時(shí),在方法內(nèi)借助this來明確表示引用的是類的數(shù)據(jù)成員,而不是形參或局部變量,從而提高程序的可讀性。簡單地說,this代表了當(dāng)前對象的一個(gè)引用,可將其理解為對象的另一個(gè)名字,通過這個(gè)名字可以順利地訪問對象,修改對象的數(shù)據(jù)成員,調(diào)用對象的方法。歸納起來,this的使用場合有下述三種:(1)用來訪問當(dāng)前對象的數(shù)據(jù)成員,其使用形式如下:
this.數(shù)據(jù)成員
(2)用來訪問當(dāng)前對象的成員方法,其使用形式如下:
this.成員方法(參數(shù))
(3)當(dāng)有重載的構(gòu)造方法時(shí),用來引用同類的其他構(gòu)造方法,其使用形式如下:
this(參數(shù))
【示例程序C5_10.java】this的使用。
classA4
{int
x=0;int
y=1;
public
voidPrintme()
{System.out.println(“x=”+x+“y=”+y);
System.out.println(“Iaman”+this.getClass().getName());
/*用this來訪問當(dāng)前對象的成員方法,通過this表示當(dāng)前對象,來打印當(dāng)前
對象的類名。其中g(shù)etClass()和getName()是系統(tǒng)類庫中提供的方法*/
}
}
public
classC5_10extendsA4
{public
static
voidmain(Stringarg[])
{C5_10p1=newC5_10();
p1.Printme();
}
}
運(yùn)行結(jié)果:x=0y=1Iamanch5.C5_10
【示例程序C5_11.java】this的使用。
classAlassArea
{double
x,y;
doublearea(double
x,double
y)
{
double
s;
this.x=x;//借助this來表示引用的是類數(shù)據(jù)成員
this.y=y;s=this.x*this.y;
return
s;
}}
public
classC5_11extendsAlassArea
{public
static
voidmain(String[]args)
{
double
a=2.2,b=3.1,z;
C5_11ss=newC5_11();//創(chuàng)建ss的對象
z=ss.area(a,b);//引用父類對象的成員方法求面積
System.out.println("z="+z);
}}運(yùn)行結(jié)果:z=6.820000000000001
【示例程序C5_12.java】計(jì)算圓的面積和周長。
public
classC5_12
{public
static
voidmain(String[]args)
{
double
x;Circlecir=newCircle(5.0);
x=cir.area();System.out.println("圓的面積="+x);
x=cir.perimeter();System.out.println("圓的周長="+x);
}}
classCircle
{double
r;//定義半徑
final
double
PI=3//定義圓周率
publicCircle(double
r)//類的構(gòu)造方法
{this.r=r;}//通過構(gòu)造方法給r賦值
doublearea(){return
PI*r*r;}//計(jì)算圓面積的方法
doubleperimeter()//計(jì)算圓周長的方法
{return2*(this.area()/r);}//使用this變量獲取圓的面積
}
運(yùn)行結(jié)果:圓的面積=78.53981633974999
圓的周長=31.415926535899995
2.super的使用場合
super代表的就是直接父類。假設(shè)類A派生出子類B,類B又派生出自己的子類C,則B是C的直接父類,而A是C的祖先類。若子類的數(shù)據(jù)成員或成員方法名與父類的數(shù)據(jù)成員或成員方法名相同,當(dāng)要調(diào)用父類的同名方法或使用父類的同名數(shù)據(jù)成員時(shí),可用關(guān)鍵字super來指明父類的數(shù)據(jù)成員和方法。
super的使用方法:
(1)用來訪問直接父類隱藏的數(shù)據(jù)成員:super.數(shù)據(jù)成員
(2)用來調(diào)用直接父類中被覆蓋的成員方法:super.成員方法(參數(shù))
(3)用來調(diào)用直接父類的構(gòu)造方法:super(參數(shù))
【示例程序C5_13.java】super的使用。
classA5
{int
x=4;int
y=1;
public
voidprintme()
{System.out.println("x="+x+"y="+y);
System.out.println("classname:"+this.getClass().getName());
}}
public
classC5_13extendsA5
{int
x;
public
voidprintme()
{
int
z=super.x+6;//引用父類A5的數(shù)據(jù)成員
super.printme();//調(diào)用父類A5的成員方法
System.out.println("Iaman"+this.getClass().getName());
x=5;
System.out.println("z="+z+"x="+x);
//打印子類的數(shù)據(jù)成員
}public
static
voidmain(Stringarg[])
{
int
k;
A5p1=newA5();C5_13p2=newC5_13();
p1.printme();p2.printme();
//super.printme();//錯(cuò),在static方法中不能引用非static成員方法
//k=super.x+23;//錯(cuò),在static方法中不能引用非static數(shù)據(jù)成員
}}
運(yùn)行結(jié)果:
x=4y=1classname:ch5.A5x=4y=1classname:ch5.C5_13Iamanch5.C5_13z=10x=55.4.5構(gòu)造方法的重載與繼承
1.構(gòu)造方法的重載
一個(gè)類的若干個(gè)構(gòu)造方法之間可以相互調(diào)用。當(dāng)一個(gè)構(gòu)造方法需要調(diào)用另一個(gè)構(gòu)造方法時(shí),可以使用關(guān)鍵字this,同時(shí)這個(gè)調(diào)用語句應(yīng)該是整個(gè)構(gòu)造方法的第一條可執(zhí)行語句。使用關(guān)鍵字this來調(diào)用同類的其他構(gòu)造函數(shù)時(shí),優(yōu)點(diǎn)同樣是可以最大限度地提高對已有代碼的利用程度,提高程序的抽象度和封裝性,減少程序的維護(hù)工作量。
【示例程序C5_14.java】構(gòu)造方法的重載。
classAddclass
{
public
int
x=0,y=0,z=0;
//以下是多個(gè)同名不同參數(shù)的構(gòu)造方法
Addclass(int
x){this.x=x;}//可重載的構(gòu)造方法1
Addclass(int
x,int
y)//可重載的構(gòu)造方法2
{this(x);//當(dāng)前構(gòu)造方法調(diào)用可重載的構(gòu)造方法1
this.y=y;
}
Addclass(int
x,int
y,int
z)//可重載的構(gòu)造方法3
{this(x,y);
//當(dāng)前構(gòu)造方法調(diào)用可重載的構(gòu)造方法2
this.z=z;
}
public
intadd(){return
x+y+z;}
}public
classC5_14
{
public
static
voidmain(String[]args)
{
Addclassp1=newAddclass(2,3,5);
Addclassp2=newAddclass(10,20);
Addclassp3=newAddclass(1);
System.out.println("x+y+z="+p1.add());
System.out.println("x+y="+p2.add());
System.out.println("x="+p3.add());
}
}
運(yùn)行結(jié)果:x+y+z=10x+y=30x=12.構(gòu)造方法的繼承
子類可以繼承父類的構(gòu)造方法,構(gòu)造方法的繼承遵循以下的原則:
(1)子類無條件地繼承父類的不含參數(shù)的構(gòu)造方法。
(2)如果子類自己沒有構(gòu)造方法,則它將繼承父類的無參數(shù)構(gòu)造方法,并將這些方法作為自己的構(gòu)造方法;如果子類自己定義了構(gòu)造方法,則在創(chuàng)建新對象時(shí),它將先執(zhí)行繼承自父類的無參數(shù)構(gòu)造方法,然后再執(zhí)行自己的構(gòu)造方法。
(3)對于父類的含參數(shù)構(gòu)造方法,子類可以通過在自己的構(gòu)造方法中使用super關(guān)鍵字來調(diào)用它,但這個(gè)調(diào)用語句必須是子類構(gòu)造方法的第一條可執(zhí)行語句。
【示例程序C5_15.java】構(gòu)造方法的繼承。
classAddclass2
{public
int
x=0,y=0,z=0;
Addclass2(int
x){this.x=x;}//父類可重載的構(gòu)造方法1
Addclass2(int
x,int
y)//父類可重載的構(gòu)造方法2
{this.x=x;this.y=y;}
Addclass2(int
x,int
y,int
z)//父類可重載的構(gòu)造方法3
{this.x=x;this.y=y;this.z=z;}
public
intadd(){return
x+y+z;}
}
public
classC5_15extendsAddclass2
{int
a=0,b=0,c=0;
C5_15(int
x){super(x);
a=x+7;}
//子類可重載的構(gòu)造方法1
C5_15(int
x,int
y)//子類可重載的構(gòu)造方法2
{super(x,y);
a=x+5;b=y+5;
}
C5_15(int
x,int
y,int
z)//子類可重載的構(gòu)造方法3
{super(x,y,z);
a=x+4;b=y+4;c=z+4;
}
public
intadd()
{System.out.println("super:x+y+z="+super.add());
return
a+b+c;
}
public
static
voidmain(String[]args)
{
C5_15p1=newC5_15(2,3,5);
C5_15p2=newC5_15(10,20);
C5_15p3=newC5_15(1);
System.out.println("a+b+c="+p1.add());
System.out.println("a+b="+p2.add());
System.out.println("a="+p3.add());
}
}
運(yùn)行結(jié)果:super:x+y+z=10
a+b+c=22
super:x+y+z=30
a+b=40
super:x+y+z=1
a=85.4.6向方法傳遞對象
傳遞給方法的參數(shù)可以是表達(dá)式、對象等,傳遞給方法的參數(shù)若是變量,則只能由實(shí)參傳遞給形參,而不能由形參帶回,即它是一種單向值傳遞。也就是說,在方法的引用過程中,對于形參變量值的修改并不影響實(shí)參變量值。但是,傳遞給方法的參數(shù)若是對象,則實(shí)參與形參的對象的引用指向同一個(gè)對象,因此成員方法中對對象的數(shù)據(jù)成員的修改,會(huì)使實(shí)參對象的數(shù)據(jù)成員值也發(fā)生同樣的變化。這種參數(shù)的傳遞方式被稱為“雙向地址傳遞”。
【示例程序C5_16.java】方法中的參數(shù)是對象時(shí)的情形。
classStudent1
{publicStringName;public
int
age=16;public
int
score=0;
public
voidShowStudent(){System.out.println("Name:"+Name);
System.out.println("age:"+age);System.out.println("score:"+score);
}}
public
classC5_16{
staticvoidstudentAttributes(Student1s,StringName,int
age,int
score)
{s.Name=Name;s.age=age;s.score=score;}
public
static
voidmain(String[]args)
{Student1st1=newStudent1();//創(chuàng)建st1的對象
Student1st2=newStudent1();//創(chuàng)建st2的對象
studentAttributes(st1,"zhang",23,81);//對象st1作為實(shí)參
studentAttributes(st2,"li",24,90);//對象st2作為實(shí)參
st1.ShowStudent();//執(zhí)行此方法可發(fā)現(xiàn)st1的對象將新值帶回
st2.ShowStudent();//再次執(zhí)行此方法可發(fā)現(xiàn)st2的對象將新值帶回
}}運(yùn)行結(jié)果:Name:zhangage:23score:81
Name:liage:24score:90st1對象存儲(chǔ)區(qū)st2對象存儲(chǔ)區(qū)S形參S形參Name=Age=16Score=0Showstudent(){…}Name=Age=16Score=0Showstudent(){…}雙向地址傳遞
5.4.7繼承與封裝的關(guān)系
在面向?qū)ο笙到y(tǒng)中,封裝性主要指的是對象的封裝性,即將屬于某一類的一個(gè)具體的對象封裝起來,使其數(shù)據(jù)和操作成為一個(gè)整體。引入了繼承機(jī)制,對象依然是封裝得很好的實(shí)體,其他對象與它進(jìn)行通信的途徑是發(fā)送消息;
類機(jī)制是一種靜態(tài)機(jī)制,不管是基類還是派生類,對于對象來說,它仍然是一個(gè)類的實(shí)例,既可能是基類的實(shí)例,也可能是派生類的實(shí)例。因此,繼承機(jī)制的引入絲毫沒有影響對象的封裝性;繼承和封裝機(jī)制具有一定的相似性,它們都是一種共享代碼的手段。
繼承是一種靜態(tài)共享代碼的手段,通過派生類對象的創(chuàng)建,可以接受某一消息,啟動(dòng)其基類所定義的代碼段,從而使基類和派生類共享這一段代碼。
封裝機(jī)制所提供的是一種動(dòng)態(tài)共享代碼的手段,通過封裝,可將一段代碼定義在一個(gè)類中,在另一個(gè)類所定義的操作中,可以通過創(chuàng)建該類的實(shí)例,并向它發(fā)送消息而啟動(dòng)這一段代碼,同樣也達(dá)到共享的目的。
5.5抽象類、接口與包
抽象類體現(xiàn)數(shù)據(jù)抽象的思想,是實(shí)現(xiàn)程序多態(tài)性的一種手段。接口則是Java中實(shí)現(xiàn)多重繼承的唯一途徑。包是一個(gè)更大的程序單位,主要實(shí)現(xiàn)軟件復(fù)用。
5.5.1抽象類
具有相同特征卻彼此獨(dú)立的幾個(gè)類例如:抽象類及其應(yīng)用三個(gè)類都要計(jì)算面積與周長,雖然公式不同但目標(biāo)相同。因此,可以為這三個(gè)類抽象出一個(gè)父類,在父類里定義圓、三角形和矩形三個(gè)類共同的數(shù)據(jù)成員及成員方法。把計(jì)算面積與周長的成員方法名放在父類給予說明,再將具體的計(jì)算公式在子類中實(shí)現(xiàn),通過父類就大概知道子類所要完成的任務(wù),這種結(jié)構(gòu)就是抽象類的概念。
Java程序用抽象類(AbstractClass)來實(shí)現(xiàn)自然界的抽象概念。抽象類刻畫了公有行為的特征,并通過繼承機(jī)制傳送給它的派生類。在抽象類中定義的方法稱為抽象方法,這些方法只有方法頭的聲明,而用一個(gè)分號來代替方法體的定義,即只定義成員方法的接口形式,而沒有具體操作。只有派生類對抽象成員方法的重定義才真正實(shí)現(xiàn)與該派生類相關(guān)的操作。在各子類繼承了父類的抽象方法之后,再分別用不同的語句和方法體來重新定義它,形成若干個(gè)名字相同、返回值相同、參數(shù)列表也相同,目的一致但是具體實(shí)現(xiàn)有一定差別的方法。抽象類中定義抽象方法的目的是實(shí)現(xiàn)一個(gè)接口,即所有的子類對外都呈現(xiàn)一個(gè)相同名字的方法。抽象類是它的所有子類的公共屬性的集合,是包含一個(gè)或多個(gè)抽象方法的類。使用抽象類的一大優(yōu)點(diǎn)就是可以充分利用這些公共屬性來提高開發(fā)和維護(hù)程序的效率。對于抽象類與抽象方法的限制如下:
(1)凡是用abstract修飾符修飾的類被稱為抽象類。凡是用abstract修飾符修飾的成員方法被稱為抽象方法。
(2)抽象類中可以有零個(gè)或多個(gè)抽象方法,也可以包含非抽象的方法。
(3)抽象類中可以沒有抽象方法,但是,有抽象方法的類必須是抽象類。(4)對于抽象方法來說,在抽象類中只指定其方法名及其類型,而不書寫其實(shí)現(xiàn)代碼。
(5)抽象類可以派生子類,在抽象類派生的子類中必須實(shí)現(xiàn)抽象類中定義的所有抽象方法。
(6)抽象類不能創(chuàng)建對象,創(chuàng)建對象的工作由抽象類派生的子類來實(shí)現(xiàn)。
(7)如果父類中已有同名的abstract方法,則子類中就不能再有同名的抽象方法。
(8)?abstract不能與final并列修飾同一個(gè)類。
(9)?abstract不能與private、static、final或native并列修飾同一個(gè)方法。
【示例程序C5_17.java】抽象類的應(yīng)用。
public
classC5_17{
public
static
voidmain(String[]args)
{
SquareBox=newSquare(5,15,25,25);
System.out.println("SquareArea="+Box.getArea()+"
Perimeter="+Box.getPerimeter());
Triangletri=newTriangle(5,50,8,4);
System.out.println("TriangleArea="+tri.getArea()+"
Perimeter="+tri.getPerimeter());
Circle1Oval=newCircle1(5,90,25,25);
System.out.println("CircleArea="+Oval.getArea()+"
Perimeter="+Oval.getPerimeter());
}
}
abstract
classShapes//定義一個(gè)抽象類Shapes
{
public
int
x,y;//x、y為畫圖的坐標(biāo)
public
int
width,height;
publicShapes(int
x,int
y,int
width,int
height)
{this.x=x;this.y=y;
this.width=width;
this.height=height;
}
abstract
doublegetArea();//求圖形面積的抽象方法
abstract
doublegetPerimeter();//求圖形周長的抽象方法
}
classSquareextendsShapes//由抽象類Shapes派生的子類——矩形類
{
public
doublegetArea(){return(width*height);}
public
doublegetPerimeter(){return(2*width+2*height);}
publicSquare(int
x,int
y,int
width,int
height)
{super(x,y,width,height);}
}
classTriangleextendsShapes//由抽象類Shapes派生的子類——三角形類
{
public
double
c;//斜邊
public
doublegetArea(){return(0.5*width*height);}
public
doublegetPerimeter(){return(width+height+c);}
publicTriangle(int
x,int
y,int
base,int
height)
{super(x,y,base,height);
c=Math.sqrt(width*width+height*height);
}
}
classCircle1extendsShapes//由抽象類Shapes派生的子類——圓類
{
public
double
r;//半徑
public
doublegetArea(){return(r*r*Math.PI);}
public
doublegetPerimeter(){return(2*Math.PI*r);}
publicCircle1(int
x,int
y,int
width,int
height)
{super(x,y,width,height);
r=(double)width/2.0;
}
}5.5.2接口
多重繼承是指一個(gè)子類可以有多個(gè)直接父類。
Java出于安全性,不支持類間的多重繼承,而只支持單重繼承。然而在解決實(shí)際問題的過程中,僅僅依靠單重繼承并不能將復(fù)雜的問題描述清楚。因此,Java語言提供接口來實(shí)現(xiàn)多重繼承機(jī)制。
1.聲明接口
格式如下:
[修飾符]interface接口名[extends父接口名列表]
{常量數(shù)據(jù)成員聲明
抽象方法聲明
}
說明:
(1)?interface是聲明接口的關(guān)鍵字,可以把它看成一個(gè)特殊類。
(2)接口名要求符合Java標(biāo)識符規(guī)定。
(3)修飾符有兩種:public和默認(rèn)。public修飾的接口是公共接口,可以被所有的類和接口使用;默認(rèn)修飾符的接口只能被同一個(gè)包中的其他類和接口使用。
(4)父接口名列表。接口也具有繼承性。定義一個(gè)接口時(shí)可以通過extends關(guān)鍵字聲明該接口是某個(gè)已經(jīng)存在的父接口的派生接口,它將繼承父接口的所有屬性和方法。與類的繼承不同的是,一個(gè)接口可以有一個(gè)以上的父接口,它們之間用逗號分隔。
(5)常量數(shù)據(jù)成員聲明。常量數(shù)據(jù)成員前可以有也可以沒有修飾符。修飾符是publicfinalstatic和fina1static;接口中的數(shù)據(jù)成員都是用final修飾的常量,
寫法如下:修飾符數(shù)據(jù)成員類型數(shù)據(jù)成員名=常量值
或
數(shù)據(jù)成員名=常量值
例如:publicfinalstaticdoublePI=3.14159;
finalstaticinta=9;
intSUM=100;(等價(jià)于finalstaticintSUM=100;)(6)抽象方法聲明。接口中的方法都是用abstract修飾的抽象方法。在接口中只能給出這些抽象方法的方法名、返回值類型和參數(shù)列表,而不能定義方法體,即這些接口僅僅是規(guī)定了一組信息交換、傳輸和處理的“接口”。
其格式如下:返回值類型方法名(參數(shù)列表);
其中,接口中的方法默認(rèn)為publicabstract方法。接口中方法的方法體可以由Java語言書寫,也可以由其他語言書寫。方法體由其他語言書寫時(shí),接口方法由native修飾符修飾。定義接口與定義類非常相似??梢园呀涌诶斫獬梢环N特殊的類,即由常量和抽象方法組成的特殊類。一個(gè)類只能有一個(gè)父類,但是它可以同時(shí)實(shí)現(xiàn)若干個(gè)接口。把接口理解成特殊的類,那么這個(gè)類利用接口就獲得了多個(gè)父類,即實(shí)現(xiàn)了多重繼承。
接口定義僅僅是實(shí)現(xiàn)某一特定功能的對外接口和規(guī)范,而不能真正地實(shí)現(xiàn)這個(gè)功能,這個(gè)功能的真正實(shí)現(xiàn)是在“繼承”這個(gè)接口的各個(gè)類中完成的,即要由這些類來具體定義接口中各抽象方法的方法體。因而在Java中,通常把對接口功能的“繼承”稱為“實(shí)現(xiàn)”。2.定義接口注意事項(xiàng)
(1)接口定義用關(guān)鍵字interface,而不是用class。
(2)接口中定義的數(shù)據(jù)成員全是finalstatic修飾的,即常量。
(3)接口中沒有自身的構(gòu)造方法,所有成員方法都是抽象方法。
(4)接口也具有繼承性,可以通過extends關(guān)鍵字聲明該接口的父接口。
3.類實(shí)現(xiàn)接口的注意事項(xiàng)
(1)在類中,用implements關(guān)鍵字就可以調(diào)用接口。一個(gè)類若要調(diào)用多個(gè)接口,可在implements后用逗號隔開多個(gè)接口的名字。
(2)如果實(shí)現(xiàn)某接口的類不是abstract的抽象類,則在類的定義部分必須實(shí)現(xiàn)指定接口的所有抽象方法,即為所有抽象方法定義方法體,而且方法頭部分應(yīng)該與接口中的定義完全一致,即有完全相同的返回值和參數(shù)列表。
(3)如果實(shí)現(xiàn)某接口的類是abstract的抽象類,則它可以不實(shí)現(xiàn)該接口所有的方法。但是對于這個(gè)抽象類的任何一個(gè)非抽象的子類而言,它們的父類所實(shí)現(xiàn)的接口中的所有抽象方法都必須有實(shí)在的方法體。這些方法體可以來自抽象的父類,也可以來自子類自身,但是不允許存在未被實(shí)現(xiàn)的接口方法。這主要體現(xiàn)了非抽象類中不能存在抽象方法的原則。
(4)接口的抽象方法的訪問限制符都已指定為public,所以類在實(shí)現(xiàn)方法時(shí),必須顯式地使用public修飾符,否則將被系統(tǒng)警告為縮小了接口中定義的方法的訪問控制范圍。
【示例程序C5_18.java】將例C5_17.java改寫為接口程序。
public
classC5_18{
public
static
voidmain(String[]args)
{
SquareBox=newSquare(5,15,25,25);
System.out.println("SquareArea="+Box.getArea()+"Perimeter="+Box.getPerimeter());
Triangletri=newTriangle(5,50,8,4);
System.out.println("TriangleArea="+tri.getArea()+"Perimeter="+tri.getPerimeter());
CircleOval=newCircle(5,90,25,25);
System.out.println("CircleArea="+Oval.getArea()+"Perimeter="+Oval.getPerimeter());
}
}interfaceShapes//定義一個(gè)接口
{abstract
doublegetArea();//自動(dòng)被定義為public
doublegetPerimeter();
}
classSquareimplementsShapes//類要實(shí)現(xiàn)接口
{public
int
x,y;
public
int
width,height;
public
doublegetArea(){return(width*height);}
public
doublegetPerimeter(){return(2*width+2*height);}
publicSquare(int
x,int
y,int
width,int
height)
{
this.x=x;
this.y=y;
this.width=width;
this.height=height;
}
}
classTriangleimplementsShapes//類要實(shí)現(xiàn)接口
{
public
int
x,y;
public
int
width,height;
public
double
c;
public
doublegetArea(){return(0.5*width*height);}
public
doublegetPerimeter(){return(width+height+c);}
publicTriangle(int
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2024年短視頻平臺(tái)與內(nèi)容創(chuàng)作者之間的合作協(xié)議
- 2024版三方入股合作協(xié)議書范本
- 浙江省初中學(xué)業(yè)水平考試模擬語文卷三套【附參考答案】
- 2024汽車租賃合同范文格式
- 2024年高端住宅底商返租合同
- 2024施工項(xiàng)目用重型挖掘機(jī)租賃合同3篇
- 2024年高速公路廣告投放權(quán)轉(zhuǎn)讓合同
- 1學(xué)會(huì)尊重 第二課時(shí) (說課稿) -2023-2024學(xué)年道德與法治六年級下冊統(tǒng)編版
- 2024物業(yè)保安服務(wù)外包規(guī)定合同
- 2024年魚種養(yǎng)殖技術(shù)引進(jìn)與購銷合作合同3篇
- 獾子油壓瘡護(hù)理
- 2025年中考語文備考之名著導(dǎo)讀:《水滸傳》主要人物梳理
- 中華人民共和國殘疾評定表
- 小學(xué)科學(xué)學(xué)情分析報(bào)告總結(jié)
- 2024年國考行測真題-言語理解與表達(dá)真題及完整答案1套
- 2024屆高考數(shù)學(xué)復(fù)習(xí) 立體幾何考情分析及備考策略
- 基于課程標(biāo)準(zhǔn)的學(xué)生創(chuàng)新素養(yǎng)培育的學(xué)科教學(xué)改進(jìn)研究課題申報(bào)評審書
- 醫(yī)療人員廉潔從業(yè)九項(xiàng)準(zhǔn)則
- 健康中國產(chǎn)業(yè)園規(guī)劃方案
- (2024年)二年級上冊音樂
- 2024屆高考英語一輪復(fù)習(xí)讀后續(xù)寫脫險(xiǎn)類續(xù)寫講義
評論
0/150
提交評論