340 likes | 361 Vues
Learn the principles of inheritance, OOP pillars, and design guidelines for creating an effective inheritance hierarchy. Understand when and how to use inheritance in Java programming.
E N D
OO Design with Inheritance C Sc 335 Rick Mercer
Justification and Outline • I introduce inheritance different than most books that just show the mechanics and a silly example (like this from Sun) • I start with a reason to use inheritance • Show the objects found for a Library System • Recognize when to use of inheritance • Build an inheritance hierarchy • Design Guidelines related to inheritance • See another use of polymorphism
The 3 Pillars of OOP&D • Object-Oriented Programming • Encapsulation • Hide details in a class, provide methods • Polymorphism • Same name, different behavior, based on type • Inheritance • Capture common attributes and behaviors in a base class and extend it for different types
Object-Oriented Technology • OOT began with Simula 67 • developed in Norway • acronym for simulation language • Why this “new” language? • to build accurate models of complex working systems • The modularization occurs at the physical object level (not at a procedural level)
The Beginnings • Simula 67 was designed for system simulation (in Norway by Kristen Nygaard and Ole-Johan Dahl) • Caller and called subprogram had equal relationship • First notion of objects including class/instance distinctions • Ability to put code inside an instance to be executed • The class concept was first used here • Kristen Nygaard invented inheritance • Won the Turing award for 2002
One way to Start OOA and D • Identify candidate objects that model (shape) the system as a natural and sensible set of abstractions • Determine main responsibility of each • what an instance of the class must be able to do and what is should remember • This is part of Responsibility Driven Design ala Rebecca Wirfs-Brocks
System Specification The college library has requested a system that supports a small set of library operations. The librarian allows a student to borrow certain items, return those borrowed items, and pay fees. Late fees and due dates have been established at the following rates: Late feeLength of Borrow Books $0.50 per day 14 days Video tapes $5.00 plus 1.50 each additional day 2 days CDs $2.50 per day 7 days The due date is set when the borrowed item is checked out. A student with three (3) borrowed items, one late item, or late fees greater than $25.00 may not borrow anything new.
Identify candidate objects • Candidate objects that model a solution with main responsibility. The model (no GUIs, events, networking) • Librarian: Coordinates activities • Student, renamed Borrower • Book: Knows due date, late fees, Borrower, checkin... • Video: Knows due date, late fees, Borrower, checkin... • CD: Know due date, late fees, Borrower, checkin... • Three borrowed books: A collection of the things that can be borrowed, name it LendableList • BorrowList: maintains all possible borrowers
What do Books, Videos, and CDs have in common? • Common responsibilities (methods and data): • know call number, borrower, availability • can be borrowed • can be returned • Differences: • compute due date • compute late fee • may have additional state • Books have an author, CDs an artist
When is inheritance appropriate? • Object-Oriented Design Heuristic: If two or more classes have common data and behavior, then those classes should inherit from a common base class that captures those data and methods
An inheritance hierarchy The abstract class never instantiated Lendable is also known as the base class or superclass Lendable is shown to abstract (in italic) Book, CD, and Video are concrete subclasses
Why not have just one class? • Some of the behavior differs • Determine due date (2, 7, or 14 days) • Compute late fee not always daysLate * dayLateFee • Data differs • books have ISBNs and videos may have studio name • Inheritance • allows you to share implementations • allows one change in a common method to affect all • allows other Lendables to be added later more easily
Examples of inheritance in Java • You've seen HAS-A relationships: • A Song HAS-A most recent date played • Inheritance models IS-A relationships: • an oval IS-A shape • a rectangle IS-A shape • MyFrame extends JFrame makes MyFrame a JFrame with additional methods and listeners for a specific application
Java Examples of Inheritance • All Exceptions extend Exception: • RunTimeException IS-AN Exception • NullPointerException IS-A RunTimeException • Many classes extend the Component class • a JButton IS-A Component • a JTextField IS-A Component • A GregorianCalender IS-A Calendar • Vector and ArrayList extend AbstractList • UnplayableSongException IS-A RuntimeException
Designing An Inheritance Hierarchy • Start with an abstract class to define common-alities and differences (abstract methods) publicabstractclass Lendable { private instance variables constructor(s) methods that Lendable implements when the behavior is common to all of its subclasses (what is common) abstract methods the subclasses must implement that to represent what varies }
Some common data fields • Every class in the hierarchy ended with these private instance variables in class Lendable: • Subclasses can not directly reference these private String callNumber; private String title; private boolean availability; private String borrowerID; private DayTracker dueDate;
Lendable's constructor • Constructor needs a callNumber and title since it seems that all Lendables will need both • The actual values will later come from the subclasses constructor public Lendable(String callNumber, String initTitle) { callNumber = callNumber; // from subclass title = initTitle; // from subclass // Initialize others in a special way borrowerID = null; dueDate = null; availability = true; }
Common Behavior public String getCallNumber() { return callNumber; } public String getTitle() { return title; } public boolean isAvailable() { return availability; } public DayTracker getDueDate() { return dueDate; }
Common Behavior continued public int daysLate() { // return a positive if dueDate is before today DayTracker today = new DayTracker(); return dueDate.daysBetween(today); } public boolean isOverdue() { if(this.isAvailable()) return false; // not even checked out // Or check to see if this Lendable is overdue DayTracker today = new DayTracker(); // Return true if today is greater than // the due date for this Lendable return daysLate() > 0; }
Common Behavior continued // In superclass Lendable public boolean checkSelfIn() { if(this.isAvailable()) return false; else { // Adjust state so this is checked out dueDate = null; availability = true; return true; } }
checkSelfOut is splitbetween Lendable and its subclasses // this will be called from checkSelfOut in a subclass protected void checkOutAnyLendable(String borrowerID, int borrowLength) { // Record who is borrowing this Lendable thisborrowerID= borrowerID; // Set the due date dueDate = new DayTracker(); // today's date dueDate.adjustDaysBy(borrowLength); // Mark this Lendable as not available to borrow availability = false; } // This method is in class Video. You can call a // superclass method: prepend message with super. voidcheckSelfOut(String ID) { super.checkOutAnyLendable(DAYS_TO_BORROW_VIDEO); }
Protected • The protected access mode means that subclasses inherit this method • In Java, subclasses inherit all public & protected elements • The checkOutAnyLendablepu method is only known in Lendable and classes that extend Lendable • or classes that extend the class that extends Lendable
Design Issues • So far, we have good design because: • subclasses can't change the private variables of the superclass, even though the subclasses have them • protected methods are not visible outside of the inheritance hierarchy • doesn't require a bunch of setter methods in Lendable for subclasses to modify their own instance variables • the common behavior is in the superclass • the same thing is done for all subclasses • We'll get a polymorphic checkSelfOut message
Abstract methods • Subclasses differ: setDueDategetLateFee • declare the appropriate methods abstract, to force sub-classes to implement them in their own appropriate ways public abstract class Lendable { // Don't really borrow a Lendable // Nor do you or eat a Fruit ... // Subclass must implement these two methods abstractpublic void checkSelfOut(String ID); abstractpublic double getLateFee(); }// Done with Lendable for now
What can a subclass do? • General form for inheriting a Java class: public class subclass extends superclass { // subclass inherits all public and // protected methods of superclass may add class constants may add instance variables may add 1 to many constructors may add new public and protected methods may override methods in the superclass may add additional private methods }
The Constructor and Super • A subclass typically defines its constructor. If not • You will still get a default constructor with 0 parameters that automatically calls the base class constructor • Constructors in a subclass often call the superclass constructor to initialize the objects • Access superclass with the keyword super • can pass along arguments super(callNum, title) • if used, supermust be the first message in the constructor of the derived classes
Book extends Lendable public class Book extends Lendable { public static final int DAYS_TO_BORROW_BOOK = 14; public static final double BOOK_LATE_DAY_FEE = 0.50; private String author; public Book(String callNumber, String title, String author){ super(callNumber, title); author = author; } public String getAuthor() { return author; } }
Complete both required methods • checkSelfOut delegates some work to Lendable and passes along the unique information Book.DAYS_TO_BORROW_BOOK /** Modify the state of this object so it is borrowed * @param borrowerID: The identification of borrower */ public void checkSelfOut(String borrowerID) { checkOutAnyLendable(borrowerID,Book.DAYS_TO_BORROW_BOOK); }
getLateFee differs among the Lendable subclasses public double getLateFee() { if(this.isAvailable()) // Not even checked out! return 0.00; else { // A positive daysLate means due date has passed int daysOverdue = this.daysLate(); if(daysOverdue > 0) // This Lendable is overdue return daysOverdue * Book.BOOK_LATE_DAY_FEE; else return 0.00; // The due date has not passed } } }
A few assertions @Test public void testGetters() { // Show that Book has many methods via inheritance Book aBook = new Book("QA76.1", "C++", "Jo"); assertTrue(aBook.isAvailable()); assertFalse(aBook.isOverdue()); assertNull(aBook.getBorrowerID()); assertEquals("QA76.1", aBook.getCallNumber()); assertEquals("C++", aBook.getTitle()); assertEquals("Jo", aBook.getAuthor()); // Use the 2 methods that once were abstract and // now have concrete realizations assertEquals(0.00, aBook.getLateFee(), .001); assertTrue(aBook.isAvailable()); aBook.checkSelfOut("Rick"); assertFalse(aBook.isAvailable()); }
Adding Other Subclasses • Extend Lendable • Optional: add class constants • days to borrow, late fee amounts • Add a constructor that passes along arguments to the constructor in Lendable (super) • Add the methods that Lendable requires of all sublclasses: use Override • checkSelfOut • getLateFee