Pages

BibCity2

public class BibCity2
{
    public static void main(String[] args)
    {
        Publisher addisonWesley =
            PublishersTradeAssociation.newPublisher("Addison Wesley");

        Author gbt = new Author("George B. Thomas");
        Book calculus_e1 = gbt.newNonfiction(
            addisonWesley, "Calculus and Analytic Geometry, 1st ed.");
        
        AuthorCollector dek = new AuthorCollector("Donald E. Knuth");
        dek.library().add(calculus_e1.copy());
        dek.signBookDeal(addisonWesley);
        Book taocp_v1 = dek.newNonfiction(
            "The Art of Computer Programming, vol. 1 (Fundamental Algorithms)");
        dek.library().add(taocp_v1.copy());
        Book surreal = dek.newFiction("Surreal Numbers");
        dek.library().add(surreal.copy());
        
        Author[] collaborator = { new Author("Ross L. Finney") };
        gbt.collaborateWith(collaborator);
        Book calculus_e8 = gbt.newNonfiction(
            addisonWesley, "Calculus and Analytic Geometry, 8th ed.");
        
        Collector jps = new Collector("Mr. Spurgeon");
        jps.library().add(calculus_e8.copy());
        jps.library().add(taocp_v1.copy());
        
        System.out.println("DEK's Collection:");
        System.out.println(dek.library());
        System.out.println("JPS's Collection:");
        System.out.println(jps.library());
    }
}
class Publisher
{
    private final String name;
    private final Integer id;
    private Integer currentBookNumber = 0;
    
    public Publisher(String name, int id)
    {
        this.name = name;
        this.id = id;
    }
    public String name() { return name; }
    public Integer id()   { return id; }
    public String nextIsbn()
    {
        currentBookNumber++;
        return id.toString() + "-" + currentBookNumber.toString();
    }
}

class PublishersTradeAssociation
{
    private static int currentPublisherNumber = 0;
    
    public static Publisher newPublisher(String name)
    {
        currentPublisherNumber++;
        return new Publisher(name, currentPublisherNumber);
    }
}
class Library
{
    private Collection<PrintCopy> collection = new CircularList<PrintCopy>();
    
    public boolean hasBook(Book book)
    {
        Iterator<PrintCopy> i = collection.iterator();
        while (i.hasNext())
        {
            PrintCopy copy = i.next();
            if (copy.isbn().equals(book.isbn()))
                return true;
        }
        return false;
    }
    public void add(PrintCopy copy)
    {
        collection.add(copy);
    }
    public String toString()
    {
        String s = "";
        Iterator<PrintCopy> i = collection.iterator();
        while (i.hasNext())
        {
            PrintCopy copy = i.next();
            s += copy.toString() + "\n";
        }
        return s;
    }
}
class Author
{
    private String name;
    private Author[] collaborators = {};
    private Publisher myPublisher = null;
    private List<Book> myBooks = new CircularList<Book>();
    
    public Author(String name)                     { this.name = name; }
    public void collaborateWith(Author[] others)   { collaborators = others; }
    public void stopCollaborating()                { collaborators = new Author[0]; }
    public List<Book> myBooks()                    { return myBooks; }
    public void signBookDeal(Publisher publisher)  { this.myPublisher = publisher; }
    public Book newFiction(String title)
    {
        return this.newFiction(this.myPublisher, title);
    }
    public Book newNonfiction(String title)
    {
        return this.newNonfiction(this.myPublisher, title);
    }
    public Book newFiction(Publisher publisher, String title)
    {
        Book newBook = collaborators.length == 0 ?
            new FictionBook(publisher, title, this) :
            new CollaborativeFictionBook(publisher, title, this, collaborators);
        for (Author collaborator : collaborators)
            collaborator.myBooks().add(newBook);
        this.myBooks.add(newBook);
        return newBook;
    }
    public Book newNonfiction(Publisher publisher, String title)
    {
        Book newBook = collaborators.length == 0 ?
            new NonFictionBook(publisher, title, this) :
            new NonFictionBook(publisher, title, this, collaborators);
        for (Author collaborator : collaborators)
            collaborator.myBooks().add(newBook);
        this.myBooks.add(newBook);
        return newBook;
    }
}
interface Bibliophile
{
    public abstract Library library();
}

class AuthorCollector extends Author implements Bibliophile
{
    private Library myLibrary = new Library();
    
    public AuthorCollector(String name) { super(name); }
    public Library library()            { return myLibrary; }
}

class Collector implements Bibliophile
{
    private String name;
    private Library myLibrary = new Library();
    
    public Collector(String name) { this.name = name; }
    public Library library()      { return myLibrary; }
    
}
interface BookInfo
{
    public abstract String isbn();
    public abstract Publisher publisher();
    public abstract String title();
    public abstract Author author();
    public abstract Author author(int i);
    public abstract int authorCount();
    public abstract String category();
}

