Monday, 23 January 2017

Chapter 24 Exercise 1, Introduction to Java Programming, Tenth Edition Y. Daniel LiangY.

24.1 (Add set operations in MyList ) Define the following methods in MyList and
implement them in MyAbstractList :
/** Adds the elements in otherList to this list.
* Returns true if this list changed as a result of the call */
public boolean addAll(MyList<E> otherList);
/** Removes all the elements in otherList from this list
* Returns true if this list changed as a result of the call */
public boolean removeAll(MyList<E> otherList);
/** Retains the elements in this list that are also in otherList
* Returns true if this list changed as a result of the call */
public boolean retainAll(MyList<E> otherList);
Write a test program that creates two MyArrayList s, list1 and list2 , with
the initial values {"Tom", "George", "Peter", "Jean", "Jane"} and
{"Tom", "George", "Michael", "Michelle", "Daniel"} , then per-
form the following operations:
■Invokes list1.addAll(list2) , and displays list1 and list2 .
■Recreates list1 and list2 with the same initial values, invokes
list1.removeAll(list2) , and displays list1 and list2 .
■Recreates list1 and list2 with the same initial values, invokes
list1.retainAll(list2) , and displays list1 and list2 .

public class Exercise01 {

 public static void main(String[] args) {
  MyArrayList<String> list1 = new MyArrayList<>(new String[] { "Tom",
    "George", "Peter", "Jean", "Jane" });
  MyArrayList<String> list2 = new MyArrayList<>(new String[] { "Tom",
    "George", "Michael", "Michelle", "Daniel" });
  System.out.println(list1 + "\n" + list2 + "\n");

  list1.addAll(list2);
  System.out.println("addAll\n" + list1 + "\n" + list2 + "\n");

  list1 = new MyArrayList<>(new String[] { "Tom", "George", "Peter",
    "Jean", "Jane" });
  list1.removeAll(list2);
  System.out.println("removeAll\n" + list1 + "\n" + list2 + "\n");

  list1 = new MyArrayList<>(new String[] { "Tom", "George", "Peter",
    "Jean", "Jane" });
  list1.retainAll(list2);
  System.out.println("retainAll\n" + list1 + "\n" + list2 + "\n");
 }

}

class MyArrayList<E> extends MyAbstractList<E> {
 public static final int INITIAL_CAPACITY = 16;
 @SuppressWarnings("unchecked")
 private E[] data = (E[]) new Object[INITIAL_CAPACITY];

 /** Create a default list */
 public MyArrayList() {
 }

 /** Create a list from an array of objects */
 public MyArrayList(E[] objects) {
  for (int i = 0; i < objects.length; i++)
   add(objects[i]); // Warning: don't use super(objects)!
 }

 @Override
 /** Add a new element at the specified index */
 public void add(int index, E e) {
  ensureCapacity();

  // Move the elements to the right after the specified index
  for (int i = size - 1; i >= index; i--)
   data[i + 1] = data[i];

  // Insert new element to data[index]
  data[index] = e;

  // Increase size by 1
  size++;
 }

 /** Create a new larger array, double the current size + 1 */
 private void ensureCapacity() {
  if (size >= data.length) {
   @SuppressWarnings("unchecked")
   E[] newData = (E[]) (new Object[size * 2 + 1]);
   System.arraycopy(data, 0, newData, 0, size);
   data = newData;
  }
 }

 @SuppressWarnings("unchecked")
 @Override
 /** Clear the list */
 public void clear() {
  data = (E[]) new Object[INITIAL_CAPACITY];
  size = 0;
 }

 @Override
 /** Return true if this list contains the element */
 public boolean contains(E e) {
  for (int i = 0; i < size; i++)
   if (e.equals(data[i]))
    return true;

  return false;
 }

 @Override
 /** Return the element at the specified index */
 public E get(int index) {
  checkIndex(index);
  return data[index];
 }

 private void checkIndex(int index) {
  if (index < 0 || index >= size)
   throw new IndexOutOfBoundsException("Index: " + index + ", Size: "
     + size);
 }

 @Override
 /** Return the index of the first matching element 
  *  in this list. Return -1 if no match. */
 public int indexOf(E e) {
  for (int i = 0; i < size; i++)
   if (e.equals(data[i]))
    return i;

  return -1;
 }

 @Override
 /** Return the index of the last matching element 
  *  in this list. Return -1 if no match. */
 public int lastIndexOf(E e) {
  for (int i = size - 1; i >= 0; i--)
   if (e.equals(data[i]))
    return i;

  return -1;
 }

 @Override
 /** Remove the element at the specified position 
  *  in this list. Shift any subsequent elements to the left.
  *  Return the element that was removed from the list. */
 public E remove(int index) {
  checkIndex(index);

  E e = data[index];

  // Shift data to the left
  for (int j = index; j < size - 1; j++)
   data[j] = data[j + 1];

  data[size - 1] = null; // This element is now null

  // Decrement size
  size--;

  return e;
 }

