4. Luyện Tập về Thừa Kế Trong Java
4.1 Ex: The Circle and Cylinder Classes
Bài tập này sẽ hướng dẫn bạn thông qua các khái niệm quan trọng trong thừa kế.
Trong bài tập này, một lớp con được gọi Cylinderlà bắt nguồn từ siêu lớp Circlenhư được hiển thị trong sơ đồ lớp (trong đó một mũi tên hướng lên từ lớp con đến siêu lớp của nó). Nghiên cứu cách lớp con Cylindergọi các hàm tạo của lớp cha (thông qua super()và super(radius)) và kế thừa các biến và phương thức từ lớp cha Circle.
Bạn có thể sử dụng lại Circlelớp mà bạn đã tạo trong bài tập trước. Đảm bảo rằng bạn giữ " Circle.class" trong cùng thư mục.
public class Cylinder extends Circle {
private double height;
public Cylinder() {
super();
height = 1.0;
}
public Cylinder(double height) {
super();
this.height = height;
}
public Cylinder(double radius, double height) {
super(radius);
this.height = height;
}
public double getHeight() {
return height;
}
public double getVolume() {
return getArea()*height;
}
}
Viết chương trình kiểm tra (TestCylinder) để kiểm tra lớp Cylinder đã tạo, như sau:
public class TestCylinder {
public static void main (String[] args) {
Cylinder c1 = new Cylinder();
System.out.println("Cylinder:"
+ " radius=" + c1.getRadius()
+ " height=" + c1.getHeight()
+ " base area=" + c1.getArea()
+ " volume=" + c1.getVolume());
Cylinder c2 = new Cylinder(10.0);
System.out.println("Cylinder:"
+ " radius=" + c2.getRadius()
+ " height=" + c2.getHeight()
+ " base area=" + c2.getArea()
+ " volume=" + c2.getVolume());
Cylinder c3 = new Cylinder(2.0, 10.0);
System.out.println("Cylinder:"
+ " radius=" + c3.getRadius()
+ " height=" + c3.getHeight()
+ " base area=" + c3.getArea()
+ " volume=" + c3.getVolume());
}
}
Method Overriding and "Super": Lớp con lớp Cylinder kế thừa getArea()phương thức từ Vòng tròn siêu lớp của nó. Hãy thử cách ghi đè các getArea()phương pháp trong các lớp con Cylinderđể tính diện tích bề mặt (= 2π × bán kính × chiều cao + 2 × cơ sở khu vực) của xi lanh thay vì của vùng căn cứ. Đó là, nếu getArea()được gọi bởi một Circlethể hiện, nó trả về khu vực. Nếu getArea()được gọi bởi một Cylinderthể hiện, nó trả về diện tích bề mặt của hình trụ.
Nếu bạn ghi đè lên getArea()trong lớp con Cylinder, nó getVolume()không còn hoạt động. Điều này là do getVolume()sử dụng phương thức ghi đè được tìm thấy trong cùng một lớp. (Thời gian chạy Java sẽ chỉ tìm kiếm siêu lớp nếu nó không thể định vị phương thức trong lớp này). Sửa lỗi .getArea()getVolume()
Gợi ý: Sau khi ghi đè lên getArea()lớp con Cylinder, bạn có thể chọn gọi getArea()siêu lớp Circlebằng cách gọi super.getArea().
THỬ:
Cung cấp một toString()phương thức cho Cylinderlớp, ghi đè lên toString()kế thừa từ siêu lớp Circle, ví dụ:
@Override
public String toString() {
return "Cylinder: subclass of " + super.toString()
+ " height=" + height;
}
Hãy thử toString()phương pháp trong TestCylinder.
Lưu ý: @Overrideđược gọi là chú thích (được giới thiệu trong JDK 1.5), yêu cầu trình biên dịch kiểm tra xem có phương thức nào như vậy trong siêu lớp được ghi đè không. Điều này giúp rất nhiều nếu bạn viết sai tên của toString(). Nếu @Overridekhông được sử dụng và toString()bị sai chính tả ToString(), nó sẽ được coi là một phương thức mới trong lớp con, thay vì ghi đè lên lớp cha. Nếu @Overrideđược sử dụng, trình biên dịch sẽ báo hiệu lỗi. @Overridechú thích là tùy chọn, nhưng chắc chắn tốt đẹp để có.
4.2 Ví dụ: Superclass Person và các lớp con của nó
4.3 Ex: Point2D and Point3D
4.4 Ex: Point and MovablePoint
4.5 Ex: Superclass Shape and its subclasses Circle, Rectangle and Square
Viết một siêu lớp được gọi Shape(như thể hiện trong sơ đồ lớp), chứa:
- Hai biến đối tượng
color( String) và filled( boolean).
- Hai hàm tạo: một hàm tạo không có đối số (không đối số) khởi tạo
colorthành "xanh" và filledđến truevà một hàm tạo khởi tạo colorvà filledcho các giá trị đã cho.
- Getter và setter cho tất cả các biến thể hiện. Theo quy ước, getter cho một
booleanbiến xxxđược gọi isXXX()(thay vì getXxx()cho tất cả các loại khác).
- Một
toString()phương thức trả về " A Shape with color of xxx and filled/Not filled".
Viết chương trình kiểm tra để kiểm tra tất cả các phương thức được định nghĩa trong Shape.
Viết hai lớp con Shapeđược gọi Circlevà Rectangle, như thể hiện trong sơ đồ lớp.
Các Circlelớp học bao gồm:
- Một biến đối tượng
radius( double).
- Ba Constructor như hình. Trong constructor không có đối số thì khởi tạo bán kính tới
1.0.
- Getter và setter cho biến thể hiện (instance variable)
radius.
- Phương thức
getArea()và getPerimeter().
- Ghi đè
toString()phương thức được kế thừa, để trả về " A Circle with radius=xxx, which is a subclass of yyy", đây yyylà đầu ra của toString()phương thức từ lớp cha.
Các lớp Rectangle bao gồm:
- Hai biến đối tượng
width( double) và length( double).
- Ba Constructor như hình. Trong constructor không có đối số khởi tạo
width và length bằng 1.0.
- Getter và setter cho tất cả các biến thể hiện.
- Phương thức
getArea()và getPerimeter().
- Ghi đè
toString()phương thức được kế thừa, để trả về " A Rectangle with width=xxx and length=zzz, which is a subclass of yyy", đây yyy là đầu ra của toString()phương thức từ lớp cha.
Viết một lớp được gọi Square, như là một lớp con của Rectangle. Tự thuyết phục bản thân Square có thể được mô hình hóa như là một lớp con của Rectangle. Square không có biến đối tượng, nhưng kế thừa chiều rộng và chiều dài của biến đối tượng từ hình chữ nhật siêu lớp của nó.
- Cung cấp các constructor thích hợp (như thể hiện trong sơ đồ lớp). Dấu:
5. Exercises on Composition vs Inheritance
Chúng có hai cách để sử dụng lại một lớp trong các ứng dụng của bạn: thành phần và kế thừa .
5.1 Ex: The Point and Line Classes
Chúng ta hãy bắt đầu với thành phần với tuyên bố "một dòng gồm hai điểm".
Hoàn thành định nghĩa của hai lớp sau: Pointvà Line. Lớp Linebao gồm 2 trường hợp của lớp Point, đại diện cho điểm bắt đầu và điểm kết thúc của dòng. Cũng viết các lớp kiểm tra cho Pointvà Line(nói TestPointvà TestLine).
public class Point {
private int x;
private int y;
public Point (int x, int y) {......}
public String toString() {
return "Point: (" + x + "," + y + ")";
}
public int getX() {......}
public int getY() {......}
public void setX(int x) {......}
public void setY(int y) {......}
public void setXY(int x, int y) {......}
}
public class TestPoint {
public static void main(String[] args) {
Point p1 = new Point(10, 20);
System.out.println(p1);
......
}
}
public class Line {
private Point begin;
private Point end;
public Line (Point begin, Point end) {
this.begin = begin;
......
}
public Line (int beginX, int beginY, int endX, int endY) {
begin = new Point(beginX, beginY);
......
}
public String toString() { ...... }
public Point getBegin() { ...... }
public Point getEnd() { ...... }
public void setBegin(......) { ...... }
public void setEnd(......) { ...... }
public int getBeginX() { ...... }
public int getBeginY() { ...... }
public int getEndX() { ...... }
public int getEndY() { ...... }
public void setBeginX(......) { ...... }
public void setBeginY(......) { ...... }
public void setBeginXY(......) { ...... }
public void setEndX(......) { ...... }
public void setEndY(......) { ...... }
public void setEndXY(......) { ...... }
public int getLength() { ...... }
public double getGradient() { ...... }
}
public class TestLine {
public static void main(String[] args) {
Line l1 = new Line(0, 0, 3, 4);
System.out.println(l1);
Point p1 = new Point(...);
Point p2 = new Point(...);
Line l2 = new Line(p1, p2);
System.out.println(l2);
...
}
}
Sơ đồ lớp cho thành phần như sau (trong đó một mũi tên đầu kim cương rỗng chỉ vào thành phần của nó):
Thay vì thành phần , chúng ta có thể thiết kế một Linelớp bằng cách sử dụng inheritance. Thay vì "một dòng gồm hai điểm", chúng ta có thể nói rằng "một dòng là một điểm được mở rộng bởi một điểm khác", như thể hiện trong sơ đồ lớp sau:
Chúng ta hãy thiết kế lại Linelớp (được gọi LineSub) là một lớp con của lớp Point. LineSubkế thừa điểm bắt đầu từ siêu lớp của nó Pointvà thêm điểm kết thúc. Hoàn thành định nghĩa lớp. Viết một lớp kiểm tra được gọi TestLineSubđể kiểm tra LineSub.
public class LineSub extends Point {
Point end;
public LineSub (int beginX, int beginY, int endX, int endY) {
super(beginX, beginY);
this.end = new Point(endX, endY);
}
public LineSub (Point begin, Point end) {
super(begin.getX(), begin.getY());
this.end = end;
}
public String toString() { ... }
public Point getBegin() { ... }
public Point getEnd() { ... }
public void setBegin(...) { ... }
public void setEnd(...) { ... }
public int getBeginX() { ... }
public int getBeginY() { ... }
public int getEndX() { ... }
public int getEndY() { ... }
public void setBeginX(...) { ... }
public void setBeginY(...) { ... }
public void setBeginXY(...) { ... }
public void setEndX(...) { ... }
public void setEndY(...) { ... }
public void setEndXY(...) { ... }
public int getLength() { ... }
public double getGradient() { ... }
}
Tóm tắt: Có hai cách tiếp cận mà bạn có thể thiết kế một dòng, composition hoặc inheritance. "Một dòng gồm hai điểm" hoặc "Một dòng là một điểm được mở rộng bằng một điểm khác". So sánh Linevà LineSub thiết kế: Line sử dụng thành phần và LineSub sử dụng kế thừa . Thiết kế nào tốt hơn?
5.2 Ex: The Circle and Cylinder Classes Using Composition
Hãy thử viết lại Circle-Cylindercác bài tập trước sử dụng thành phần (như thể hiện trong sơ đồ lớp) thay vì thừa kế . Đó là, "một hình trụ bao gồm một vòng tròn cơ sở và chiều cao".
public class Cylinder {
private Circle base;
private double height;
public Cylinder() {
base = new Circle();
height = 1.0;
}
......
}
Thiết kế nào (kế thừa hoặc thành phần) là tốt hơn?
bai 4.4 va 4.5 hoi kho co oi
ردحذفbai 4.4 phan move() no nhu nao co nhi
ردحذفإرسال تعليق