View Javadoc

1   /*
2    *  Copyright 2001-2010 Stephen Colebourne
3    *
4    *  Licensed under the Apache License, Version 2.0 (the "License");
5    *  you may not use this file except in compliance with the License.
6    *  You may obtain a copy of the License at
7    *
8    *      http://www.apache.org/licenses/LICENSE-2.0
9    *
10   *  Unless required by applicable law or agreed to in writing, software
11   *  distributed under the License is distributed on an "AS IS" BASIS,
12   *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   *  See the License for the specific language governing permissions and
14   *  limitations under the License.
15   */
16  package org.joda.primitives.collection.impl;
17  
18  import java.lang.reflect.Array;
19  import java.util.Collection;
20  import java.util.Iterator;
21  
22  import org.joda.primitives.CharUtils;
23  import org.joda.primitives.collection.CharCollection;
24  import org.joda.primitives.iterator.CharIterator;
25  
26  /**
27   * Abstract base class for collections of primitive <code>char</code> elements.
28   * <p>
29   * This class implements {@link java.util.Collection Collection} allowing
30   * seamless integration with other APIs.
31   * <p>
32   * The <code>iterator</code> and <code>size</code> must be implemented by subclases.
33   * To make the subclass modifiable, the <code>add(char)</code> and
34   * iterator <code>remove()</code> methods must also be implemented.
35   * Subclasses may override other methods to increase efficiency.
36   *
37   * @author Stephen Colebourne
38   * @author Jason Tiscione
39   * @version CODE GENERATED
40   * @since 1.0
41   */
42  public abstract class AbstractCharCollection
43          extends AbstractPrimitiveCollectable<Character>
44          implements CharCollection {
45      // This file is CODE GENERATED. Do not change manually.
46  
47      /**
48       * Constructor.
49       */
50      protected AbstractCharCollection() {
51          super();
52      }
53      
54      // Mandatory operations
55      //-----------------------------------------------------------------------
56      /**
57       * Checks whether this collection contains a specified primitive value.
58       * <p>
59       * This implementation uses <code>charIterator()</code>.
60       *
61       * @param value  the value to search for
62       * @return <code>true</code> if the value is found
63       */
64      public boolean contains(char value) {
65          for (CharIterator it = iterator(); it.hasNext();) {
66              if (it.nextChar() == value) {
67                  return true;
68              }
69          }
70          return false;
71      }
72  
73      /**
74       * Checks if this collection contains all of the values in the specified array.
75       * If the specified array is empty, <code>true</code> is returned.
76       * <p>
77       * This implementation uses <code>contains(char)</code>.
78       *
79       * @param values  the values to search for, null treated as empty array
80       * @return <code>true</code> if all of the values are found
81       */
82      public boolean containsAll(char[] values) {
83          if (values != null) {
84              for (int i = 0; i < values.length; i++) {
85                  if (contains(values[i]) == false) {
86                      return false;
87                  }
88              }
89          }
90          return true;
91      }
92  
93      /**
94       * Checks if this collection contains all of the values in the specified collection.
95       * If the specified collection is empty, <code>true</code> is returned.
96       * <p>
97       * This implementation uses <code>contains(char)</code>.
98       *
99       * @param values  the values to search for, null treated as empty collection
100      * @return <code>true</code> if all of the values are found
101      */
102     public boolean containsAll(CharCollection values) {
103         if (values != null) {
104             for (CharIterator it = values.iterator(); it.hasNext(); ) {
105                 if (contains(it.nextChar()) == false) {
106                     return false;
107                 }
108             }
109         }
110         return true;
111     }
112 
113     /**
114      * Checks if this collection contain all the values in the specified range.
115      * <p>
116      * The range is defined to be inclusive of the start and end.
117      * If the start is greater than the end then the result is <code>true</code>
118      * as the range is equivalent to an empty collection.
119      * <p>
120      * This implementation uses <code>contains(char)</code>.
121      *
122      * @param startInclusive  the inclusive range start value
123      * @param endInclusive  the inclusive range end value
124      * @return <code>true</code> if the collection contains the entire range
125      */
126     public boolean containsAll(char startInclusive, char endInclusive) {
127         if (startInclusive > endInclusive) {
128             return true;
129         }
130         for (char i = startInclusive; i <= endInclusive; i++) {
131             if (contains(i) == false) {
132                 return false;
133             }
134         }
135         return true;
136     }
137 
138     /**
139      * Checks if this collection contains any of the values in the specified array.
140      * If the specified array is empty, <code>false</code> is returned.
141      * <p>
142      * This implementation uses <code>contains(char)</code>.
143      *
144      * @param values  the values to search for, null treated as empty array
145      * @return <code>true</code> if at least one of the values is found
146      */
147     public boolean containsAny(char[] values) {
148         if (values != null) {
149             for (int i = 0; i < values.length; i++) {
150                 if (contains(values[i])) {
151                     return true;
152                 }
153             }
154         }
155         return false;
156     }
157 
158     /**
159      * Checks if this collection contains any of the values in the specified collection.
160      * If the specified collection is empty, <code>false</code> is returned.
161      * <p>
162      * This implementation uses <code>contains(char)</code>.
163      *
164      * @param values  the values to search for, null treated as empty collection
165      * @return <code>true</code> if at least one of the values is found
166      */
167     public boolean containsAny(CharCollection values) {
168         if (values != null) {
169             for (CharIterator it = values.iterator(); it.hasNext(); ) {
170                 if (contains(it.nextChar())) {
171                     return true;
172                 }
173             }
174         }
175         return false;
176     }
177 
178     /**
179      * Checks if this collection contain some of the values in the specified range.
180      * <p>
181      * The range is defined to be inclusive of the start and end.
182      * If the start is greater than the end then the result is <code>false</code>
183      * as the range is equivalent to an empty collection.
184      * <p>
185      * This implementation uses <code>contains(char)</code>.
186      *
187      * @param startInclusive  the inclusive range start value
188      * @param endInclusive  the inclusive range end value
189      * @return <code>true</code> if the collection contains at least one of the range
190      */
191     public boolean containsAny(char startInclusive, char endInclusive) {
192         if (startInclusive > endInclusive) {
193             return false;
194         }
195         for (char i = startInclusive; i <= endInclusive; i++) {
196             if (contains(i)) {
197                 return true;
198             }
199         }
200         return false;
201     }
202 
203     /**
204      * Gets the elements of this collection as an array.
205      * <p>
206      * This implementation uses <code>arrayCopy</code>.
207      *
208      * @return a new array containing a copy of the elements of this collection
209      */
210     public char[] toCharArray() {
211         if (size() == 0) {
212             return CharUtils.EMPTY_CHAR_ARRAY;
213         }
214         char[] result = new char[size()];
215         arrayCopy(0, result, 0, size());
216         return result;
217     }
218 
219     /**
220      * Copies the elements of this collection into an array at a specified position.
221      * Previous values in the array are overwritten.
222      * <p>
223      * If the array specified is null a new array is created.
224      * If the array specified is large enough, it will be modified.
225      * If the array is not large enough, a new array will be created containing the
226      * values from the specified array before the startIndex plus those from this collection.
227      * <p>
228      * This implementation uses <code>arrayCopy</code>.
229      *
230      * @param array  the array to add the elements to, null treated as empty array
231      * @param startIndex  the position in the array to start setting elements
232      * @return the array with the populated collection
233      * @throws IndexOutOfBoundsException if the index is negative
234      */
235     public char[] toCharArray(char[] array, int startIndex) {
236         if (startIndex < 0) {
237             throw new IndexOutOfBoundsException("Start index must not be negative: " + startIndex);
238         }
239         char[] result = null;
240         if (array == null) {
241             // create new
242             result = new char[startIndex + size()];
243             
244         } else if (array.length - startIndex - size() >= 0) {
245             // room to fit data
246             result = array;
247             
248         } else {
249             // expand array
250             result = new char[startIndex + size()];
251             System.arraycopy(array, 0, result, 0, startIndex);
252         }
253         arrayCopy(0, result, startIndex, size());
254         return result;
255     }
256 
257     // Optional operations
258     //-----------------------------------------------------------------------
259     /**
260      * Clears the collection/map of all elements (optional operation).
261      * <p>
262      * The collection/map will have a zero size after this method completes.
263      * This method is optional, throwing an UnsupportedOperationException if the
264      * collection/map cannot be cleared.
265      * <p>
266      * This implementation uses <code>iterator()</code>.
267      *
268      * @throws UnsupportedOperationException if method not supported by this collection
269      */
270     public void clear() {
271         checkRemoveModifiable();
272         for (CharIterator it = iterator(); it.hasNext();) {
273             it.nextChar();
274             it.remove();
275         }
276     }
277 
278     /**
279      * Adds a primitive value to this collection (optional operation).
280      * <p>
281      * This implementation throws UnsupportedOperationException.
282      *
283      * @param value  the value to add to this collection
284      * @return <code>true</code> if this collection was modified by this method call
285      * @throws IllegalArgumentException if value is rejected by this collection
286      * @throws UnsupportedOperationException if not supported by this collection
287      */
288     public boolean add(char value) {
289         throw new UnsupportedOperationException("Collection does not support add");
290     }
291 
292     /**
293      * Adds an array of primitive values to this collection (optional operation).
294      * <p>
295      * This implementation uses <code>add(char)</code>.
296      *
297      * @param values  the values to add to this collection, null treated as empty array
298      * @return <code>true</code> if this collection was modified by this method call
299      * @throws IllegalArgumentException if a value is rejected by this collection
300      * @throws UnsupportedOperationException if not supported by this collection
301      */
302     public boolean addAll(char[] values) {
303         checkAddModifiable();
304         boolean changed = false;
305         if (values != null) {
306             for (int i = 0; i < values.length; i++) {
307                 changed |= add(values[i]);
308             }
309         }
310         return changed;
311     }
312 
313     /**
314      * Adds a collection of primitive values to this collection (optional operation).
315      * <p>
316      * This implementation uses <code>add(char)</code>.
317      *
318      * @param values  the values to add to this collection, null treated as empty collection
319      * @return <code>true</code> if this collection was modified by this method call
320      * @throws IllegalArgumentException if a value is rejected by this collection
321      * @throws UnsupportedOperationException if not supported by this collection
322      */
323     public boolean addAll(CharCollection values) {
324         checkAddModifiable();
325         boolean changed = false;
326         if (values != null) {
327             for (CharIterator it = values.iterator(); it.hasNext(); ) {
328                 changed |= add(it.nextChar());
329             }
330         }
331         return changed;
332     }
333 
334     /**
335      * Adds a range of primitive values to this collection (optional operation).
336      * <p>
337      * The range is defined to be inclusive of the start and end.
338      * If the start is greater than the end then the range is equivalent to an empty collection.
339      * <p>
340      * This implementation uses <code>add(char)</code>.
341      *
342      * @param startInclusive  the inclusive range start value
343      * @param endInclusive  the inclusive range end value
344      * @return <code>true</code> if this collection was modified by this method call
345      * @throws IllegalArgumentException if a value is rejected by this set
346      * @throws UnsupportedOperationException if not supported by this set
347      */
348     public boolean addAll(char startInclusive, char endInclusive) {
349         checkAddModifiable();
350         if (startInclusive > endInclusive) {
351             return false;
352         }
353         boolean changed = false;
354         for (char i = startInclusive; i <= endInclusive; i++) {
355             changed |= add(i);
356         }
357         return false;
358     }
359 
360     /**
361      * Removes the first occurrence of the specified primitive value from this collection
362      * <p>
363      * This implementation uses <code>iterator().remove()</code>.
364      *
365      * @param value  the value to remove
366      * @return <code>true</code> if this collection was modified by this method call
367      * @throws UnsupportedOperationException if not supported by this collection
368      */
369     public boolean removeFirst(char value) {
370         checkRemoveModifiable();
371         for (CharIterator it = iterator(); it.hasNext(); ) {
372             if (it.nextChar() == value) {
373                 it.remove();
374                 return true;
375             }
376         }
377         return false;
378     }
379 
380     /**
381      * Removes all occurrences of the specified primitive value from this collection.
382      * <p>
383      * This implementation uses <code>iterator().remove()</code>.
384      *
385      * @param value  the value to remove
386      * @return <code>true</code> if this collection was modified by this method call
387      * @throws UnsupportedOperationException if not supported by this collection
388      */
389     public boolean removeAll(char value) {
390         checkRemoveModifiable();
391         boolean changed = false;
392         for (CharIterator it = iterator(); it.hasNext(); ) {
393             if (it.nextChar() == value) {
394                 it.remove();
395                 changed = true;
396             }
397         }
398         return changed;
399     }
400 
401     /**
402      * Removes all occurrences from this collection of each primitive in the specified array.
403      * <p>
404      * This implementation uses <code>iterator().remove()</code>.
405      *
406      * @param values  the values to remove from this collection, null treated as empty array
407      * @return <code>true</code> if this list was modified by this method call
408      * @throws UnsupportedOperationException if not supported by this collection
409      */
410     public boolean removeAll(char[] values) {
411         checkRemoveModifiable();
412         boolean changed = false;
413         if (values != null) {
414             for (CharIterator it = iterator(); it.hasNext(); ) {
415                 char value = it.nextChar();
416                 for (int i = 0; i < values.length; i++) {
417                     if (values[i] == value) {
418                         it.remove();
419                         changed = true;
420                     }
421                 }
422             }
423         }
424         return changed;
425     }
426 
427     /**
428      * Removes all occurrences from this collection of each primitive in the specified collection.
429      * <p>
430      * This implementation uses <code>iterator().remove()</code>.
431      *
432      * @param values  the values to remove from this collection, null treated as empty collection
433      * @return <code>true</code> if this list was modified by this method call
434      * @throws UnsupportedOperationException if not supported by this collection
435      */
436     public boolean removeAll(CharCollection values) {
437         checkRemoveModifiable();
438         boolean changed = false;
439         if (values != null) {
440             for (CharIterator it = iterator(); it.hasNext(); ) {
441                 if (values.contains(it.nextChar())) {
442                     it.remove();
443                     changed = true;
444                 }
445             }
446         }
447         return changed;
448     }
449 
450     /**
451      * Removes all occurrences of a range of primitive values from this collection.
452      * <p>
453      * The range is defined to be inclusive of the start and end.
454      * The elements removed are greater than or equal to the start and
455      * less than or equal to the end. Thus if the start is greater than the
456      * end then no elements are removed.
457      * <p>
458      * This method is optional, throwing an UnsupportedOperationException if the
459      * set cannot be changed.
460      * <p>
461      * This implementation uses <code>iterator().remove()</code>.
462      *
463      * @param startInclusive  the inclusive range start value
464      * @param endInclusive  the inclusive range end value
465      * @return <code>true</code> if this collection was modified by this method call
466      * @throws UnsupportedOperationException if not supported by this collection
467      */
468     public boolean removeAll(char startInclusive, char endInclusive) {
469         checkRemoveModifiable();
470         if (startInclusive > endInclusive) {
471             return false;
472         }
473         boolean changed = false;
474         for (CharIterator it = iterator(); it.hasNext(); ) {
475             char value = it.nextChar();
476             if (value >= startInclusive && value <= endInclusive) {
477                 it.remove();
478                 changed = true;
479             }
480         }
481         return changed;
482     }
483 
484     /**
485      * Retains each element of this collection that is present in the specified array
486      * removing all other values.
487      * <p>
488      * This implementation uses <code>iterator().remove()</code>.
489      *
490      * @param values  the values to remove from this collection, null treated as empty array
491      * @return <code>true</code> if this list was modified by this method call
492      * @throws UnsupportedOperationException if not supported by this collection
493      */
494     public boolean retainAll(char[] values) {
495         checkRemoveModifiable();
496         boolean changed = false;
497         if (values == null || values.length == 0) {
498             changed = !isEmpty();
499             clear();
500         } else {
501             for (CharIterator it = iterator(); it.hasNext(); ) {
502                 char next = it.nextChar();
503                 boolean match = false;
504                 for (int i = 0; i < values.length; i++) {
505                     if (values[i] == next) {
506                         match = true;
507                         break;
508                     }
509                 }
510                 if (match == false) {
511                     it.remove();
512                     changed = true;
513                 }
514             }
515         }
516         return changed;
517     }
518 
519     /**
520      * Retains each element of this collection that is present in the specified collection
521      * removing all other values.
522      * <p>
523      * This implementation uses <code>iterator().remove()</code>.
524      *
525      * @param values  the values to retain in this collection, null treated as empty collection
526      * @return <code>true</code> if this collection was modified by this method call
527      * @throws UnsupportedOperationException if not supported by this collection
528      */
529     public boolean retainAll(CharCollection values) {
530         checkRemoveModifiable();
531         boolean changed = false;
532         if (values == null || values.isEmpty()) {
533             changed = !isEmpty();
534             clear();
535         } else {
536             for (CharIterator it = iterator(); it.hasNext(); ) {
537                 if (values.contains(it.nextChar()) == false) {
538                     it.remove();
539                     changed = true;
540                 }
541             }
542         }
543         return changed;
544     }
545 
546     /**
547      * Retains all occurences of a range of primitive values within this collection
548      * removing all values outside the range (optional operation).
549      * <p>
550      * The range is defined to be inclusive of the start and end.
551      * If the start is greater than the end then the range is equivalent to an empty collection.
552      * <p>
553      * This method is optional, throwing an UnsupportedOperationException if the
554      * set cannot be changed.
555      *
556      * @param startInclusive  the inclusive range start value
557      * @param endInclusive  the inclusive range end value
558      * @return <code>true</code> if this collection was modified by this method call
559      * @throws UnsupportedOperationException if not supported by this collection
560      */
561     public boolean retainAll(char startInclusive, char endInclusive) {
562         checkRemoveModifiable();
563         boolean changed = false;
564         for (CharIterator it = iterator(); it.hasNext(); ) {
565             char value = it.nextChar();
566             if (value < startInclusive || value > endInclusive) {
567                 it.remove();
568                 changed = true;
569             }
570         }
571         return changed;
572     }
573 
574     // Collection integration
575     //-----------------------------------------------------------------------
576     /**
577      * Checks whether this collection contains a specified <code>Character</code> value.
578      * <p>
579      * This implementation uses <code>contains(char)</code>.
580      *
581      * @param value  the value to search for
582      * @return <code>true</code> if the value is found
583      */
584     public boolean contains(Object value) {
585         return contains(toPrimitive(value));
586     }
587 
588     /**
589      * Checks if the collection contains all of the primitive values.
590      * <p>
591      * This implementation uses <code>containsAll(char[])</code>.
592      *
593      * @param coll  the collection of values to search for
594      * @return <code>true</code> if all the values are found
595      */
596     public boolean containsAll(Collection<?> coll) {
597         if (coll == this || coll.size() == 0) {
598             return true;
599         }
600         if (size() == 0) {
601             return false;
602         }
603         return containsAll(toPrimitiveArray(coll));
604     }
605 
606     /**
607      * Checks if the collection contains any of the primitive values in the array.
608      * If the specified collection is empty, <code>false</code> is returned.
609      * <p>
610      * This implementation uses <code>containsAny(char[])</code>.
611      *
612      * @param coll  the collection of values to search for
613      * @return <code>true</code> if at least one of the values is found
614      */
615     public boolean containsAny(Collection<?> coll) {
616         if (size() == 0 || coll.size() == 0) {
617             return false;
618         }
619         if (coll == this) {
620             return true;
621         }
622         return containsAny(toPrimitiveArray(coll));
623     }
624 
625     /**
626      * Gets the collection as an array of <code>Character</code>.
627      * 
628      * @return an array of <code>Character</code>
629      */
630     public Object[] toArray() {
631         Object[] result = new Character[size()];
632         CharIterator it = iterator();
633         for (int i = 0; it.hasNext(); i++) {
634             result[i] = it.next();
635         }
636         return result;
637     }
638 
639     /**
640      * Gets the collection as an array, using the array provided.
641      * 
642      * @param array  the array to populate
643      * @return an array of <code>Character</code>
644      */
645     @SuppressWarnings("unchecked")
646     public <T> T[] toArray(T[] array) {
647         int size = size();
648         if (array.length < size) {
649           array = (T[]) Array.newInstance(array.getClass().getComponentType(), size);
650         }
651 
652         Iterator<Character> it = iterator();
653         for (int i = 0; i < size; i++) {
654             array[i] = (T)it.next();
655         }
656 
657         if (array.length > size) {
658             array[size] = null;
659         }
660         
661         return array;
662     }
663 
664     /**
665      * Adds the <code>Character</code> value to this collection (optional operation).
666      * <p>
667      * This method is optional, throwing an UnsupportedOperationException if the
668      * collection cannot be added to.
669      * <p>
670      * This implementation uses <code>add(char)</code>.
671      *
672      * @param value  the value to add to this collection
673      * @return <code>true</code> if this collection was modified by this method call
674      * @throws IllegalArgumentException if value is rejected by this collection
675      * @throws UnsupportedOperationException if not supported by this collection
676      */
677     public boolean add(Character value) {
678         checkAddModifiable();
679         return add(toPrimitive(value));
680     }
681 
682     /**
683      * Adds a collection of <code>Character</code> values to this collection (optional operation).
684      * <p>
685      * This method is optional, throwing an UnsupportedOperationException if the
686      * collection cannot be added to.
687      * <p>
688      * This implementation uses <code>addAll(char[])</code>.
689      *
690      * @param coll  the values to add to this collection
691      * @return <code>true</code> if this list was modified by this method call
692      * @throws IndexOutOfBoundsException if the index is invalid
693      * @throws ClassCastException if any object is not <code>Character</code>
694      * @throws IllegalArgumentException if value is rejected by this collection
695      * @throws UnsupportedOperationException if not supported by this collection
696      */
697     public boolean addAll(Collection<? extends Character> coll) {
698         checkAddModifiable();
699         return addAll(toPrimitiveArray(coll));
700     }
701 
702     /**
703      * Removes the first occurrance of the specified <code>Character</code> value from
704      * this collection (optional operation).
705      * <p>
706      * This method is optional, throwing an UnsupportedOperationException if the
707      * collection cannot be removed from.
708      * <p>
709      * This implementation uses <code>removeFirst(char)</code>.
710      *
711      * @param value  the value to remove
712      * @return <code>true</code> if this collection was modified by this method call
713      * @throws UnsupportedOperationException if not supported by this collection
714      */
715     public boolean remove(Object value) {
716         checkRemoveModifiable();
717         return removeFirst(toPrimitive(value));
718     }
719 
720     /**
721      * Removes each of a collection of <code>Character</code> values from this collection (optional operation).
722      * <p>
723      * This method is optional, throwing an UnsupportedOperationException if the
724      * collection cannot be added to.
725      * <p>
726      * This implementation uses <code>removeAll(char[])</code>.
727      *
728      * @param coll  the values to remove from this collection
729      * @return <code>true</code> if this list was modified by this method call
730      * @throws UnsupportedOperationException if not supported by this collection
731      */
732     public boolean removeAll(Collection<?> coll) {
733         checkRemoveModifiable();
734         if (coll == this) {
735             int size = size();
736             clear();
737             return (size() != size);
738         }
739         return removeAll(toPrimitiveArray(coll));
740     }
741 
742     /**
743      * Retains each of a collection of <code>Character</code> values, removing other
744      * values (optional operation).
745      * <p>
746      * This method is optional, throwing an UnsupportedOperationException if the
747      * collection cannot be added to.
748      * <p>
749      * This implementation uses <code>retainAll(char[])</code>.
750      *
751      * @param coll  the values to retain in this collection
752      * @return <code>true</code> if this list was modified by this method call
753      * @throws UnsupportedOperationException if not supported by this collection
754      */
755     public boolean retainAll(Collection<?> coll) {
756         checkRemoveModifiable();
757         if (coll == this) {
758             return false;
759         }
760         return retainAll(toPrimitiveArray(coll));
761     }
762 
763     // Basics
764     //-----------------------------------------------------------------------
765     /**
766      * Gets a string representing this collection.
767      * <p>
768      * The format used is as per <code>Collection</code>.
769      * 
770      * @return collection as a String
771      */
772     public String toString() {
773         StringBuffer buf = new StringBuffer();
774         buf.append("[");
775 
776         CharIterator it = iterator();
777         boolean hasNext = it.hasNext();
778         while (hasNext) {
779             buf.append(it.nextChar());
780             hasNext = it.hasNext();
781             if (hasNext) {
782                 buf.append(", ");
783             }
784         }
785 
786         buf.append("]");
787         return buf.toString();
788     }
789     
790     // Internals
791     //-----------------------------------------------------------------------
792     /**
793      * Copies data from this collection into the specified array.
794      * This method is pre-validated.
795      * 
796      * @param fromIndex  the index to start from
797      * @param dest  the destination array
798      * @param destIndex  the destination start index
799      * @param size  the number of items to copy
800      */
801     protected void arrayCopy(int fromIndex, char[] dest, int destIndex, int size) {
802         CharIterator it = iterator();
803         for (int i = 0; it.hasNext() && i < size; i++) {
804             dest[destIndex + i] = it.nextChar();
805         }
806     }
807 
808     /**
809      * Are the add methods supported.
810      * <p>
811      * This implementation returns false.
812      *
813      * @return true if supported
814      */
815     protected boolean isAddModifiable() {
816         return false;
817     }
818 
819     /**
820      * Are the remove methods supported.
821      * <p>
822      * This implementation returns false.
823      *
824      * @return true if supported
825      */
826     protected boolean isRemoveModifiable() {
827         return false;
828     }
829 
830     /**
831      * Is the collection modifiable in any way.
832      *
833      * @return true if supported
834      */
835     public boolean isModifiable() {
836         return isAddModifiable() || isRemoveModifiable();
837     }
838 
839     /**
840      * Check whether add is suported and throw an exception.
841      */
842     protected void checkAddModifiable() {
843         if (isAddModifiable() == false) {
844             throw new UnsupportedOperationException("Collection does not support add");
845         }
846     }
847 
848     /**
849      * Check whether remove is suported and throw an exception.
850      */
851     protected void checkRemoveModifiable() {
852         if (isRemoveModifiable() == false) {
853             throw new UnsupportedOperationException("Collection does not support remove");
854         }
855     }
856 
857     /**
858      * Wraps an <code>char</code> with an Object wrapper.
859      * 
860      * @param value  the primitive value
861      * @return the Object wrapper
862      */
863     protected Character toObject(char value) {
864         return CharUtils.toObject(value);
865     }
866 
867     /**
868      * Checks if the object can be converted to a primitive successfully.
869      * <p>
870      * This implementation only allows non-null Char objects.
871      * 
872      * @param value  the Object wrapper
873      * @return true if a primitive value can be successfully extracted
874      */
875     protected boolean isToPrimitivePossible(Object value) {
876         return (value instanceof Character);
877     }
878 
879     /**
880      * Unwraps the <code>Character</code> to retrieve the primitive <code>char</code>.
881      * <p>
882      * This implementation only allows non-null Char objects.
883      * 
884      * @param value  the Object to convert to a primitive
885      * @return the primitive value
886      * @throws NullPointerException if the value is null and this is unacceptable
887      * @throws ClassCastException if the object is of an unsuitable type
888      */
889     protected char toPrimitive(Object value) {
890         return CharUtils.toPrimitive(value);
891     }
892 
893     /**
894      * Unwraps a <code>Collection</code> to retrieve the primitive <code>char</code>.
895      * <p>
896      * This implementation only allows non-null Char objects.
897      * 
898      * @param coll  the Collection to convert to primitives
899      * @return the primitive value
900      * @throws NullPointerException if the value is null and this is unacceptable
901      * @throws ClassCastException if any object is of an unsuitable type
902      */
903     protected char[] toPrimitiveArray(Collection<?> coll) {
904         return CharUtils.toPrimitiveArray(coll);
905     }
906 
907 }