class A\x0d\x0a{\x0d\x0a int x=12;//被繼承\(zhòng)x0d\x0a static double y=34;//被繼承\(zhòng)x0d\x0a final int xx=123;//被繼承,不能修改xx的值,一經(jīng)初始化就被看作常量使用\x0d\x0a private float r=12.0f;//不能被繼承\(zhòng)x0d\x0a A()//不能被繼承\(zhòng)x0d\x0a {\x0d\x0a //====構(gòu)造方法體\x0d\x0a }\x0d\x0a public void f(){}//被繼承\(zhòng)x0d\x0a protected void d(){}//被繼承\(zhòng)x0d\x0a private void k(){}//不能被繼承\(zhòng)x0d\x0a final void gg(){}//被繼承,不能被覆蓋\x0d\x0a void w(){}//被繼承\(zhòng)x0d\x0a//也就是說構(gòu)造函數(shù)和被關(guān)鍵字private修飾的東西都不能被繼承 \x0d\x0a}\x0d\x0aclass B extends A\x0d\x0a{\x0d\x0a //========\x0d\x0a //=======\x0d\x0a //A的類體\x0d\x0a}
在原州等地區(qū),都構(gòu)建了全面的區(qū)域性戰(zhàn)略布局,加強發(fā)展的系統(tǒng)性、市場前瞻性、產(chǎn)品創(chuàng)新能力,以專注、極致的服務(wù)理念,為客戶提供網(wǎng)站制作、成都網(wǎng)站設(shè)計 網(wǎng)站設(shè)計制作定制設(shè)計,公司網(wǎng)站建設(shè),企業(yè)網(wǎng)站建設(shè),品牌網(wǎng)站制作,營銷型網(wǎng)站,成都外貿(mào)網(wǎng)站制作,原州網(wǎng)站建設(shè)費用合理。
繼承是面向?qū)ο缶幊碳夹g(shù)的一塊基石,因為它允許創(chuàng)建分等級層次的類。運用繼承,你能夠創(chuàng)建一個通用類,它定義了一系列相關(guān)項目的一般特性。該類可以被更具體的類繼承,每個具體的類都增加一些自己特有的東西。在Java 術(shù)語學(xué)中,被繼承的類叫超類(superclass ),繼承超類的類叫子類(subclass )。因此,子類是超類的一個專門用途的版本,它繼承了超類定義的所有實例變量和方法,并且為它自己增添了獨特的元素。
繼承一個類,只要用extends 關(guān)鍵字把一個類的定義合并到另一個中就可以了。為了理解怎樣繼承,讓我們從簡短的程序開始。下面的例子創(chuàng)建了一個超類A和一個名為B的子類。注意怎樣用關(guān)鍵字extends 來創(chuàng)建A的一個子類。
// A simple example of inheritance.
// Create a superclass.
class A {
int i, j;
void showij() {
System.out.println("i and j: " + i + " " + j);
}
}
class B extends A {
int k;
void showk() {
System.out.println("k: " + k);
}
void sum() {
System.out.println("i+j+k: " + (i+j+k));
}
}
class SimpleInheritance {
public static void main(String args[]) {
A superOb = new A();
B subOb = new B();
System.out.println("Contents of superOb: ");
superOb.showij();
System.out.println();
subOb.i = 7;
subOb.j = 8;
subOb.k = 9;
System.out.println("Contents of subOb: ");
subOb.showij();
subOb.showk();
System.out.println();
System.out.println("Sum of i, j and k in subOb:");
subOb.sum();
}
}
該程序的輸出如下:
Contents of superOb:
i and j: 10 20
Contents of subOb:
i and j: 7 8
k: 9
Sum of i, j and k in subOb:
i+j+k: 24
像你所看到的,子類B包括它的超類A中的所有成員。這是為什么subOb 可以獲取i和j 以及調(diào)用showij( ) 方法的原因。同樣,sum( ) 內(nèi)部,i和j可以被直接引用,就像它們是B的一部分。
盡管A是B的超類,它也是一個完全獨立的類。作為一個子類的超類并不意味著超類不能被自己使用。而且,一個子類可以是另一個類的超類。聲明一個繼承超類的類的通常形式如下:
class subclass-name extends superclass-name {
// body of class
}
你只能給你所創(chuàng)建的每個子類定義一個超類。Java 不支持多超類的繼承(這與C++ 不同,在C++中,你可以繼承多個基礎(chǔ)類)。你可以按照規(guī)定創(chuàng)建一個繼承的層次。該層次中,一個子類成為另一個子類的超類。然而,沒有類可以成為它自己的超類。
成員的訪問和繼承
盡管子類包括超類的所有成員,它不能訪問超類中被聲明成private 的成員。例如,考慮下面簡單的類層次結(jié)構(gòu):
/* In a class hierarchy, private members remain private to their class.
This program contains an error and will not compile.
*/
// Create a superclass.
class A {
int i;
private int j; // private to A
void setij(int x, int y) {
i = x; j = y;
}
}
// A"s j is not accessible here.
class B extends A {
int total; void sum() {
total = i + j; // ERROR, j is not accessible here
}
}
class Access {
public static void main(String args[]) {
B subOb = new B();
subOb.setij(10, 12);
subOb.sum();
System.out.println("Total is " + subOb.total);
}
}
該程序不會編譯,因為B中sum( ) 方法內(nèi)部對j的引用是不合法的。既然j被聲明成private,它只能被它自己類中的其他成員訪問。子類沒權(quán)訪問它。
注意:一個被定義成private 的類成員為此類私有,它不能被該類外的所有代碼訪問,包括子類。
更實際的例子
讓我們看一個更實際的例子,該例子有助于闡述繼承的作用。新的類將包含一個盒子的寬度、高度、深度。
// This program uses inheritance to extend Box.
class Box {
double width; double height; double depth;
// construct clone of an object
Box(Box ob) { // pass object to constructor
width = ob.width;
height = ob.height;
depth = ob.depth;
}
// constructor used when all dimensions specified
Box(double w, double h, double d) {
width = w;
height = h;
depth = d;
}
// constructor used when no dimensions specified
Box() {
width = -1; // use -1 to indicate
height = -1; // an uninitialized
depth = -1; // box
}
// constructor used when cube is created
Box(double len) {
width = height = depth = len;
}
// compute and return volume double
volume() {
return width * height * depth;
}
}
BoxWeight extends Box {
double weight; // weight of box
// constructor for BoxWeight
BoxWeight(double w, double h, double d, double m) {
width = w;
height = h;
depth = d;
weight = m;
}
}
class DemoBoxWeight {
public static void main(String args[]) {
BoxWeight mybox1 = new BoxWeight(10, 20, 15, 34.3);
BoxWeight mybox2 = new BoxWeight(2, 3, 4, 0.076);
double vol;
vol = mybox1.volume();
System.out.println("Volume of mybox1 is " + vol);
System.out.println("Weight of mybox1 is " + mybox1.weight);
System.out.println();
vol = mybox2.volume();
System.out.println("Volume of mybox2 is " + vol);
System.out.println("Weight of mybox2 is " + mybox2.weight);
}
}
該程序的輸出顯示如下:
Volume of mybox1 is 3000.0
Weight of mybox1 is 34.3
Volume of mybox2 is 24.0
Weight of mybox2 is 0.076
BoxWeight 繼承了Box 的所有特征并為自己增添了一個weight 成員。沒有必要讓BoxWeight 重新創(chuàng)建Box 中的所有特征。為滿足需要我們只要擴展Box就可以了。
繼承的一個主要優(yōu)勢在于一旦你已經(jīng)創(chuàng)建了一個超類,而該超類定義了適用于一組對象的屬性,它可用來創(chuàng)建任何數(shù)量的說明更多細(xì)節(jié)的子類。每一個子類能夠正好制作它自己的分類。例如,下面的類繼承了Box并增加了一個顏色屬性:
// Here, Box is extended to include color.
class ColorBox extends Box {
int color; // color of box
ColorBox(double w, double h, double d, int c) {
width = w;
height = h;
depth = d;
color = c;
}
}
記住,一旦你已經(jīng)創(chuàng)建了一個定義了對象一般屬性的超類,該超類可以被繼承以生成特殊用途的類。每一個子類只增添它自己獨特的屬性。這是繼承的本質(zhì)。
超類變量可以引用子類對象
超類的一個引用變量可以被任何從該超類派生的子類的引用賦值。你將發(fā)現(xiàn)繼承的這個方面在很多條件下是很有用的。例如,考慮下面的程序:
class RefDemo {
public static void main(String args[]) {
BoxWeight weightbox = new BoxWeight(3, 5, 7, 8.37);
Box plainbox = new Box(); double vol;
vol = weightbox.volume();
System.out.println("Volume of weightbox is " + vol);
System.out.println("Weight of weightbox is " + weightbox.weight);
System.out.println();
// assign BoxWeight reference to Box reference
plainbox = weightbox;
vol = plainbox.volume(); // OK, volume() defined in Box
System.out.println("Volume of plainbox is " + vol);
/* The following statement is invalid because plainbox does not define a weight member. */
// System.out.println("Weight of plainbox is " + plainbox.weight);
}
}
這里,weightbox 是BoxWeight 對象的一個引用,plainbox 是Box對象的一個引用。既然BoxWeight 是Box的一個子類,允許用一個weightbox 對象的引用給plainbox 賦值。
當(dāng)一個子類對象的引用被賦給一個超類引用變量時,你只能訪問超類定義的對象的那一部分。這是為什么plainbox 不能訪問weight 的原因,甚至是它引用了一個BoxWeight 對象也不行。仔細(xì)想一想,這是有道理的,因為超類不知道子類增加的屬性。這就是本程序中的最后一行被注釋掉的原因。Box的引用訪問weight 域是不可能的,因為它沒有定義。
代碼如下:
abstract?class?DongWu?{
public?abstract?void?info();
}
class?Bird?extends?DongWu?{
@Override
public?void?info()?{
System.out.println("我是一只鳥。");
}
}
class?Fish?extends?DongWu?{
@Override
public?void?info()?{
System.out.println("我是一條魚。");
}
}
public?class?App5?{
public?static?void?main(String[]?args)?{
DongWu?bird?=?new?Bird();
bird.info();
DongWu?fish?=?new?Fish();
fish.info();
}
}
我寫了一個,內(nèi)容比較簡單的。代碼如下:public class AnimalTest {
Animal animal;
public void eat(Animal animal){
animal.eat();
}
public void walk(Animal animal){
animal.walk();
}
public static void main(String args[]){
Animal animal=new Animal("animal");
Wolf w=new Wolf("wolf");
Goat g=new Goat("goat");
AnimalTest at=new AnimalTest();
at.eat(animal);
at.eat(w);
at.eat(g);
at.walk(animal);
at.walk(w);
at.walk(g);
}
}
class Animal {
String name;
public Animal(String name){
this.name=name;
}
public Animal(){}
public void setName(String name){
this.name=name;
}
public String getName(){
return name;
}
public void eat(){
System.out.println("animal eat");
}
public void walk(){
System.out.println("animal walk");
}
public String toString(){
return name;
}
}class Wolf extends Animal {
public Wolf(String name){
super(name);
}
public Wolf(){}
public void eat(){
System.out.println("wolf eat meat");
}
public void walk(){
System.out.println("wolf walk");
}
public String toString(){
return name;
}
}class Goat extends Animal {
public Goat(String name){
super(name);
}
public Goat(){}
public void eat(){
System.out.println("goat eat grass");
}
public void walk(){
System.out.println("goat walk");
}
public String toString(){
return name;
}
}