classpath
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

PATCH: java.util.Vector


From: Bryce McKinlay
Subject: PATCH: java.util.Vector
Date: Thu, 23 Nov 2000 00:52:52 +1300

Classpath's Vector implementation was not synchronized. It should be. It
also had a bug where indexOf() and lastIndexOf() would not handle null
entries correctly.

This patch implements all the methods specified in the JDK 1.3 docs.
Many of these are simply synchronized wrappers around the default
implementations in AbstractList/AbstractCollection. However, where a
collections method is also implemented in ArrayList, I've implemented it
likewise in Vector.

There are also a couple of minor formatting fixes here.

regards

  [ bryce ]

P.S. Whats with the mangled classpath ChangeLog? Do we really want to be
generating this from the cvs logs? I find manual changelogs to be more
useful, and anyone can generate a CVS log if they want one.


2000-11-23  Bryce McKinlay  <address@hidden>

        * java/util/ArrayList.java (addAll(Collection)): Call 
        addAll(int,Collection) instead of duplicating code.
        (indexOf): Clean up int initialization.
        (clear): Set cleared array entries to null, to allow garbage 
        collection.
        * java/util/List.java: Minor formatting fixes.
        * java/util/Random.java: ditto.
        * java/util/SimpleTimeZone.java: ditto.
        * java/util/Vector.java: Make many methods synchronized. Increment 
        modCount unconditionally in line with the JCL spec.
        (Vector): Simplify constructors. Use for loop in Collection copy
        constructor for efficiency.
        (indexOf): Handle `null' elements.
        (lastIndexOf): Ditto.
        (toArray): Replace with simpler implementation from ArrayList.
        (add): Implemented.
        (remove(int)): Add bounds check.
        (containsAll): Implemented.
        (addAll): Implemented.
        (removeAll): Implemented.
        (retainAll): Implemented.
        (equals): Implemented.
        (hashCode): Implemented.
        (toString): Use string concatenation, not StringBuffer.
        (subList): Implemented.
        (removeRange): Implemented.

Index: ArrayList.java
===================================================================
RCS file: /cvs/classpath/java/util/ArrayList.java,v
retrieving revision 1.9
diff -u -r1.9 ArrayList.java
--- ArrayList.java      2000/11/02 10:12:57     1.9
+++ ArrayList.java      2000/11/22 11:29:07
@@ -43,7 +43,7 @@
  * to or removing from the end of a list, checking the size, &c.
  *
  * @author        Jon A. Zeppieri
- * @version       $Id: ArrayList.java,v 1.9 2000/11/02 10:12:57 bryce Exp $
+ * @version       $Id: ArrayList.java,v 1.3 2000/11/02 10:08:03 bryce Exp $
  * @see           java.util.AbstractList
  * @see           java.util.List
  */
@@ -219,15 +219,7 @@
    */
   public boolean addAll(Collection c)
   {
-    modCount++;
-    Iterator itr = c.iterator();
-    int csize = c.size();
-    ensureCapacity(size + csize);
-    for (int pos = 0; pos < csize; pos++)
-      {
-       data[size++] = itr.next();
-      }
-    return (csize > 0);
+    return addAll(size, c);
   }
 
   /** 
@@ -295,9 +287,7 @@
    */
   public int indexOf(Object e)
   {
-    int i;
-
-    for (i = 0; i < size; i++)
+    for (int i = 0; i < size; i++)
       {
        if (e == null ? data[i] == null : e.equals(data[i]))
          return i;
@@ -330,6 +320,10 @@
   public void clear()
   {
     modCount++;
+    for (int i = 0; i < size; i++)
+      {
+       data[i] = null;
+      }    
     size = 0;
   }
 
@@ -364,8 +358,8 @@
   }
 
   /**
-   * Returns an Array whse component type is the runtime component type of
-   * the passes-in Array.  The returned Array is populated with all of the
+   * Returns an Array whose component type is the runtime component type of
+   * the passed-in Array.  The returned Array is populated with all of the
    * elements in this ArrayList.  If the passed-in Array is not large enough
    * to store all of the elements in this List, a new Array will be created 
    * and returned; if the passed-in Array is <i>larger</i> than the size
Index: List.java
===================================================================
RCS file: /cvs/classpath/java/util/List.java,v
retrieving revision 1.5
diff -u -r1.5 List.java
--- List.java   2000/10/26 10:19:01     1.5
+++ List.java   2000/11/22 11:29:07
@@ -311,7 +311,7 @@
    * @returns an array of type Object[] and length equal to the length of this
    *   list, containing the elements currently in this list, in order.
    */
-  Object[]toArray();
+  Object[] toArray();
 
   /**
    * Copy the current contents of this list into an array. If the array passed
@@ -329,5 +329,5 @@
    * @exception ArrayStoreException if the type of any element of the
    *   collection is not a subtype of the element type of a.
    */
-  Object[]toArray(Object[]a);
+  Object[] toArray(Object[] a);
 }
Index: Random.java
===================================================================
RCS file: /cvs/classpath/java/util/Random.java,v
retrieving revision 1.6
diff -u -r1.6 Random.java
--- Random.java 2000/10/26 10:19:01     1.6
+++ Random.java 2000/11/22 11:29:07
@@ -166,7 +166,7 @@
    * @param bytes The byte array that should be filled.
    * @since JDK1.1
    */
-  public void nextBytes(byte[]bytes)
+  public void nextBytes(byte[] bytes)
     /*{ require { bytes != null :: "bytes is null"; } } */
   {
     int random;
Index: SimpleTimeZone.java
===================================================================
RCS file: /cvs/classpath/java/util/SimpleTimeZone.java,v
retrieving revision 1.10
diff -u -r1.10 SimpleTimeZone.java
--- SimpleTimeZone.java 2000/10/27 09:53:52     1.10
+++ SimpleTimeZone.java 2000/11/22 11:29:08
@@ -763,7 +763,7 @@
     else
       {
        int length = input.readInt();
-       byte[]byteArray = new byte[length];
+       byte[] byteArray = new byte[length];
        input.read(byteArray, 0, length);
        if (length >= 4)
          {
Index: Vector.java
===================================================================
RCS file: /cvs/classpath/java/util/Vector.java,v
retrieving revision 1.10
diff -u -r1.10 Vector.java
--- Vector.java 2000/10/26 10:19:01     1.10
+++ Vector.java 2000/11/22 11:29:08
@@ -27,6 +27,7 @@
 
 package java.util;
 import java.lang.reflect.Array;
+import java.io.Serializable;
 
 /**
  * the <b>Vector</b> classes implements growable arrays of Objects.
@@ -41,11 +42,16 @@
  *
  * Vector implements the JDK 1.2 List interface, and is therefor a fully
  * compliant Collection object.
+ *
+ * @specnote The JCL claims that various methods in this class throw
+ * IndexOutOfBoundsException, which would be consistent with other collections
+ * classes. ArrayIndexOutOfBoundsException is actually thrown, per the online 
+ * docs, even for List method implementations.
  * 
  * @author Scott G. Miller
  */
-public class Vector extends AbstractList implements List,
-  Cloneable, java.io.Serializable
+public class Vector extends AbstractList 
+  implements List, Cloneable, Serializable
 {
   /**
    * The amount the Vector's internal array should be increased in size when
@@ -53,14 +59,14 @@
    * or when address@hidden #ensureCapacity} is called.
    * @serial
    */
-  protected int capacityIncrement;
+  protected int capacityIncrement = 0;
 
   /**
    * The number of elements currently in the vector, also returned by
    * address@hidden #size}.
    * @serial
    */
-  protected int elementCount;
+  protected int elementCount = 0;
 
   /**
    * The internal array used to hold members of a Vector
@@ -76,7 +82,7 @@
    */
   public Vector()
   {
-    this(10, 0);
+    this(10);
   }
 
   /**
@@ -88,13 +94,13 @@
    */
   public Vector(Collection c)
   {
-    this(c.size(), 0);
-    elementCount = elementData.length;
-    Iterator collectionIterator = c.iterator();
-    int counter = 0;
-    while (collectionIterator.hasNext())
+    int csize = c.size();
+    elementData = new Object[csize];
+    elementCount = csize;
+    Iterator itr = c.iterator();
+    for (int i = 0; i < csize; i++)
       {
-       elementData[counter++] = collectionIterator.next();
+       elementData[i] = itr.next();
       }
   }
 
@@ -109,10 +115,8 @@
    */
   public Vector(int initialCapacity, int capacityIncrement)
   {
-    modCount = 0;
     elementData = new Object[initialCapacity];
     this.capacityIncrement = capacityIncrement;
-    elementCount = 0;
   }
 
   /**
@@ -122,21 +126,21 @@
    */
   public Vector(int initialCapacity)
   {
-    this(initialCapacity, 0);
+    elementData = new Object[initialCapacity];
   }
 
   /**
    * Copies the contents of a provided array into the Vector.  If the 
-   * array is too large to fit in the Vector, an IndexOutOfBoundsException
+   * array is too large to fit in the Vector, an ArrayIndexOutOfBoundsException
    * is thrown.  Old elements in the Vector are overwritten by the new
    * elements
    *
    * @param anArray An array from which elements will be copied into the Vector
    * 
-   * @throws IndexOutOfBoundsException the array being copied
+   * @throws ArrayIndexOutOfBoundsException the array being copied
    * is larger than the Vectors internal data array
    */
-  public void copyInto(Object[]anArray)
+  public synchronized void copyInto(Object[] anArray)
   {
     System.arraycopy(elementData, 0, anArray, 0, elementCount);
   }
@@ -146,7 +150,7 @@
    * than the number of Objects its holding, a new array is constructed
    * that precisely holds the elements.  
    */
-  public void trimToSize()
+  public synchronized void trimToSize()
   {
     // Check if the Vector is already trimmed, to save execution time
     if (elementCount == elementData.length)
@@ -169,16 +173,19 @@
    * @param minCapacity The minimum capacity the internal array should be
    * able to handle after executing this method
    */
-  public void ensureCapacity(int minCapacity)
+  public synchronized void ensureCapacity(int minCapacity)
   {
+    modCount++;
     if (elementData.length >= minCapacity)
       return;
 
-    int newCapacity = elementData.length + capacityIncrement;
+    int newCapacity; 
     if (capacityIncrement <= 0)
       newCapacity = elementData.length * 2;
-
-    Object[]newArray = new Object[Math.max(newCapacity, minCapacity)];
+    else
+      newCapacity = elementData.length + capacityIncrement;
+      
+    Object[] newArray = new Object[Math.max(newCapacity, minCapacity)];
 
     System.arraycopy(elementData, 0, newArray, 0, elementData.length);
     elementData = newArray;
@@ -187,20 +194,18 @@
   /**
    * Explicitly sets the size of the internal data array, copying the 
    * old values to the new internal array.  If the new array is smaller
-   * than the old one, old values that don't fit are lost.
+   * than the old one, old values that don't fit are lost. If the new size
+   * is larger than the old one, the vector is padded with null entries.
    *
    * @param newSize The new size of the internal array
    */
-  public void setSize(int newSize)
+  public synchronized void setSize(int newSize)
   {
-    if (newSize < elementCount)
-      {
-       modCount++;
-       elementCount = newSize;
-      }
+    modCount++;
     Object[] newArray = new Object[newSize];
-    System.arraycopy(elementData, 0, newArray, 0,
-                    Math.min(newSize, elementData.length));
+    System.arraycopy(elementData, 0, newArray, 0, 
+                     Math.min(newSize, elementCount));
+    elementCount = newSize;
     elementData = newArray;
   }
 
@@ -240,16 +245,16 @@
    * and returns the index of the first occurence of this Object.  If
    * the object is not found, -1 is returned
    *
-   * @param elem The Object to search for
+   * @param e The Object to search for
    * @param index Start searching at this index
    * @returns The index of the first occurence of <b>elem</b>, or -1
    * if it is not found
    */
-  public int indexOf(Object elem, int index)
+  public synchronized int indexOf(Object e, int index)
   {
     for (int i = index; i < elementCount; i++)
       {
-       if (elementData[i].equals(elem))
+       if (e == null ? elementData[i] == null : e.equals(elementData[i]))
          return i;
       }
     return -1;
@@ -284,17 +289,18 @@
    * backwards from <b>index</b>.  If the object does not occur in this Vector,
    * -1 is returned.
    *
-   * @param elem The object to search for
+   * @param eThe object to search for
    * @param index The index to start searching in reverse from
    * @returns The index of the Object if found, -1 otherwise
    */
-  public int lastIndexOf(Object elem, int index)
+  public synchronized int lastIndexOf(Object e, int index)
   {
-    if (index > elementCount - 1)
+    if (index >= elementCount)
       throw new ArrayIndexOutOfBoundsException(index);
+
     for (int i = index; i >= 0; i--)
       {
-       if ((elementData[i] == elem) || elementData[i].equals(elem))
+       if (e == null ? elementData[i] == null : e.equals(elementData[i]))
          return i;
       }
     return -1;
@@ -321,7 +327,7 @@
    * @throws ArrayIndexOutOfBoundsException <b>index</b> is
    * larger than the Vector
    */
-  public Object elementAt(int index)
+  public synchronized Object elementAt(int index)
   {
     //Within the bounds of this Vector does not necessarily mean within 
     //the bounds of the internal array
@@ -338,9 +344,9 @@
    * @returns The first Object in the Vector
    * @throws NoSuchElementException the Vector is empty
    */
-  public Object firstElement()
+  public synchronized Object firstElement()
   {
-    if (isEmpty())
+    if (elementCount == 0)
       throw new NoSuchElementException();
 
     return elementAt(0);
@@ -353,9 +359,9 @@
    * @returns The last Object in the Vector
    * @throws NoSuchElementException the Vector is empty
    */
-  public Object lastElement()
+  public synchronized Object lastElement()
   {
-    if (isEmpty())
+    if (elementCount == 0)
       throw new NoSuchElementException();
 
     return elementAt(elementCount - 1);
@@ -370,7 +376,7 @@
    * @param index The position in the Vector to store the object
    * @throws ArrayIndexOutOfBoundsException the index is out of range
    */
-  public void setElementAt(Object obj, int index)
+  public synchronized void setElementAt(Object obj, int index)
   {
     if ((index < 0) || (index >= elementCount))
       throw new ArrayIndexOutOfBoundsException(index);
@@ -386,8 +392,9 @@
    * @param element The Object to store in the Vector
    * @returns The previous object at the specified index
    * @throws ArrayIndexOutOfBoundsException the index is out of range
+   *
    */
-  public Object set(int index, Object element)
+  public synchronized Object set(int index, Object element)
   {
     if (index >= elementCount)
       throw new ArrayIndexOutOfBoundsException(index);
@@ -403,7 +410,7 @@
    *
    * @param index The index of the element to remove
    */
-  public void removeElementAt(int index)
+  public synchronized void removeElementAt(int index)
   {
     if (index >= elementCount)
       throw new ArrayIndexOutOfBoundsException(index);
@@ -445,7 +452,7 @@
    *
    * @param obj The object to add to the Vector
    */
-  public void addElement(Object obj)
+  public synchronized void addElement(Object obj)
   {
     ensureCapacity(elementCount + 1);
     modCount++;
@@ -460,32 +467,28 @@
    * @param obj The object to remove from the Vector
    * @returns true if the Object was in the Vector, false otherwise
    */
-  public boolean removeElement(Object obj)
+  public synchronized boolean removeElement(Object obj)
   {
-    int ix = indexOf(obj);
-    if (ix != -1)
+    int idx = indexOf(obj);
+    if (idx != -1)
       {
-       removeElementAt(ix);
+       removeElementAt(idx);
        return true;
       }
-    else
-      {
-       return false;
-      }
+    return false;
   }
 
   /**
    * Removes all elements from the Vector.  Note that this does not
    * resize the internal data array.
    */
-  public void removeAllElements()
+  public synchronized void removeAllElements()
   {
-    if (size() != 0)
-      modCount++;
-    else
+    modCount++;
+    if (elementCount == 0)
       return;
 
-    for (int i = 0; i < elementData.length; i++)
+    for (int i = 0; i < elementCount; i++)
       {
        elementData[i] = null;
       }
@@ -495,12 +498,12 @@
   /**
    * Creates a new Vector with the same contents as this one.
    */
-  public Object clone()
+  public synchronized Object clone()
   {
     try
       {
        Vector clone = (Vector) super.clone();
-       clone.elementData = (Object[])elementData.clone();
+       clone.elementData = (Object[]) elementData.clone();
        return clone;
       }
     catch (CloneNotSupportedException ex)
@@ -519,7 +522,7 @@
    * @returns An Object[] containing the contents of this Vector in order
    *
    */
-  public Object[] toArray()
+  public synchronized Object[] toArray()
   {
     Object[] newArray = new Object[elementCount];
     copyInto(newArray);
@@ -540,32 +543,20 @@
    * array.
    *
    *
-   * @param a An array to copy the Vector into if large enough
+   * @param array An array to copy the Vector into if large enough
    * @returns An array with the contents of this Vector in order
    * @throws ArrayStoreException the runtime type of the provided array
    * cannot hold the elements of the Vector
    */
-  public Object[] toArray(Object[]a)
+  public synchronized Object[] toArray(Object[] array)
   {
-    if (a.length >= elementCount)
-      {
-       copyInto(a);
-       if (a.length > elementCount)
-         {
-           a[elementCount] = null;
-         }
-       return a;
-      }
+    if (array.length < elementCount)
+      array = (Object[]) 
Array.newInstance(array.getClass().getComponentType(), 
+                                          elementCount);
     else
-      {
-       //This seems like a kludge.. Is there a better way to find the
-       //Runtime type of an array?
-       Object[]newArray =
-         (Object[])Array.newInstance(a.getClass().getComponentType(),
-                                     elementCount);
-       copyInto(newArray);
-       return newArray;
-      }
+      array[elementCount] = null;
+    System.arraycopy(elementData, 0, array, 0, elementCount);
+    return array;
   }
 
   /**
@@ -575,7 +566,7 @@
    * @throws ArrayIndexOutOfBoundsException the index is not within the 
    * range of the Vector
    */
-  public Object get(int index)
+  public synchronized Object get(int index)
   {
     return elementAt(index);
   }
@@ -593,6 +584,17 @@
   }
 
   /**
+   * Adds an object to the Vector.
+   *
+   * @param o The element to add to the Vector
+   */
+  public synchronized boolean add(Object o)
+  {
+    addElement(o);
+    return true;
+  }
+
+  /**
    * Adds an object at the specified index.  Elements at or above
    * index are shifted up one position.
    *
@@ -612,9 +614,11 @@
    * @throws ArrayIndexOutOfBoundsException the index was out of the range
    * of the Vector
    */
-  public Object remove(int index)
+  public synchronized Object remove(int index)
   {
-    modCount++;
+    if (index >= elementCount)
+      throw new ArrayIndexOutOfBoundsException(index);
+  
     Object temp = elementData[index];
     removeElementAt(index);
     return temp;
@@ -628,22 +632,80 @@
     removeAllElements();
   }
 
+  public synchronized boolean containsAll(Collection c)
+  {
+    Iterator itr = c.iterator();
+    int size = c.size();
+    for (int pos = 0; pos < size; pos++)
+      {
+       if (!contains(itr.next()))
+         return false;
+      }
+    return true;
+  }
+
+  public synchronized boolean addAll(Collection c)
+  {
+    return addAll(elementCount, c);
+  }
+  
+  public synchronized boolean removeAll(Collection c)
+  {
+    return super.removeAll(c);
+  }
+  
+  public synchronized boolean retainAll(Collection c)
+  {
+    return super.retainAll(c);
+  }
+
+  public synchronized boolean addAll(int index, Collection c)
+  {
+    if (index < 0 || index > elementCount)
+      throw new ArrayIndexOutOfBoundsException(index);
+    modCount++;
+    Iterator itr = c.iterator();
+    int csize = c.size();
+
+    ensureCapacity(elementCount + csize);
+    int end = index + csize;
+    if (elementCount > 0 && index != elementCount)
+      System.arraycopy(elementData, index, elementData, end, csize);
+    elementCount += csize;
+    for (; index < end; index++)
+      {
+        elementData[index] = itr.next();
+      }
+    return (csize > 0);  
+  }
+
+  public synchronized boolean equals(Object c)
+  {
+    return super.equals(c);
+  }
+
+  public synchronized int hashCode()
+  {
+    return super.hashCode();
+  }
+
   /**
    * Returns a string representation of this Vector in the form 
    * [element0, element1, ... elementN]
    *
    * @returns the String representation of this Vector
    */
-  public String toString()
+  public synchronized String toString()
   {
-    StringBuffer toReturn = new StringBuffer("[");
-    String comma = "";
+    String r = "[";
     for (int i = 0; i < elementCount; i++)
       {
-       toReturn.append(comma).append(elementData[i].toString());
-       comma = ", ";
+       r += elementData[i];
+       if (i < elementCount - 1)
+         r += ", ";
       }
-    return toReturn.append("]").toString();
+    r += "]";
+    return r;
   }
 
   /**
@@ -654,21 +716,42 @@
    *
    * @returns an Enumeration
    */
-  public Enumeration elements()
+  public synchronized Enumeration elements()
   {
     return new Enumeration()
     {
       int i = 0;
-      public final boolean hasMoreElements()
+      public boolean hasMoreElements()
       {
-       return (i < size());
+       return (i < elementCount);
       }
-      public final Object nextElement()
+      public Object nextElement()
       {
-       if (i >= size())
+       if (i >= elementCount)
          throw new NoSuchElementException();
        return (elementAt(i++));
       }
     };
+  }
+  
+  public List subList(int fromIndex, int toIndex)
+  {
+    List sub = super.subList(fromIndex, toIndex);
+    return Collections.synchronizedList(sub);
+  }
+  
+  /** @specnote This is not specified as synchronized in the JCL, but it seems
+    * to me that is should be. If it isn't, a clear() operation on a sublist
+    * will not be synchronized w.r.t. the Vector object.
+    */
+  protected synchronized void removeRange(int fromIndex, int toIndex)
+  {
+    modCount++;
+    if (fromIndex != toIndex)
+      {
+       System.arraycopy(elementData, toIndex, elementData, fromIndex, 
+                        elementCount - toIndex);
+       elementCount -= (toIndex - fromIndex);
+      }
   }
-}                              // Vector
+}

reply via email to

[Prev in Thread] Current Thread [Next in Thread]