/******************************************************************************* * Copyright (c) 2000, 2009 IBM Corporation and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * IBM Corporation - initial API and implementation *******************************************************************************/ package org.eclipse.jdt.internal.core.util; import org.eclipse.jdt.internal.compiler.util.Util; /** * The <code>CharArrayBuffer</code> is intended as a lightweight partial implementation of the * StringBuffer class, but using <code>char[]'s</code> instead of Strings. * * <p> * The <code>CharArrayBuffer</code> maintains a list of <code>char[]'s</code> which don't get * appended until the user asks for them. The following code illustrates how to use the class. * * <code> * CharArrayBuffer buffer = new CharArrayBuffer(myCharArray); * buffer.append(moreBytes, 0, someLength); * myCharArray = buffer.getContents(); * </code> * * <p> * NOTE: This class is not Thread safe! */ public class CharArrayBuffer { /** * This is the buffer of char arrays which must be appended together during the getContents * method. */ protected char[][] buffer; /** * The default buffer size. */ public static final int DEFAULT_BUFFER_SIZE= 10; /** * The end of the buffer */ protected int end; /** * The current size of the buffer. */ protected int size; /** * A buffer of ranges which is maintained along with the buffer. Ranges are of the form {start, * length}. Enables append(char[] array, int start, int end). */ protected int[][] ranges; /** * Creates a <code>CharArrayBuffer</code> with the default buffer size (10). */ public CharArrayBuffer() { this(null, DEFAULT_BUFFER_SIZE); } /** * Creates a <code>CharArrayBuffer</code> with the default buffer size, and sets the first * element in the buffer to be the given char[]. * * @param first - the first element to be placed in the buffer, ignored if null */ public CharArrayBuffer(char[] first) { this(first, DEFAULT_BUFFER_SIZE); } /** * Creates a <code>CharArrayBuffer</code> with the given buffer size, and sets the first element * in the buffer to be the given char array. * * @param first - the first element of the buffer, ignored if null. * @param size - the buffer size, if less than 1, set to the DEFAULT_BUFFER_SIZE. */ public CharArrayBuffer(char[] first, int size) { this.size= (size > 0) ? size : DEFAULT_BUFFER_SIZE; this.buffer= new char[this.size][]; this.ranges= new int[this.size][]; this.end= 0; if (first != null) append(first, 0, first.length); } /** * Creates a <code>CharArrayBuffer</code> with the given buffer size. * * @param size - the size of the buffer. */ public CharArrayBuffer(int size) { this(null, size); } /** * Appends the entire given char array. Given for convenience. * * @param src - a char array which is appended to the end of the buffer. */ public CharArrayBuffer append(char[] src) { if (src != null) append(src, 0, src.length); return this; } /** * Appends a sub array of the given array to the buffer. * * @param src - the next array of characters to be appended to the buffer, ignored if null * @param start - the start index in the src array. * @param length - the number of characters from start to be appended * * @throws ArrayIndexOutOfBoundsException - if arguments specify an array index out of bounds. */ public CharArrayBuffer append(char[] src, int start, int length) { if (start < 0) throw new ArrayIndexOutOfBoundsException(); if (length < 0) throw new ArrayIndexOutOfBoundsException(); if (src != null) { int srcLength= src.length; if (start > srcLength) throw new ArrayIndexOutOfBoundsException(); if (length + start > srcLength) throw new ArrayIndexOutOfBoundsException(); /** do length check here to allow exceptions to be thrown */ if (length > 0) { if (this.end == this.size) { int size2= this.size * 2; System.arraycopy(this.buffer, 0, (this.buffer= new char[size2][]), 0, this.size); System.arraycopy(this.ranges, 0, (this.ranges= new int[size2][]), 0, this.size); this.size*= 2; } this.buffer[this.end]= src; this.ranges[this.end]= new int[] { start, length }; this.end++; } } return this; } /** * Appends the given char. Given for convenience. * * @param c - a char which is appended to the end of the buffer. */ public CharArrayBuffer append(char c) { append(new char[] { c }, 0, 1); return this; } /** * Appends the given String to the buffer. Given for convenience, use #append(char[]) if * possible * * @param src - a char array which is appended to the end of the buffer. */ public CharArrayBuffer append(String src) { if (src != null) append(src.toCharArray(), 0, src.length()); return this; } /** * Returns the entire contents of the buffer as one char[] or null if nothing has been put in * the buffer. */ public char[] getContents() { if (this.end == 0) return null; // determine the length of the array int length= 0; for (int i= 0; i < this.end; i++) length+= this.ranges[i][1]; if (length > 0) { char[] result= new char[length]; int current= 0; // copy the results for (int i= 0; i < this.end; i++) { int[] range= this.ranges[i]; int length2= range[1]; System.arraycopy(this.buffer[i], range[0], result, current, length2); current+= length2; } return result; } return null; } /** * Returns the contents of the buffer as a String, or an empty string if the buffer is empty. */ public String toString() { char[] contents= getContents(); return (contents != null) ? new String(contents) : Util.EMPTY_STRING; } }