 @Override
 /** Replace the element at the specified position 
  *  in this list with the specified element. */
 public E set(int index, E e) {
  checkIndex(index);
  E old = data[index];
  data[index] = e;
  return old;
 }

 @Override
 public String toString() {
  StringBuilder result = new StringBuilder("[");

  for (int i = 0; i < size; i++) {
   result.append(data[i]);
   if (i < size - 1)
    result.append(", ");
  }

  return result.toString() + "]";
 }

 /** Trims the capacity to current size */
 public void trimToSize() {
  if (size != data.length) {
   @SuppressWarnings("unchecked")
   E[] newData = (E[]) (new Object[size]);
   System.arraycopy(data, 0, newData, 0, size);
   data = newData;
  } // If size == capacity, no need to trim
 }

 @Override
 /** Override iterator() defined in Iterable */
 public java.util.Iterator<E> iterator() {
  return new ArrayListIterator();
 }

 private class ArrayListIterator implements java.util.Iterator<E> {
  private int current = 0; // Current index

  @Override
  public boolean hasNext() {
   return (current < size);
  }

  @Override
  public E next() {
   return data[current++];
  }

  @Override
  public void remove() {
   MyArrayList.this.remove(current);
  }
 }
}

interface MyList<E> extends java.lang.Iterable<E> {
 /** Add a new element at the end of this list */
 public void add(E e);

 /** Add a new element at the specified index in this list */
 public void add(int index, E e);

 /** Clear the list */
 public void clear();

 /** Return true if this list contains the element */
 public boolean contains(E e);

 /** Return the element from this list at the specified index */
 public E get(int index);

 /**
  * Return the index of the first matching element in this list. Return -1 if
  * no match.
  */
 public int indexOf(E e);

 /** Return true if this list contains no elements */
 public boolean isEmpty();

 /**
  * Return the index of the last matching element in this list Return -1 if
  * no match.
  */
 public int lastIndexOf(E e);

 /**
  * Remove the first occurrence of the element o from this list. Shift any
  * subsequent elements to the left. Return true if the element is removed.
  */
 public boolean remove(E e);

 /**
  * Remove the element at the specified position in this list Shift any
  * subsequent elements to the left. Return the element that was removed from
  * the list.
  */
 public E remove(int index);

 /**
  * Replace the element at the specified position in this list with the
  * specified element and returns the new set.
  */
 public Object set(int index, E e);

 /** Return the number of elements in this list */
 public int size();

 /** Return an iterator for the list */
 public java.util.Iterator<E> iterator();
}

abstract class MyAbstractList<E> implements MyList<E> {
 protected int size = 0; // The size of the list

 /** Create a default list */
 protected MyAbstractList() {
 }

 /** Create a list from an array of objects */
 protected MyAbstractList(E[] objects) {
  for (int i = 0; i < objects.length; i++)
   add(objects[i]);
 }

 @Override
 /** Add a new element at the end of this list */
 public void add(E e) {
  add(size, e);
 }

 @Override
 /** Return true if this list contains no elements */
 public boolean isEmpty() {
  return size == 0;
 }

 @Override
 /** Return the number of elements in this list */
 public int size() {
  return size;
 }

 @Override
 /** Remove the first occurrence of the element e 
  *  from this list. Shift any subsequent elements to the left.
  *  Return true if the element is removed. */
 public boolean remove(E e) {
  if (indexOf(e) >= 0) {
   remove(indexOf(e));
   return true;
  } else
   return false;
 }

 /**
  * Adds the elements in otherList to this list. Returns true if this list
  * changed as a result of the call
  */
 public boolean addAll(MyList<E> otherList) {
  boolean result = false;
  for (E e : otherList) {
   add(e);
   if (!result) {
    result = true;
   }
  }
  return result;
 }

 /**
  * Removes all the elements in otherList from this list Returns true if this
  * list changed as a result of the call
  */
 public boolean removeAll(MyList<E> otherList) {
  boolean result = false;
  for (E e : otherList) {
   while (remove(e)) {
    if (!result) {
     result = true;
    }
   }
  }
  return result;
 }

 /**
  * Retains the elements in this list that are also in otherList Returns true
  * if this list changed as a result of the call
  */
 public boolean retainAll(MyList<E> otherList) {
  boolean result = false;
  for (int i = 0; i < size(); i++) {
   E e = get(i);
   if (otherList.indexOf(e) < 0) {
    remove(e);
    i--;
   }
  }
  return result;
 }
}

2 comments :

  1. In retainAll method ,I don't think its correct because there is no change made under another condition to the variable "result".The method will always return false.There are other errors as well in the method.

    Please check the solution again.
    P.S thank you so much for posting all these solutions

    ReplyDelete
  2. public boolean retainAll(MyList otherList) {
    boolean changed = false;
    for (int i = 0; i < this.size(); ) {
    if (!otherList.contains(this.get(i))) {
    this.remove(get(i));
    changed = true;
    }
    else
    i++;

    }

    return changed;
    }

    ReplyDelete