Package java.lang

Source Code of java.lang.String$CaseInsensitiveComparator

/*
*  Licensed to the Apache Software Foundation (ASF) under one or more
*  contributor license agreements.  See the NOTICE file distributed with
*  this work for additional information regarding copyright ownership.
*  The ASF licenses this file to You under the Apache License, Version 2.0
*  (the "License"); you may not use this file except in compliance with
*  the License.  You may obtain a copy of the License at
*
*     http://www.apache.org/licenses/LICENSE-2.0
*
*  Unless required by applicable law or agreed to in writing, software
*  distributed under the License is distributed on an "AS IS" BASIS,
*  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*  See the License for the specific language governing permissions and
*  limitations under the License.
*/

package java.lang;

import java.io.Serializable;
import java.io.UnsupportedEncodingException;
import java.util.Comparator;
import java.util.Formatter;
import java.util.Locale;

import java.util.regex.Pattern;

import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.Charset;
import java.nio.charset.IllegalCharsetNameException;
import java.nio.charset.UnsupportedCharsetException;
import java.security.AccessController;
import java.util.regex.PatternSyntaxException;

import org.apache.harmony.kernel.vm.VM;
import org.apache.harmony.luni.util.PriviAction;

/**
* <p>
* An immutable sequence of characters/code units (<code>char</code>s). A
* <code>String</code> is represented by array of UTF-16 values, such that
* Unicode supplementary characters (code points) are stored/encoded as
* surrogate pairs via Unicode code units (<code>char</code>)
* </p>
*
* @see StringBuffer
* @see StringBuilder
* @see Charset
* @since 1.0
*/
public final class String implements Serializable, Comparable<String>,
        CharSequence {

    private static final long serialVersionUID = -6849794470754667710L;

    /**
     * An PrintStream used for System.out which performs the correct character
     * conversion for the console, since the console may use a different
     * conversion than the default file.encoding.
     */
    static class ConsolePrintStream extends java.io.PrintStream {
        private static String charset;

        static {
            charset = AccessController.doPrivileged(new PriviAction<String>(
                    "console.encoding", "ISO8859_1")); //$NON-NLS-1$ //$NON-NLS-2$
            if (!Charset.isSupported(charset)) {
                charset = "ISO-8859-1"; //$NON-NLS-1$
            }
        }

        /**
         * Create a ConsolePrintStream on the specified OutputStream, usually
         * System.out.
         *
         * @param out
         *            the console OutputStream
         */
        public ConsolePrintStream(java.io.OutputStream out) {
            super(out, true);

        }

        /**
         * Override the print(String) method from PrintStream to perform the
         * character conversion using the console character converter.
         *
         * @param str
         *            the String to convert
         */
        @Override
        public void print(String str) {
            if (str == null) {
                str = "null"; //$NON-NLS-1$
            }

            try {
                write(str.getBytes(charset));
            } catch (java.io.IOException e) {
                setError();
            }
        }
    }

    /**
     * CaseInsensitiveComparator compares Strings ignoring the case of the
     * characters.
     */
    private static final class CaseInsensitiveComparator implements
            Comparator<String>, Serializable {
        private static final long serialVersionUID = 8575799808933029326L;

        /**
         * Compare the two objects to determine the relative ordering.
         *
         * @param o1
         *            an Object to compare
         * @param o2
         *            an Object to compare
         * @return an int < 0 if object1 is less than object2, 0 if they are
         *         equal, and > 0 if object1 is greater
         *
         * @exception ClassCastException
         *                when objects are not the correct type
         */
        public int compare(String o1, String o2) {
            return o1.compareToIgnoreCase(o2);
        }
    }

    /*
     * A Comparator which compares Strings ignoring the case of the characters.
     */
    public static final Comparator<String> CASE_INSENSITIVE_ORDER = new CaseInsensitiveComparator();

    private static final char[] ascii;

    private final char[] value;

    private final int offset;

    private final int count;

    private int hashCode;

    private static Charset DefaultCharset;

    private static Charset lastCharset;

    static {
        ascii = new char[128];
        for (int i = 0; i < ascii.length; i++) {
            ascii[i] = (char) i;
        }
    }

    /**
     * Answers an empty string.
     */
    public String() {
        value = new char[0];
        offset = 0;
        count = 0;
    }

    @SuppressWarnings("unused")
    private String(String s, char c) {
        offset = 0;
        value = new char[s.count + 1];
        count = s.count + 1;
        System.arraycopy(s.value, s.offset, value, 0, s.count);
        value[s.count] = c;
    }

    /**
     * Converts the byte array to a String using the default encoding as
     * specified by the file.encoding system property. If the system property is
     * not defined, the default encoding is ISO8859_1 (ISO-Latin-1). If 8859-1
     * is not available, an ASCII encoding is used.
     *
     * @param data
     *            the byte array to convert to a String
     *
     * @throws NullPointerException
     *             when data is null
     *
     * @see #getBytes()
     * @see #getBytes(int, int, byte[], int)
     * @see #getBytes(String)
     * @see #valueOf(boolean)
     * @see #valueOf(char)
     * @see #valueOf(char[])
     * @see #valueOf(char[], int, int)
     * @see #valueOf(double)
     * @see #valueOf(float)
     * @see #valueOf(int)
     * @see #valueOf(long)
     * @see #valueOf(Object)
     *
     */
    public String(byte[] data) {
        this(data, 0, data.length);
    }

    /**
     * Converts the byte array to a String, setting the high byte of every
     * character to the specified value.
     *
     * @param data
     *            the byte array to convert to a String
     * @param high
     *            the high byte to use
     *
     * @throws NullPointerException
     *             when data is null
     *
     * @deprecated Use String(byte[]) or String(byte[], String) instead
     */
    @Deprecated
    public String(byte[] data, int high) {
        this(data, high, 0, data.length);
    }

    /**
     * Converts the byte array to a String using the default encoding as
     * specified by the file.encoding system property. If the system property is
     * not defined, the default encoding is ISO8859_1 (ISO-Latin-1). If 8859-1
     * is not available, an ASCII encoding is used.
     *
     * @param data
     *            the byte array to convert to a String
     * @param start
     *            the starting offset in the byte array
     * @param length
     *            the number of bytes to convert
     *
     * @throws IndexOutOfBoundsException
     *             when <code>length < 0, start < 0</code> or
     *             <code>start + length > data.length</code>
     * @throws NullPointerException
     *             when data is null
     *
     * @see #getBytes()
     * @see #getBytes(int, int, byte[], int)
     * @see #getBytes(String)
     * @see #valueOf(boolean)
     * @see #valueOf(char)
     * @see #valueOf(char[])
     * @see #valueOf(char[], int, int)
     * @see #valueOf(double)
     * @see #valueOf(float)
     * @see #valueOf(int)
     * @see #valueOf(long)
     * @see #valueOf(Object)
     */
    public String(byte[] data, int start, int length) {
        // start + length could overflow, start/length maybe MaxInt
        if (start >= 0 && 0 <= length && length <= data.length - start) {
            offset = 0;
            Charset charset = defaultCharset();
            int result;
            CharBuffer cb = charset
                    .decode(ByteBuffer.wrap(data, start, length));
            if ((result = cb.length()) > 0) {
                value = cb.array();
                count = result;
            } else {
                count = 0;
                value = new char[0];
            }
        } else {
            throw new StringIndexOutOfBoundsException();
        }
    }

    /**
     * Converts the byte array to a String, setting the high byte of every
     * character to the specified value.
     *
     * @param data
     *            the byte array to convert to a String
     * @param high
     *            the high byte to use
     * @param start
     *            the starting offset in the byte array
     * @param length
     *            the number of bytes to convert
     *
     * @throws IndexOutOfBoundsException
     *             when <code>length < 0, start < 0</code> or
     *             <code>start + length > data.length</code>
     * @throws NullPointerException
     *             when data is null
     *
     * @deprecated Use String(byte[], int, int) instead
     */
    @Deprecated
    public String(byte[] data, int high, int start, int length) {
        if (data != null) {
            // start + length could overflow, start/length maybe MaxInt
            if (start >= 0 && 0 <= length && length <= data.length - start) {
                offset = 0;
                value = new char[length];
                count = length;
                high <<= 8;
                for (int i = 0; i < count; i++) {
                    value[i] = (char) (high + (data[start++] & 0xff));
                }
            } else {
                throw new StringIndexOutOfBoundsException();
            }
        } else {
            throw new NullPointerException();
        }
    }

    /**
     * Converts the byte array to a String using the specified encoding.
     *
     * @param data
     *            the byte array to convert to a String
     * @param start
     *            the starting offset in the byte array
     * @param length
     *            the number of bytes to convert
     * @param encoding
     *            the encoding
     *
     * @throws IndexOutOfBoundsException
     *             when <code>length < 0, start < 0</code> or
     *             <code>start + length > data.length</code>
     * @throws UnsupportedEncodingException
     *             when encoding is not supported
     * @throws NullPointerException
     *             when data is null
     *
     * @see #getBytes()
     * @see #getBytes(int, int, byte[], int)
     * @see #getBytes(String)
     * @see #valueOf(boolean)
     * @see #valueOf(char)
     * @see #valueOf(char[])
     * @see #valueOf(char[], int, int)
     * @see #valueOf(double)
     * @see #valueOf(float)
     * @see #valueOf(int)
     * @see #valueOf(long)
     * @see #valueOf(Object)
     * @see UnsupportedEncodingException
     */
    public String(byte[] data, int start, int length, final String encoding)
            throws UnsupportedEncodingException {
        if (encoding == null) {
            throw new NullPointerException();
        }
        // start + length could overflow, start/length maybe MaxInt
        if (start >= 0 && 0 <= length && length <= data.length - start) {
            offset = 0;
            Charset charset = getCharset(encoding);

            int result;
          CharBuffer cb;
            try {
              cb = charset.decode(ByteBuffer.wrap(data, start, length));
            } catch (Exception e) {
              // do nothing. according to spec:
              // behavior is unspecified for invalid array
              cb = CharBuffer.wrap("\u003f".toCharArray()); //$NON-NLS-1$
            }
            if ((result = cb.length()) > 0) {
                value = cb.array();
                count = result;
            } else {
                count = 0;
                value = new char[0];
            }
        } else {
            throw new StringIndexOutOfBoundsException();
        }
    }

    /**
     * Converts the byte array to a String using the specified encoding.
     *
     * @param data
     *            the byte array to convert to a String
     * @param encoding
     *            the encoding
     *
     * @throws UnsupportedEncodingException
     *             when encoding is not supported
     * @throws NullPointerException
     *             when data is null
     *
     * @see #getBytes()
     * @see #getBytes(int, int, byte[], int)
     * @see #getBytes(String)
     * @see #valueOf(boolean)
     * @see #valueOf(char)
     * @see #valueOf(char[])
     * @see #valueOf(char[], int, int)
     * @see #valueOf(double)
     * @see #valueOf(float)
     * @see #valueOf(int)
     * @see #valueOf(long)
     * @see #valueOf(Object)
     * @see UnsupportedEncodingException
     */
    public String(byte[] data, String encoding) throws UnsupportedEncodingException {
        this(data, 0, data.length, encoding);
    }

    /**
     * Initializes this String to contain the characters in the specified
     * character array. Modifying the character array after creating the String
     * has no effect on the String.
     *
     * @param data
     *            the array of characters
     *
     * @throws NullPointerException
     *             when data is null
     */
    public String(char[] data) {
        this(data, 0, data.length);
    }

    /**
     * Initializes this String to contain the specified characters in the
     * character array. Modifying the character array after creating the String
     * has no effect on the String.
     *
     * @param data
     *            the array of characters
     * @param start
     *            the starting offset in the character array
     * @param length
     *            the number of characters to use
     *
     * @throws IndexOutOfBoundsException
     *             when <code>length < 0, start < 0</code> or
     *             <code>start + length > data.length</code>
     * @throws NullPointerException
     *             when data is null
     */
    public String(char[] data, int start, int length) {
        // range check everything so a new char[] is not created
        // start + length could overflow, start/length maybe MaxInt
        if (start >= 0 && 0 <= length && length <= data.length - start) {
            offset = 0;
            value = new char[length];
            count = length;
            System.arraycopy(data, start, value, 0, count);
        } else {
            throw new StringIndexOutOfBoundsException();
        }
    }

    /*
     * Internal version of string constructor. Does not range check, null check,
     * or copy the character array.
     */
    String(int start, int length, char[] data) {
        value = data;
        offset = start;
        count = length;
    }

    /**
     * Creates a string that is a copy of another string
     *
     * @param string
     *            the String to copy
     */
    public String(String string) {
        value = string.value;
        offset = string.offset;
        count = string.count;
    }

    /**
     * Creates a string from the contents of a StringBuffer.
     *
     * @param stringbuffer
     *            the StringBuffer
     */
    public String(StringBuffer stringbuffer) {
        offset = 0;
        synchronized (stringbuffer) {
            value = stringbuffer.shareValue();
            count = stringbuffer.length();
        }
    }

    /**
     * <p>
     * Constructs a <code>String</code> from the sub-array of Unicode code
     * points.
     * </p>
     *
     * @param codePoints
     *            The array of Unicode code points to convert.
     * @param offset
     *            The inclusive index into <code>codePoints</code> to begin
     *            converting from.
     * @param count
     *            The number of element in <code>codePoints</code> to copy.
     * @throws NullPointerException
     *             if <code>codePoints</code> is null.
     * @throws IllegalArgumentException
     *             if any of the elements of <code>codePoints</code> are not
     *             valid Unicode code points.
     * @throws IndexOutOfBoundsException
     *             if <code>offset</code> or <code>count</code> are not
     *             within the bounds of <code>codePoints</code>.
     * @since 1.5
     */
    public String(int[] codePoints, int offset, int count) {
        super();
        if (codePoints == null) {
            throw new NullPointerException();
        }
        if (offset < 0 || count < 0
                || (long) offset + (long) count > codePoints.length) {
            throw new IndexOutOfBoundsException();
        }
        this.offset = 0;
        this.value = new char[count * 2];
        int end = offset + count;
        int c = 0;
        for (int i = offset; i < end; i++) {
            c += Character.toChars(codePoints[i], this.value, c);
        }
        this.count = c;
    }

    /**
     * <p>
     * Constructs a <code>String</code> from a <code>StringBuilder</code>.
     * </p>
     *
     * @param sb
     *            The StringBuilder to copy from.
     * @throws NullPointerException
     *             if <code>sb</code> is <code>null</code>.
     * @since 1.5
     */
    public String(StringBuilder sb) {
        if (sb == null) {
            throw new NullPointerException();
        }
        this.offset = 0;
        this.count = sb.length();
        this.value = new char[this.count];
        sb.getChars(0, this.count, this.value, 0);
    }

    /*
     * Creates a string that is s1 + v1. May be used by JIT code.
     */
    @SuppressWarnings("unused")
    private String(String s1, int v1) {
        if (s1 == null) {
            s1 = "null"; //$NON-NLS-1$
        }
        String s2 = String.valueOf(v1);
        int len = s1.count + s2.count;
        value = new char[len];
        offset = 0;
        System.arraycopy(s1.value, s1.offset, value, 0, s1.count);
        System.arraycopy(s2.value, s2.offset, value, s1.count, s2.count);
        count = len;
    }

    /**
     * Answers the character at the specified offset in this String.
     *
     * @param index
     *            the zero-based index in this string
     * @return the character at the index
     *
     * @throws IndexOutOfBoundsException
     *             when <code>index < 0</code> or
     *             <code>index >= length()</code>
     */
    public char charAt(int index) {
        if (0 <= index && index < count) {
            return value[offset + index];
        }
        throw new StringIndexOutOfBoundsException();
    }

    /**
     * Compares the specified String to this String using the Unicode values of
     * the characters. Answer 0 if the strings contain the same characters in
     * the same order. Answer a negative integer if the first non-equal
     * character in this String has a Unicode value which is less than the
     * Unicode value of the character at the same position in the specified
     * string, or if this String is a prefix of the specified string. Answer a
     * positive integer if the first non-equal character in this String has a
     * Unicode value which is greater than the Unicode value of the character at
     * the same position in the specified string, or if the specified String is
     * a prefix of the this String.
     *
     * @param string
     *            the string to compare
     * @return 0 if the strings are equal, a negative integer if this String is
     *         before the specified String, or a positive integer if this String
     *         is after the specified String
     *
     * @throws NullPointerException
     *             when string is null
     */
    public int compareTo(String string) {
        // Code adapted from K&R, pg 101
        int o1 = offset, o2 = string.offset, result;
        int end = offset + (count < string.count ? count : string.count);
        char[] target = string.value;
        while (o1 < end) {
            if ((result = value[o1++] - target[o2++]) != 0) {
                return result;
            }
        }
        return count - string.count;
    }

    /**
     * Compare the receiver to the specified String to determine the relative
     * ordering when the case of the characters is ignored.
     *
     * @param string
     *            a String
     * @return an int < 0 if this String is less than the specified String, 0 if
     *         they are equal, and > 0 if this String is greater
     */
    public int compareToIgnoreCase(String string) {
        int o1 = offset, o2 = string.offset, result;
        int end = offset + (count < string.count ? count : string.count);
        char c1, c2;
        char[] target = string.value;
        while (o1 < end) {
            if ((c1 = value[o1++]) == (c2 = target[o2++])) {
                continue;
            }
            c1 = Character.toLowerCase(Character.toUpperCase(c1));
            c2 = Character.toLowerCase(Character.toUpperCase(c2));
            if ((result = c1 - c2) != 0) {
                return result;
            }
        }
        return count - string.count;
    }

    /**
     * Concatenates this String and the specified string.
     *
     * @param string
     *            the string to concatenate
     * @return a new String which is the concatenation of this String and the
     *         specified String
     *
     * @throws NullPointerException
     *             if string is null
     */
    public String concat(String string) {
        if (string.count > 0 && count > 0) {
            char[] buffer = new char[count + string.count];
            System.arraycopy(value, offset, buffer, 0, count);
            System.arraycopy(string.value, string.offset, buffer, count,
                    string.count);
            return new String(0, buffer.length, buffer);
        }
        return count == 0 ? string : this;
    }

    /**
     * Creates a new String containing the characters in the specified character
     * array. Modifying the character array after creating the String has no
     * effect on the String.
     *
     * @param data
     *            the array of characters
     * @return the new String
     *
     * @throws NullPointerException
     *             if data is null
     */
    public static String copyValueOf(char[] data) {
        return new String(data, 0, data.length);
    }

    /**
     * Creates a new String containing the specified characters in the character
     * array. Modifying the character array after creating the String has no
     * effect on the String.
     *
     * @param data
     *            the array of characters
     * @param start
     *            the starting offset in the character array
     * @param length
     *            the number of characters to use
     * @return the new String
     *
     * @throws IndexOutOfBoundsException
     *             if <code>length < 0, start < 0</code> or
     *             <code>start + length > data.length</code>
     * @throws NullPointerException
     *             if data is null
     */
    public static String copyValueOf(char[] data, int start, int length) {
        return new String(data, start, length);
    }

    private Charset defaultCharset() {
        if (DefaultCharset == null) {
            String encoding = AccessController
                    .doPrivileged(new PriviAction<String>(
                            "file.encoding", "ISO8859_1")); //$NON-NLS-1$ //$NON-NLS-2$
            // calling System.getProperty() may cause DefaultCharset to be
            // initialized
            try {
                DefaultCharset = Charset.forName(encoding);
            } catch (IllegalCharsetNameException e) {
                // Ignored
            } catch (UnsupportedCharsetException e) {
                // Ignored
            }

            if (DefaultCharset == null) {
                DefaultCharset = Charset.forName("ISO-8859-1"); //$NON-NLS-1$
            }
        }
        return DefaultCharset;
    }

    /**
     * Compares the specified string to this String to determine if the
     * specified string is a suffix.
     *
     * @param suffix
     *            the string to look for
     * @return true when the specified string is a suffix of this String, false
     *         otherwise
     *
     * @throws NullPointerException
     *             if suffix is null
     */
    public boolean endsWith(String suffix) {
        return regionMatches(count - suffix.count, suffix, 0, suffix.count);
    }

    /**
     * Compares the specified object to this String and answer if they are
     * equal. The object must be an instance of String with the same characters
     * in the same order.
     *
     * @param object
     *            the object to compare
     * @return true if the specified object is equal to this String, false
     *         otherwise
     *
     * @see #hashCode
     */
    @Override
    public boolean equals(Object object) {
        if (object == this) {
            return true;
        }
        if (object instanceof String) {
            String s = (String) object;
            if (count != s.count
                    || (hashCode != s.hashCode && hashCode != 0 && s.hashCode != 0)) {
                return false;
            }
            return regionMatches(0, s, 0, count);
        }
        return false;
    }

    /**
     * Compares the specified String to this String ignoring the case of the
     * characters and answer if they are equal.
     *
     * @param string
     *            the string to compare
     * @return true if the specified string is equal to this String, false
     *         otherwise
     */
    public boolean equalsIgnoreCase(String string) {
        if (string == this) {
            return true;
        }
        if (string == null || count != string.count) {
            return false;
        }

        int o1 = offset, o2 = string.offset;
        int end = offset + count;
        char c1, c2;
        char[] target = string.value;
        while (o1 < end) {
            if ((c1 = value[o1++]) != (c2 = target[o2++])
                    && Character.toUpperCase(c1) != Character.toUpperCase(c2)
                    // Required for unicode that we test both cases
                    && Character.toLowerCase(c1) != Character.toLowerCase(c2)) {
                return false;
            }
        }
        return true;
    }

    /**
     * Converts this String to a byte encoding using the default encoding as
     * specified by the file.encoding system property. If the system property is
     * not defined, the default encoding is ISO8859_1 (ISO-Latin-1). If 8859-1
     * is not available, an ASCII encoding is used.
     *
     * @return the byte array encoding of this String
     *
     * @see String
     */
    public byte[] getBytes() {
        ByteBuffer buffer = defaultCharset().encode(
                CharBuffer.wrap(this.value, this.offset, this.count));
        byte[] bytes = new byte[buffer.limit()];
        buffer.get(bytes);
        return bytes;
    }

    /**
     * Converts this String to a byte array, ignoring the high order bits of
     * each character.
     *
     * @param start
     *            the starting offset of characters to copy
     * @param end
     *            the ending offset of characters to copy
     * @param data
     *            the destination byte array
     * @param index
     *            the starting offset in the byte array
     *
     * @throws NullPointerException
     *             when data is null
     * @throws IndexOutOfBoundsException
     *             when
     *             <code>start < 0, end > length(), index < 0, end - start > data.length - index</code>
     *
     * @deprecated Use getBytes() or getBytes(String)
     */
    @Deprecated
    public void getBytes(int start, int end, byte[] data, int index) {
        if (0 <= start && start <= end && end <= count) {
            end += offset;
            try {
                for (int i = offset + start; i < end; i++) {
                    data[index++] = (byte) value[i];
                }
            } catch (ArrayIndexOutOfBoundsException e) {
                throw new StringIndexOutOfBoundsException();
            }
        } else {
            throw new StringIndexOutOfBoundsException();
        }
    }

    /**
     * Converts this String to a byte encoding using the specified encoding.
     *
     * @param encoding
     *            the encoding
     * @return the byte array encoding of this String
     *
     * @throws UnsupportedEncodingException
     *             when the encoding is not supported
     *
     * @see String
     * @see UnsupportedEncodingException
     */
    public byte[] getBytes(String encoding) throws UnsupportedEncodingException {
        ByteBuffer buffer = getCharset(encoding).encode(
                CharBuffer.wrap(this.value, this.offset, this.count));
        byte[] bytes = new byte[buffer.limit()];
        buffer.get(bytes);
        return bytes;
    }

    private Charset getCharset(final String encoding)
            throws UnsupportedEncodingException {
        Charset charset = lastCharset;
        if (charset == null || !encoding.equalsIgnoreCase(charset.name())) {
            try {
                charset = Charset.forName(encoding);
            } catch (IllegalCharsetNameException e) {
                throw (UnsupportedEncodingException) (new UnsupportedEncodingException(
                        encoding).initCause(e));
            } catch (UnsupportedCharsetException e) {
                throw (UnsupportedEncodingException) (new UnsupportedEncodingException(
                        encoding).initCause(e));
            }
            lastCharset = charset;
        }
        return charset;
    }

    /**
     * Copies the specified characters in this String to the character array
     * starting at the specified offset in the character array.
     *
     * @param start
     *            the starting offset of characters to copy
     * @param end
     *            the ending offset of characters to copy
     * @param buffer
     *            the destination character array
     * @param index
     *            the starting offset in the character array
     *
     * @throws IndexOutOfBoundsException
     *             when <code>start < 0, end > length(),
     *        start > end, index < 0, end - start > buffer.length - index</code>
     * @throws NullPointerException
     *             when buffer is null
     */
    public void getChars(int start, int end, char[] buffer, int index) {
        // NOTE last character not copied!
        // Fast range check.
        if (0 <= start && start <= end && end <= count) {
            System.arraycopy(value, start + offset, buffer, index, end - start);
        } else {
            throw new StringIndexOutOfBoundsException();
        }
    }

    /**
     * Answers an integer hash code for the receiver. Objects which are equal
     * answer the same value for this method.
     *
     * @return the receiver's hash
     *
     * @see #equals
     */
    @Override
    public int hashCode() {
        if (hashCode == 0) {
            int hash = 0, multiplier = 1;
            for (int i = offset + count - 1; i >= offset; i--) {
                hash += value[i] * multiplier;
                int shifted = multiplier << 5;
                multiplier = shifted - multiplier;
            }
            hashCode = hash;
        }
        return hashCode;
    }

    /**
     * Searches in this String for the first index of the specified character.
     * The search for the character starts at the beginning and moves towards
     * the end of this String.
     *
     * @param c
     *            the character to find
     * @return the index in this String of the specified character, -1 if the
     *         character isn't found
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int indexOf(int c) {
        return indexOf(c, 0);
    }

    /**
     * Searches in this String for the index of the specified character. The
     * search for the character starts at the specified offset and moves towards
     * the end of this String.
     *
     * @param c
     *            the character to find
     * @param start
     *            the starting offset
     * @return the index in this String of the specified character, -1 if the
     *         character isn't found
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int indexOf(int c, int start) {
        if (start < count) {
            if (start < 0) {
                start = 0;
            }
            for (int i = offset + start; i < offset + count; i++) {
                if (value[i] == c) {
                    return i - offset;
                }
            }
        }
        return -1;
    }

    /**
     * Searches in this String for the first index of the specified string. The
     * search for the string starts at the beginning and moves towards the end
     * of this String.
     *
     * @param string
     *            the string to find
     * @return the index in this String of the specified string, -1 if the
     *         string isn't found
     *
     * @throws NullPointerException
     *             when string is null
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int indexOf(String string) {
        return indexOf(string, 0);
    }

    /**
     * Searches in this String for the index of the specified string. The search
     * for the string starts at the specified offset and moves towards the end
     * of this String.
     *
     * @param subString
     *            the string to find
     * @param start
     *            the starting offset
     * @return the index in this String of the specified string, -1 if the
     *         string isn't found
     *
     * @throws NullPointerException
     *             when string is null
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int indexOf(String subString, int start) {
        if (start < 0) {
            start = 0;
        }
        int subCount = subString.count;
        if (subCount > 0) {
            if (subCount + start > count) {
                return -1;
            }
            char[] target = subString.value;
            int subOffset = subString.offset;
            char firstChar = target[subOffset];
            int end = subOffset + subCount;
            while (true) {
                int i = indexOf(firstChar, start);
                if (i == -1 || subCount + i > count) {
                    return -1; // handles subCount > count || start >= count
                }
                int o1 = offset + i, o2 = subOffset;
                while (++o2 < end && value[++o1] == target[o2]) {
                    // Intentionally empty
                }
                if (o2 == end) {
                    return i;
                }
                start = i + 1;
            }
        }
        return start < count ? start : count;
    }

    /**
     * Searches an internal table of strings for a string equal to this String.
     * If the string is not in the table, it is added. Answers the string
     * contained in the table which is equal to this String. The same string
     * object is always answered for strings which are equal.
     *
     * @return the interned string equal to this String
     */
    public String intern() {
        return VM.intern(this);
    }

    /**
     * Searches in this String for the last index of the specified character.
     * The search for the character starts at the end and moves towards the
     * beginning of this String.
     *
     * @param c
     *            the character to find
     * @return the index in this String of the specified character, -1 if the
     *         character isn't found
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int lastIndexOf(int c) {
        return lastIndexOf(c, count - 1);
    }

    /**
     * Searches in this String for the index of the specified character. The
     * search for the character starts at the specified offset and moves towards
     * the beginning of this String.
     *
     * @param c
     *            the character to find
     * @param start
     *            the starting offset
     * @return the index in this String of the specified character, -1 if the
     *         character isn't found
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int lastIndexOf(int c, int start) {
        if (start >= 0) {
            if (start >= count) {
                start = count - 1;
            }
            for (int i = offset + start; i >= offset; --i) {
                if (value[i] == c) {
                    return i - offset;
                }
            }
        }
        return -1;
    }

    /**
     * Searches in this String for the last index of the specified string. The
     * search for the string starts at the end and moves towards the beginning
     * of this String.
     *
     * @param string
     *            the string to find
     * @return the index in this String of the specified string, -1 if the
     *         string isn't found
     *
     * @throws NullPointerException
     *             when string is null
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int lastIndexOf(String string) {
        // Use count instead of count - 1 so lastIndexOf("") answers count
        return lastIndexOf(string, count);
    }

    /**
     * Searches in this String for the index of the specified string. The search
     * for the string starts at the specified offset and moves towards the
     * beginning of this String.
     *
     * @param subString
     *            the string to find
     * @param start
     *            the starting offset
     * @return the index in this String of the specified string, -1 if the
     *         string isn't found
     *
     * @throws NullPointerException
     *             when string is null
     *
     * @see #lastIndexOf(int)
     * @see #lastIndexOf(int, int)
     * @see #lastIndexOf(String)
     * @see #lastIndexOf(String, int)
     */
    public int lastIndexOf(String subString, int start) {
        int subCount = subString.count;
        if (subCount <= count && start >= 0) {
            if (subCount > 0) {
                if (start > count - subCount) {
                    start = count - subCount;
                }
                // count and subCount are both >= 1
                char[] target = subString.value;
                int subOffset = subString.offset;
                char firstChar = target[subOffset];
                int end = subOffset + subCount;
                while (true) {
                    int i = lastIndexOf(firstChar, start);
                    if (i == -1) {
                        return -1;
                    }
                    int o1 = offset + i, o2 = subOffset;
                    while (++o2 < end && value[++o1] == target[o2]) {
                        // Intentionally empty
                    }
                    if (o2 == end) {
                        return i;
                    }
                    start = i - 1;
                }
            }
            return start < count ? start : count;
        }
        return -1;
    }

    /**
     * Answers the size of this String.
     *
     * @return the number of characters in this String
     */
    public int length() {
        return count;
    }

    /**
     * Compares the specified string to this String and compares the specified
     * range of characters to determine if they are the same.
     *
     * @param thisStart
     *            the starting offset in this String
     * @param string
     *            the string to compare
     * @param start
     *            the starting offset in string
     * @param length
     *            the number of characters to compare
     * @return true if the ranges of characters is equal, false otherwise
     *
     * @throws NullPointerException
     *             when string is null
     */
    public boolean regionMatches(int thisStart, String string, int start,
            int length) {
        if (string == null) {
            throw new NullPointerException();
        }
        if (start < 0 || string.count - start < length) {
            return false;
        }
        if (thisStart < 0 || count - thisStart < length) {
            return false;
        }
        if (length <= 0) {
            return true;
        }
        int o1 = offset + thisStart, o2 = string.offset + start;
        for (int i = 0; i < length; ++i) {
            if (value[o1 + i] != string.value[o2 + i]) {
                return false;
            }
        }
        return true;
    }

    /**
     * Compares the specified string to this String and compares the specified
     * range of characters to determine if they are the same. When ignoreCase is
     * true, the case of the characters is ignored during the comparison.
     *
     * @param ignoreCase
     *            specifies if case should be ignored
     * @param thisStart
     *            the starting offset in this String
     * @param string
     *            the string to compare
     * @param start
     *            the starting offset in string
     * @param length
     *            the number of characters to compare
     * @return true if the ranges of characters is equal, false otherwise
     *
     * @throws NullPointerException
     *             when string is null
     */
    public boolean regionMatches(boolean ignoreCase, int thisStart,
            String string, int start, int length) {
        if (!ignoreCase) {
            return regionMatches(thisStart, string, start, length);
        }

        if (string != null) {
            if (thisStart < 0 || length > count - thisStart) {
                return false;
            }
            if (start < 0 || length > string.count - start) {
                return false;
            }

            thisStart += offset;
            start += string.offset;
            int end = thisStart + length;
            char c1, c2;
            char[] target = string.value;
            while (thisStart < end) {
                if ((c1 = value[thisStart++]) != (c2 = target[start++])
                        && Character.toUpperCase(c1) != Character
                                .toUpperCase(c2)
                        // Required for unicode that we test both cases
                        && Character.toLowerCase(c1) != Character
                                .toLowerCase(c2)) {
                    return false;
                }
            }
            return true;
        }
        throw new NullPointerException();
    }

    /**
     * Copies this String replacing occurrences of the specified character with
     * another character.
     *
     * @param oldChar
     *            the character to replace
     * @param newChar
     *            the replacement character
     * @return a new String with occurrences of oldChar replaced by newChar
     */
    public String replace(char oldChar, char newChar) {
        int index = indexOf(oldChar, 0);
        if (index == -1) {
            return this;
        }

        char[] buffer = new char[count];
        System.arraycopy(value, offset, buffer, 0, count);
        do {
            buffer[index++] = newChar;
        } while ((index = indexOf(oldChar, index)) != -1);
        return new String(0, count, buffer);
    }
   
    /**
     * Copies this String replacing occurrences of the specified
     * target sequence with another sequence.
     * The string is processed from the beginning to the end.
     *
     * @param target
     *            the sequence to replace
     * @param replacement
     *            the replacement sequence
     * @return the resulting String
     *
     * @throws NullPointerException if either of the arguments
     * is <code>null</code>
     */
    public String replace(CharSequence target, CharSequence replacement) {
        if (target == null) {
            throw new NullPointerException("target should not be null");
        }
        if (replacement == null) {
            throw new NullPointerException("replacement should not be null");
        }
        String ts = target.toString();
        int index = indexOf(ts, 0);

        if (index == -1)
            return this;

        String rs = replacement.toString();
        StringBuilder buffer = new StringBuilder(count);
        int tl = target.length();
        int tail = 0;
        do {
            buffer.append(value, offset + tail, index - tail);
            buffer.append(rs);
            tail = index + tl;
        } while ((index = indexOf(ts, tail)) != -1);
        //append trailing chars
        buffer.append(value, offset + tail, count - tail);

        return buffer.toString();
    }

    /**
     * Compares the specified string to this String to determine if the
     * specified string is a prefix.
     *
     * @param prefix
     *            the string to look for
     * @return true when the specified string is a prefix of this String, false
     *         otherwise
     *
     * @throws NullPointerException
     *             when prefix is null
     */
    public boolean startsWith(String prefix) {
        return startsWith(prefix, 0);
    }

    /**
     * Compares the specified string to this String, starting at the specified
     * offset, to determine if the specified string is a prefix.
     *
     * @param prefix
     *            the string to look for
     * @param start
     *            the starting offset
     * @return true when the specified string occurs in this String at the
     *         specified offset, false otherwise
     *
     * @throws NullPointerException
     *             when prefix is null
     */
    public boolean startsWith(String prefix, int start) {
        return regionMatches(start, prefix, 0, prefix.count);
    }

    /**
     * Copies a range of characters into a new String.
     *
     * @param start
     *            the offset of the first character
     * @return a new String containing the characters from start to the end of
     *         the string
     *
     * @throws IndexOutOfBoundsException
     *             when <code>start < 0</code> or
     *             <code>start > length()</code>
     */
    public String substring(int start) {
        if (start == 0) {
            return this;
        }
        if (0 <= start && start <= count) {
            return new String(offset + start, count - start, value);
        }
        throw new StringIndexOutOfBoundsException(start);
    }

    /**
     * Copies a range of characters into a new String.
     *
     * @param start
     *            the offset of the first character
     * @param end
     *            the offset one past the last character
     * @return a new String containing the characters from start to end - 1
     *
     * @throws IndexOutOfBoundsException
     *             when <code>start < 0, start > end</code> or
     *             <code>end > length()</code>
     */
    public String substring(int start, int end) {
        if (start == 0 && end == count) {
            return this;
        }
        // NOTE last character not copied!
        // Fast range check.
        if (0 <= start && start <= end && end <= count) {
            return new String(offset + start, end - start, value);
        }
        throw new StringIndexOutOfBoundsException();
    }

    /**
     * Copies the characters in this String to a character array.
     *
     * @return a character array containing the characters of this String
     */
    public char[] toCharArray() {
        char[] buffer = new char[count];
        System.arraycopy(value, offset, buffer, 0, count);
        return buffer;
    }

    /**
     * Converts the characters in this String to lowercase, using the default
     * Locale.
     *
     * @return a new String containing the lowercase characters equivalent to
     *         the characters in this String
     */
    public String toLowerCase() {
        return toLowerCase(Locale.getDefault());
    }

    /**
     * Converts the characters in this String to lowercase, using the specified
     * Locale.
     *
     * @param locale
     *            the Locale
     * @return a new String containing the lowercase characters equivalent to
     *         the characters in this String
     */
    public String toLowerCase(Locale locale) {
        for (int o = offset, end = offset + count; o < end; o++) {
            char ch = value[o];
            if (ch != Character.toLowerCase(ch)) {
                char[] buffer = new char[count];
                int i = o - offset;
                // Not worth checking for i == 0 case
                System.arraycopy(value, offset, buffer, 0, i);
                // Turkish
                if (!"tr".equals(locale.getLanguage())) { //$NON-NLS-1$
                    while (i < count) {
                        buffer[i++] = Character.toLowerCase(value[o++]);
                    }
                } else {
                    while (i < count) {
                        buffer[i++] = (ch = value[o++]) != 0x49 ? Character
                                .toLowerCase(ch) : (char) 0x131;
                    }
                }
                return new String(0, count, buffer);
            }
        }
        return this;
    }

    /**
     * Answers a string containing a concise, human-readable description of the
     * receiver.
     *
     * @return this String
     */
    @Override
    public String toString() {
        return this;
    }

    /**
     * Converts the characters in this String to uppercase, using the default
     * Locale.
     *
     * @return a new String containing the uppercase characters equivalent to
     *         the characters in this String
     */
    public String toUpperCase() {
        return toUpperCase(Locale.getDefault());
    }

    private static final char[] upperValues = "SS\u0000\u02bcN\u0000J\u030c\u0000\u0399\u0308\u0301\u03a5\u0308\u0301\u0535\u0552\u0000H\u0331\u0000T\u0308\u0000W\u030a\u0000Y\u030a\u0000A\u02be\u0000\u03a5\u0313\u0000\u03a5\u0313\u0300\u03a5\u0313\u0301\u03a5\u0313\u0342\u1f08\u0399\u0000\u1f09\u0399\u0000\u1f0a\u0399\u0000\u1f0b\u0399\u0000\u1f0c\u0399\u0000\u1f0d\u0399\u0000\u1f0e\u0399\u0000\u1f0f\u0399\u0000\u1f08\u0399\u0000\u1f09\u0399\u0000\u1f0a\u0399\u0000\u1f0b\u0399\u0000\u1f0c\u0399\u0000\u1f0d\u0399\u0000\u1f0e\u0399\u0000\u1f0f\u0399\u0000\u1f28\u0399\u0000\u1f29\u0399\u0000\u1f2a\u0399\u0000\u1f2b\u0399\u0000\u1f2c\u0399\u0000\u1f2d\u0399\u0000\u1f2e\u0399\u0000\u1f2f\u0399\u0000\u1f28\u0399\u0000\u1f29\u0399\u0000\u1f2a\u0399\u0000\u1f2b\u0399\u0000\u1f2c\u0399\u0000\u1f2d\u0399\u0000\u1f2e\u0399\u0000\u1f2f\u0399\u0000\u1f68\u0399\u0000\u1f69\u0399\u0000\u1f6a\u0399\u0000\u1f6b\u0399\u0000\u1f6c\u0399\u0000\u1f6d\u0399\u0000\u1f6e\u0399\u0000\u1f6f\u0399\u0000\u1f68\u0399\u0000\u1f69\u0399\u0000\u1f6a\u0399\u0000\u1f6b\u0399\u0000\u1f6c\u0399\u0000\u1f6d\u0399\u0000\u1f6e\u0399\u0000\u1f6f\u0399\u0000\u1fba\u0399\u0000\u0391\u0399\u0000\u0386\u0399\u0000\u0391\u0342\u0000\u0391\u0342\u0399\u0391\u0399\u0000\u1fca\u0399\u0000\u0397\u0399\u0000\u0389\u0399\u0000\u0397\u0342\u0000\u0397\u0342\u0399\u0397\u0399\u0000\u0399\u0308\u0300\u0399\u0308\u0301\u0399\u0342\u0000\u0399\u0308\u0342\u03a5\u0308\u0300\u03a5\u0308\u0301\u03a1\u0313\u0000\u03a5\u0342\u0000\u03a5\u0308\u0342\u1ffa\u0399\u0000\u03a9\u0399\u0000\u038f\u0399\u0000\u03a9\u0342\u0000\u03a9\u0342\u0399\u03a9\u0399\u0000FF\u0000FI\u0000FL\u0000FFIFFLST\u0000ST\u0000\u0544\u0546\u0000\u0544\u0535\u0000\u0544\u053b\u0000\u054e\u0546\u0000\u0544\u053d\u0000".value; //$NON-NLS-1$

    /**
     * Return the index of the specified character into the upperValues table.
     * The upperValues table contains three entries at each position. These
     * three characters are the upper case conversion. If only two characters
     * are used, the third character in the table is \u0000.
     *
     * @param ch
     *            the char being converted to upper case
     *
     * @return the index into the upperValues table, or -1
     */
    private int upperIndex(int ch) {
        int index = -1;
        if (ch >= 0xdf) {
            if (ch <= 0x587) {
                if (ch == 0xdf) {
                    index = 0;
                } else if (ch <= 0x149) {
                    if (ch == 0x149) {
                        index = 1;
                    }
                } else if (ch <= 0x1f0) {
                    if (ch == 0x1f0) {
                        index = 2;
                    }
                } else if (ch <= 0x390) {
                    if (ch == 0x390) {
                        index = 3;
                    }
                } else if (ch <= 0x3b0) {
                    if (ch == 0x3b0) {
                        index = 4;
                    }
                } else if (ch <= 0x587) {
                    if (ch == 0x587) {
                        index = 5;
                    }
                }
            } else if (ch >= 0x1e96) {
                if (ch <= 0x1e9a) {
                    index = 6 + ch - 0x1e96;
                } else if (ch >= 0x1f50 && ch <= 0x1ffc) {
                    index = "\u000b\u0000\f\u0000\r\u0000\u000e\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u000f\u0010\u0011\u0012\u0013\u0014\u0015\u0016\u0017\u0018\u0019\u001a\u001b\u001c\u001d\u001e\u001f !\"#$%&'()*+,-./0123456789:;<=>\u0000\u0000?@A\u0000BC\u0000\u0000\u0000\u0000D\u0000\u0000\u0000\u0000\u0000EFG\u0000HI\u0000\u0000\u0000\u0000J\u0000\u0000\u0000\u0000\u0000KL\u0000\u0000MN\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000OPQ\u0000RS\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000TUV\u0000WX\u0000\u0000\u0000\u0000Y".value[ch - 0x1f50]; //$NON-NLS-1$
                    if (index == 0) {
                        index = -1;
                    }
                } else if (ch >= 0xfb00) {
                    if (ch <= 0xfb06) {
                        index = 90 + ch - 0xfb00;
                    } else if (ch >= 0xfb13 && ch <= 0xfb17) {
                        index = 97 + ch - 0xfb13;
                    }
                }
            }
        }
        return index;
    }

    /**
     * Converts the characters in this String to uppercase, using the specified
     * Locale.
     *
     * @param locale
     *            the Locale
     * @return a new String containing the uppercase characters equivalent to
     *         the characters in this String
     */
    public String toUpperCase(Locale locale) {
        boolean turkish = "tr".equals(locale.getLanguage()); //$NON-NLS-1$
        char[] output = null;
        int i = 0;
        for (int o = offset, end = offset + count; o < end; o++) {
            char ch = value[o];
            int index = upperIndex(ch);
            if (index == -1) {
                if (output != null && i >= output.length) {
                    char[] newoutput = new char[output.length + (count / 6) + 2];
                    System.arraycopy(output, 0, newoutput, 0, output.length);
                    output = newoutput;
                }
                char upch = !turkish ? Character.toUpperCase(ch)
                        : (ch != 0x69 ? Character.toUpperCase(ch)
                                : (char) 0x130);
                if (ch != upch) {
                    if (output == null) {
                        output = new char[count];
                        i = o - offset;
                        System.arraycopy(value, offset, output, 0, i);

                    }
                    output[i++] = upch;
                } else if (output != null) {
                    output[i++] = ch;
                }
            } else {
                int target = index * 3;
                char val3 = upperValues[target + 2];
                if (output == null) {
                    output = new char[count + (count / 6) + 2];
                    i = o - offset;
                    System.arraycopy(value, offset, output, 0, i);
                } else if (i + (val3 == 0 ? 1 : 2) >= output.length) {
                    char[] newoutput = new char[output.length + (count / 6) + 3];
                    System.arraycopy(output, 0, newoutput, 0, output.length);
                    output = newoutput;
                }

                char val = upperValues[target];
                output[i++] = val;
                val = upperValues[target + 1];
                output[i++] = val;
                if (val3 != 0) {
                    output[i++] = val3;
                }
            }
        }
        if (output == null) {
            return this;
        }
        return output.length == i || output.length - i < 8 ? new String(0, i,
                output) : new String(output, 0, i);
    }

    /**
     * Copies this String removing white space characters from the beginning and
     * end of the string.
     *
     * @return a new String with characters <code><= \\u0020</code> removed
     *         from the beginning and the end
     */
    public String trim() {
        int start = offset, last = offset + count - 1;
        int end = last;
        while ((start <= end) && (value[start] <= ' ')) {
            start++;
        }
        while ((end >= start) && (value[end] <= ' ')) {
            end--;
        }
        if (start == offset && end == last) {
            return this;
        }
        return new String(start, end - start + 1, value);
    }

    /**
     * Creates a new String containing the characters in the specified character
     * array. Modifying the character array after creating the String has no
     * effect on the String.
     *
     * @param data
     *            the array of characters
     * @return the new String
     *
     * @throws NullPointerException
     *             when data is null
     */
    public static String valueOf(char[] data) {
        return new String(data, 0, data.length);
    }

    /**
     * Creates a new String containing the specified characters in the character
     * array. Modifying the character array after creating the String has no
     * effect on the String.
     *
     * @param data
     *            the array of characters
     * @param start
     *            the starting offset in the character array
     * @param length
     *            the number of characters to use
     * @return the new String
     *
     * @throws IndexOutOfBoundsException
     *             when <code>length < 0, start < 0</code> or
     *             <code>start + length > data.length</code>
     * @throws NullPointerException
     *             when data is null
     */
    public static String valueOf(char[] data, int start, int length) {
        return new String(data, start, length);
    }

    /**
     * Converts the specified character to its string representation.
     *
     * @param value
     *            the character
     * @return the character converted to a string
     */
    public static String valueOf(char value) {
        String s;
        if (value < 128) {
            s = new String(value, 1, ascii);
        } else {
            s = new String(0, 1, new char[] { value });
        }
        s.hashCode = value;
        return s;
    }

    /**
     * Converts the specified double to its string representation.
     *
     * @param value
     *            the double
     * @return the double converted to a string
     */
    public static String valueOf(double value) {
        return Double.toString(value);
    }

    /**
     * Converts the specified float to its string representation.
     *
     * @param value
     *            the float
     * @return the float converted to a string
     */
    public static String valueOf(float value) {
        return Float.toString(value);
    }

    /**
     * Converts the specified integer to its string representation.
     *
     * @param value
     *            the integer
     * @return the integer converted to a string
     */
    public static String valueOf(int value) {
        return Integer.toString(value);
    }

    /**
     * Converts the specified long to its string representation.
     *
     * @param value
     *            the long
     * @return the long converted to a string
     */
    public static String valueOf(long value) {
        return Long.toString(value);
    }

    /**
     * Converts the specified object to its string representation. If the object
     * is null answer the string <code>"null"</code>, otherwise use
     * <code>toString()</code> to get the string representation.
     *
     * @param value
     *            the object
     * @return the object converted to a string
     */
    public static String valueOf(Object value) {
        return value != null ? value.toString() : "null"; //$NON-NLS-1$
    }

    /**
     * Converts the specified boolean to its string representation. When the
     * boolean is true answer <code>"true"</code>, otherwise answer
     * <code>"false"</code>.
     *
     * @param value
     *            the boolean
     * @return the boolean converted to a string
     */
    public static String valueOf(boolean value) {
        return value ? "true" : "false"; //$NON-NLS-1$ //$NON-NLS-2$
    }

    /**
     * Answers whether the characters in the StringBuffer strbuf are the same as
     * those in this String.
     *
     * @param strbuf
     *            the StringBuffer to compare this String to
     * @return true when the characters in strbuf are identical to those in this
     *         String. If they are not, false will be returned.
     *
     * @throws NullPointerException
     *             when strbuf is null
     *
     * @since 1.4
     */
    public boolean contentEquals(StringBuffer strbuf) {
        synchronized (strbuf) {
            int size = strbuf.length();
            if (count != size) {
                return false;
            }
            return regionMatches(0, new String(0, size, strbuf.getValue()), 0,
                    size);
        }
    }

    /**
     * <p>
     * Compares a <code>CharSequence</code> to this <code>String</code> to
     * determine if their contents are equal.
     * </p>
     *
     * @param cs
     *            The character sequence to compare to.
     * @return <code>true</code> if equal, otherwise <code>false</code>
     * @since 1.5
     */
    public boolean contentEquals(CharSequence cs) {
        if (cs == null) {
            throw new NullPointerException();
        }

        int len = cs.length();

        if (len != count) {
            return false;
        }

        if (len == 0 && count == 0) {
            return true; // since both are empty strings
        }

        return regionMatches(0, cs.toString(), 0, len);
    }

    /**
     * Determines whether a this String matches a given regular expression.
     *
     * @param expr
     *            the regular expression to be matched
     * @return true if the expression matches, otherwise false
     *
     * @throws PatternSyntaxException
     *             if the syntax of the supplied regular expression is not valid
     * @throws NullPointerException
     *             if expr is null
     *
     * @since 1.4
     */
    public boolean matches(String expr) {
        return Pattern.matches(expr, this);
    }

    /**
     * Replace any substrings within this String that match the supplied regular
     * expression expr, with the String substitute.
     *
     * @param expr
     *            the regular expression to match
     * @param substitute
     *            the string to replace the matching substring with
     * @return the new string
     *
     * @throws NullPointerException
     *             if expr is null
     *
     * @since 1.4
     */
    public String replaceAll(String expr, String substitute) {
        return Pattern.compile(expr).matcher(this).replaceAll(substitute);
    }

    /**
     * Replace any substrings within this String that match the supplied regular
     * expression expr, with the String substitute.
     *
     * @param expr
     *            the regular expression to match
     * @param substitute
     *            the string to replace the matching substring with
     * @return the new string
     *
     * @throws NullPointerException
     *             if expr is null
     *
     * @since 1.4
     */
    public String replaceFirst(String expr, String substitute) {
        return Pattern.compile(expr).matcher(this).replaceFirst(substitute);
    }

    /**
     * Replace any substrings within this String that match the supplied regular
     * expression expr, with the String substitute.
     *
     * @param expr
     *            the regular expression to match
     * @return the new string
     *
     * @throws NullPointerException
     *             if expr is null
     *
     * @since 1.4
     */
    public String[] split(String expr) {
        return Pattern.compile(expr).split(this);
    }

    /**
     * Splits this String using the supplied regular expression expr. max
     * controls the number of times that the pattern is applied to the string.
     *
     * @param expr
     *            the regular expression used to divide the string
     * @param max
     *            the number of times to apply the pattern
     * @return an array of Strings created by separating the string along
     *         matches of the regular expression.
     *
     * @throws NullPointerException
     *             if expr is null
     *
     * @since 1.4
     */
    public String[] split(String expr, int max) {
        return Pattern.compile(expr).split(this, max);
    }

    /**
     * Has the same result as the substring function, but is present so that
     * String may implement the CharSequence interface.
     *
     * @param start
     *            the offset the first character
     * @param end
     *            the offset of one past the last character to include
     *
     * @return the subsequence requested
     *
     * @throws IndexOutOfBoundsException
     *             when start or end is less than zero, start is greater than
     *             end, or end is greater than the length of the String.
     *
     * @see java.lang.CharSequence#subSequence(int, int)
     *
     * @since 1.4
     */
    public CharSequence subSequence(int start, int end) {
        return substring(start, end);
    }

    /**
     * <p>
     * Retrieves the Unicode code point value at the <code>index</code>.
     * </p>
     *
     * @param index
     *            The index to the <code>char</code> code unit within this
     *            object.
     * @return The Unicode code point value.
     * @throws IndexOutOfBoundsException
     *             if <code>index</code> is negative or greater than or equal
     *             to {@link #length()}.
     * @see Character
     * @see Character#codePointAt(char[], int, int)
     * @since 1.5
     */
    public int codePointAt(int index) {
        if (index < 0 || index >= count) {
            throw new IndexOutOfBoundsException();
        }
        int s = index + offset;
        return Character.codePointAt(value, s, offset + count);
    }

    /**
     * <p>
     * Retrieves the Unicode code point value that precedes the
     * <code>index</code>.
     * </p>
     *
     * @param index
     *            The index to the <code>char</code> code unit within this
     *            object.
     * @return The Unicode code point value.
     * @throws IndexOutOfBoundsException
     *             if <code>index</code> is less than 1 or greater than
     *             {@link #length()}.
     * @see Character
     * @see Character#codePointBefore(char[], int, int)
     * @since 1.5
     */
    public int codePointBefore(int index) {
        if (index < 1 || index > count) {
            throw new IndexOutOfBoundsException();
        }
        int s = index + offset;
        return Character.codePointBefore(value, s);
    }

    /**
     * <p>
     * Calculates the number of Unicode code points between
     * <code>beginIndex</code> and <code>endIndex</code>.
     * </p>
     *
     * @param beginIndex
     *            The inclusive beginning index of the subsequence.
     * @param endIndex
     *            The exclusive end index of the subsequence.
     * @return The number of Unicode code points in the subsequence.
     * @throws IndexOutOfBoundsException
     *             if <code>beginIndex</code> is negative or greater than
     *             <code>endIndex</code> or <code>endIndex</code> is greater
     *             than {@link #length()}.
     * @since 1.5
     */
    public int codePointCount(int beginIndex, int endIndex) {
        if (beginIndex < 0 || endIndex > count || beginIndex > endIndex) {
            throw new IndexOutOfBoundsException();
        }
        int s = beginIndex + offset;
        return Character.codePointCount(value, s, endIndex - beginIndex);
    }

    /**
     * <p>
     * Determines if this <code>String</code> contains the sequence of
     * characters in the <code>CharSequence</code> passed.
     * </p>
     *
     * @param cs
     *            The character sequence to search for.
     * @return <code>true</code> if the sequence of characters are contained
     *         in this object; otherwise <code>false</code>
     * @since 1.5
     */
    public boolean contains(CharSequence cs) {
        if (cs == null) {
            throw new NullPointerException();
        }
        return indexOf(cs.toString()) >= 0;
    }

    /**
     * <p>
     * Returns the index within this object that is offset from
     * <code>index</code> by <code>codePointOffset</code> code points.
     * </p>
     *
     * @param index
     *            The index within this object to calculate the offset from.
     * @param codePointOffset
     *            The number of code points to count.
     * @return The index within this object that is the offset.
     * @throws IndexOutOfBoundsException
     *             if <code>index</code> is negative or greater than
     *             {@link #length()} or if there aren't enough code points
     *             before or after <code>index</code> to match
     *             <code>codePointOffset</code>.
     * @since 1.5
     */
    public int offsetByCodePoints(int index, int codePointOffset) {
        int s = index + offset;
        int r = Character.offsetByCodePoints(value, offset, count, s,
                codePointOffset);
        return r - offset;
    }

    /**
     * Returns a printf-style formatted string, using the supplied format and
     * arguments. This function is a shortcut to
     * <code>format(Locale.getDefault(), format, args)</code>.
     *
     * @param format
     *            a format string
     * @param args
     *            arguments to replace format specifiers, may be none
     * @throws NullPointerException
     *             if the format is null
     * @throws IllegalArgumentException
     *             if the format is invalid
     * @return The formatted string
     * @since 1.5
     * @see java.util.Formatter
     */
    public static String format(String format, Object... args) {
        return format(Locale.getDefault(), format, args);
    }

    /**
     * Returns a printf-style formatted string, using the supplied format and
     * arguments, accordingly to the specified locale.
     *
     * @param loc
     *            the locale to apply; <code>null</code> value means no
     *            localization
     * @param format
     *            a format string
     * @param args
     *            arguments to replace format specifiers, may be none
     * @throws NullPointerException
     *             if the format is null
     * @throws IllegalArgumentException
     *             if the format is invalid
     * @return The formatted string
     * @since 1.5
     * @see java.util.Formatter
     */
    public static String format(Locale loc, String format, Object... args) {
        if (format == null) {
            throw new NullPointerException("null format argument");
        }
        int bufferSize = format.length()
                + (args == null ? 0 : args.length * 10);
        Formatter f = new Formatter(new StringBuilder(bufferSize), loc);
        return f.format(format, args).toString();
    }

    /*
     * An implementation of a String.indexOf that is supposed to perform
     * substantially better than the default algorithm if the "needle" (the
     * subString being searched for) is a constant string.
     *
     * For example, a JIT, upon encoutering a call to String.indexOf(String),
     * where the needle is a constant string, may compute the values cache, md2
     * and lastChar, and change the call to the following method.
     */
    @SuppressWarnings("unused")
    private static int indexOf(String haystackString, String needleString,
            int cache, int md2, char lastChar) {
        char[] haystack = haystackString.value;
        int haystackOffset = haystackString.offset;
        int haystackLength = haystackString.count;
        char[] needle = needleString.value;
        int needleOffset = needleString.offset;
        int needleLength = needleString.count;
        int needleLengthMinus1 = needleLength - 1;
        int haystackEnd = haystackOffset + haystackLength;
        outer_loop: for (int i = haystackOffset + needleLengthMinus1; i < haystackEnd;) {
            if (lastChar == haystack[i]) {
                for (int j = 0; j < needleLengthMinus1; ++j) {
                    if (needle[j + needleOffset] != haystack[i + j
                            - needleLengthMinus1]) {
                        int skip = 1;
                        if ((cache & (1 << haystack[i])) == 0) {
                            skip += j;
                        }
                        i += Math.max(md2, skip);
                        continue outer_loop;
                    }
                }
                return i - needleLengthMinus1 - haystackOffset;
            }

            if ((cache & (1 << haystack[i])) == 0) {
                i += needleLengthMinus1;
            }
            i++;
        }
        return -1;
    }

    /*
     * Returns the character array for this String.
     */
    char[] getValue() {
        return value;
    }
}
TOP

Related Classes of java.lang.String$CaseInsensitiveComparator

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.