class PrintCopy implements BookInfo
{
    private final Book book;
    private final int copyNumber;
    
    public PrintCopy(Book book)
    {
        this.book = book;
        this.copyNumber = book.copiesPrinted();
    }
    public String isbn()                  { return this.book.isbn(); }
    public String title()                 { return this.book.title(); }
    public Author author()                { return this.book.author(); }
    public Author author(int i)           { return this.book.author(i); }
    public int authorCount()              { return this.book.authorCount(); }
    public String category()              { return this.book.category(); }
    public Publisher publisher()          { return this.book.publisher(); }
    public boolean equals(PrintCopy copy) { return this.isbn().equals(copy.isbn()); }
    public String toString()
    {
        return "copy #" + copyNumber + ": " + this.book.toString();
    }
}
abstract class Book implements BookInfo
{
    private int copiesPrinted = 0;
    private final String isbn;
    private final String title;
    private final Author[] authors;
    private final Publisher publisher;
    
    protected Book(Publisher publisher, String title, Author author)
    {
        this.publisher = publisher;
        this.isbn = publisher.nextIsbn();
        this.title = title;
        this.authors = new Author[1];
        authors[0] = author;
    }
    protected Book(Publisher publisher, String title, Author author, Author[] others)
    {
        this.publisher = publisher;
        this.isbn = publisher.nextIsbn();
        this.title = title;
        int n = others.length;
        this.authors = new Author[n + 1];
        this.authors[0] = author;
        for (int i = 0; i < n; i++)
            this.authors[i + 1] = others[i];
    }
    public PrintCopy copy()
    {
        this.copiesPrinted++;
        return new PrintCopy(this);
    }
    public String isbn()               { return this.isbn; }
    public String title()              { return this.title; }
    public int copiesPrinted()         { return this.copiesPrinted; }
    public Author author()             { return this.authors[0]; }
    public Author author(int i)        { return this.authors[i]; }
    public int authorCount()           { return this.authors.length; }
    public Publisher publisher()       { return this.publisher; }
    public String toString()           { return this.title; }
    public abstract String category();
}
class FictionBook extends Book
{
    public FictionBook(Publisher publisher, String title, Author author)
    {
        super(publisher, title, author);
    }
    public String category() { return "fiction"; }
}

class CollaborativeFictionBook extends Book
{
    public CollaborativeFictionBook(
        Publisher publisher, String title, Author author, Author[] others)
    {
        super(publisher, title, author, others);
    }
    public String category() { return "collaborative fiction"; }
}

class NonFictionBook extends Book
{
    public NonFictionBook(
        Publisher publisher, String title, Author author, Author[] others)
    {
        super(publisher, title, author, others);
    }
    public NonFictionBook(Publisher publisher, String title, Author author)
    {
        super(publisher, title, author);
    }
    public String category() { return "nonfiction"; }
}
interface Iterator<T>
{
    public abstract T next();
    public abstract boolean hasNext();
}

interface Iterable<T>
{
    public abstract Iterator<T> iterator();
}

interface Collection<T> extends Iterable<T>
{
    public abstract boolean contains(T object);
    public abstract boolean add(T object);
}

interface List<T> extends Collection<T>
{
}

class ListIterator<T> implements Iterator<T>
{
    private ListNode<T> current;
    private final ListNode<T> stop;
    
    public ListIterator(ListNode<T> start, ListNode<T> stop)
    {
        this.current = start;
        this.stop = stop;
    }
    public T next()
    {
        current = this.current.getNext();
        return this.current.getValue();
    }
    public boolean hasNext()
    {
        return this.current.getNext() != this.stop;
    }
}
class CircularList<T> implements List<T>
{
    private ListNode<T> head, tail;
    
    public CircularList()
    {
        head = new ListNode<T>();
        tail = head.setNext(head);
    }
    public ListIterator<T> iterator()
    {
        return new ListIterator<T>(head, head);
    }
    public boolean contains(T object)
    {
        ListIterator<T> i = this.iterator();
        while (i.hasNext())
        {
            if (i.next().equals(object))
                return true;
        }
        return false;
    }
    public boolean add(T object)
    {
        tail.setNext(new ListNode<T>(object, tail.getNext()));            
        return true;
    }
}
class ListNode<T>
{
    private T value;
    private ListNode<T> nextNode;
    
    public ListNode()
    {
        this(null, null);
    }
    public ListNode(T value)
    {
        this(value, null);
    }
    public ListNode(T value, ListNode<T> node)
    {
        this.value = value;
        this.nextNode = node;
    }
    public ListNode<T> getNext()
    {
        return nextNode;
    }
    public ListNode<T> setNext(ListNode<T> node)
    {
        return this.nextNode = node;
    }
    public T getValue()
    {
        return this.value;
    }
}