mirror of
https://github.com/corda/corda.git
synced 2025-01-22 20:38:05 +00:00
a8bed52097
such as %s, %d, %x... also width, left justify, zerofill flags are implemented. Many of the other formats do not work, see the comments in avian.FormatString javadoc and look for FIXME and TODO items for more information on features that are known to not work correctly.
699 lines
17 KiB
Java
699 lines
17 KiB
Java
/* Copyright (c) 2008-2015, Avian Contributors
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
for any purpose with or without fee is hereby granted, provided
|
|
that the above copyright notice and this permission notice appear
|
|
in all copies.
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
details. */
|
|
|
|
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 avian.Iso88591;
|
|
import avian.Utf8;
|
|
|
|
public final class String
|
|
implements Comparable<String>, CharSequence, Serializable
|
|
{
|
|
private static final String UTF_8_ENCODING = "UTF-8";
|
|
private static final String ISO_8859_1_ENCODING = "ISO-8859-1";
|
|
private static final String LATIN_1_ENCODING = "LATIN-1";
|
|
private static final String DEFAULT_ENCODING = UTF_8_ENCODING;
|
|
|
|
public static Comparator<String> CASE_INSENSITIVE_ORDER
|
|
= new Comparator<String>() {
|
|
@Override
|
|
public int compare(String a, String b) {
|
|
return a.compareToIgnoreCase(b);
|
|
}
|
|
};
|
|
|
|
private final Object data;
|
|
private final int offset;
|
|
private final int length;
|
|
private int hashCode;
|
|
|
|
public String() {
|
|
this(new char[0], 0, 0);
|
|
}
|
|
|
|
public String(char[] data, int offset, int length, boolean copy) {
|
|
this((Object) data, offset, length, copy);
|
|
}
|
|
|
|
public String(char[] data, int offset, int length) {
|
|
this(data, offset, length, true);
|
|
}
|
|
|
|
public String(char[] data) {
|
|
this(data, 0, data.length);
|
|
}
|
|
|
|
public String(byte bytes[], int offset, int length, String charsetName)
|
|
throws UnsupportedEncodingException
|
|
{
|
|
this(bytes, offset, length);
|
|
if (! (charsetName.equalsIgnoreCase(UTF_8_ENCODING)
|
|
|| charsetName.equalsIgnoreCase(ISO_8859_1_ENCODING)))
|
|
{
|
|
throw new UnsupportedEncodingException(charsetName);
|
|
}
|
|
}
|
|
|
|
public String(byte[] data, int offset, int length, boolean copy) {
|
|
this((Object) data, offset, length, copy);
|
|
}
|
|
|
|
public String(byte[] data, int offset, int length) {
|
|
this(data, offset, length, true);
|
|
}
|
|
|
|
public String(byte[] data) {
|
|
this(data, 0, data.length);
|
|
}
|
|
|
|
public String(String s) {
|
|
this(s.toCharArray());
|
|
}
|
|
|
|
public String(byte[] data, String charset)
|
|
throws UnsupportedEncodingException
|
|
{
|
|
this(data, 0, data.length, charset);
|
|
}
|
|
|
|
public String(byte bytes[], int highByte, int offset, int length) {
|
|
if (offset < 0 )
|
|
throw new StringIndexOutOfBoundsException(offset);
|
|
else if (offset + length > bytes.length)
|
|
throw new StringIndexOutOfBoundsException(offset + length);
|
|
else if (length < 0)
|
|
throw new StringIndexOutOfBoundsException(length);
|
|
|
|
char[] c = new char[length];
|
|
int mask = highByte << 8;
|
|
for (int i = 0; i < length; ++i) {
|
|
c[i] = (char) ((bytes[offset + i] & 0xFF) | mask);
|
|
}
|
|
|
|
this.data = c;
|
|
this.offset = 0;
|
|
this.length = length;
|
|
}
|
|
|
|
private String(Object data, int offset, int length, boolean copy) {
|
|
int l;
|
|
if (data instanceof char[]) {
|
|
l = ((char[]) data).length;
|
|
} else {
|
|
l = ((byte[]) data).length;
|
|
}
|
|
|
|
if (offset < 0 )
|
|
throw new StringIndexOutOfBoundsException(offset);
|
|
else if (offset + length > l)
|
|
throw new StringIndexOutOfBoundsException(offset + length);
|
|
else if (length < 0)
|
|
throw new StringIndexOutOfBoundsException(length);
|
|
|
|
if(!copy && Utf8.test(data)) copy = true;
|
|
|
|
if (copy) {
|
|
Object c;
|
|
if (data instanceof char[]) {
|
|
c = new char[length];
|
|
System.arraycopy(data, offset, c, 0, length);
|
|
} else {
|
|
c = Utf8.decode((byte[])data, offset, length);
|
|
if(c instanceof char[]) length = ((char[])c).length;
|
|
if (c == null) {
|
|
throw new RuntimeException
|
|
("unable to parse \"" + new String(data, offset, length, false)
|
|
+ "\"");
|
|
}
|
|
}
|
|
|
|
this.data = c;
|
|
this.offset = 0;
|
|
this.length = length;
|
|
} else {
|
|
this.data = data;
|
|
this.offset = offset;
|
|
this.length = length;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return this;
|
|
}
|
|
|
|
@Override
|
|
public int length() {
|
|
return length;
|
|
}
|
|
|
|
@Override
|
|
public int hashCode() {
|
|
if (hashCode == 0) {
|
|
int h = 0;
|
|
for (int i = 0; i < length; ++i) h = (h * 31) + charAt(i);
|
|
hashCode = h;
|
|
}
|
|
return hashCode;
|
|
}
|
|
|
|
@Override
|
|
public boolean equals(Object o) {
|
|
if (this == o) {
|
|
return true;
|
|
} else if (o instanceof String) {
|
|
String s = (String) o;
|
|
return s.length == length && compareTo(s) == 0;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean equalsIgnoreCase(String s) {
|
|
if (this == s) {
|
|
return true;
|
|
} else {
|
|
return s != null && s.length == length && compareToIgnoreCase(s) == 0;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public int compareTo(String s) {
|
|
if (this == s) return 0;
|
|
|
|
int idx = 0;
|
|
int result;
|
|
|
|
int end = (length < s.length ? length : s.length);
|
|
|
|
while (idx < end) {
|
|
if ((result = charAt(idx) - s.charAt(idx)) != 0) {
|
|
return result;
|
|
}
|
|
idx++;
|
|
}
|
|
return length - s.length;
|
|
}
|
|
|
|
public int compareToIgnoreCase(String s) {
|
|
if (this == s) return 0;
|
|
|
|
int idx = 0;
|
|
int result;
|
|
|
|
int end = (length < s.length ? length : s.length);
|
|
|
|
while (idx < end) {
|
|
if ((result =
|
|
Character.toLowerCase(charAt(idx)) -
|
|
Character.toLowerCase(s.charAt(idx))) != 0) {
|
|
return result;
|
|
}
|
|
idx++;
|
|
}
|
|
return length - s.length;
|
|
}
|
|
|
|
public String trim() {
|
|
int start = -1;
|
|
for (int i = 0; i < length; ++i) {
|
|
char c = charAt(i);
|
|
if (start == -1 && ! Character.isWhitespace(c)) {
|
|
start = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
int end = -1;
|
|
for (int i = length - 1; i >= 0; --i) {
|
|
char c = charAt(i);
|
|
if (end == -1 && ! Character.isWhitespace(c)) {
|
|
end = i + 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (start >= end) {
|
|
return "";
|
|
} else {
|
|
return substring(start, end);
|
|
}
|
|
}
|
|
|
|
public String toLowerCase() {
|
|
for (int j = 0; j < length; ++j) {
|
|
char ch = charAt(j);
|
|
if (Character.toLowerCase(ch) != ch) {
|
|
char[] b = new char[length];
|
|
for (int i = 0; i < length; ++i) {
|
|
b[i] = Character.toLowerCase(charAt(i));
|
|
}
|
|
return new String(b, 0, length, false);
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
|
|
public String toUpperCase() {
|
|
for (int j = 0; j < length; ++j) {
|
|
char ch = charAt(j);
|
|
if (Character.toUpperCase(ch) != ch) {
|
|
char[] b = new char[length];
|
|
for (int i = 0; i < length; ++i) {
|
|
b[i] = Character.toUpperCase(charAt(i));
|
|
}
|
|
return new String(b, 0, length, false);
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
|
|
public int indexOf(int c) {
|
|
return indexOf(c, 0);
|
|
}
|
|
|
|
public int indexOf(int c, int start) {
|
|
for (int i = start; i < length; ++i) {
|
|
if (charAt(i) == c) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public int lastIndexOf(int ch) {
|
|
return lastIndexOf(ch, length-1);
|
|
}
|
|
|
|
public int indexOf(String s) {
|
|
return indexOf(s, 0);
|
|
}
|
|
|
|
public int indexOf(String s, int start) {
|
|
if (s.length == 0) return start;
|
|
|
|
for (int i = start; i < length - s.length + 1; ++i) {
|
|
int j = 0;
|
|
for (; j < s.length; ++j) {
|
|
if (charAt(i + j) != s.charAt(j)) {
|
|
break;
|
|
}
|
|
}
|
|
if (j == s.length) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public int lastIndexOf(String s) {
|
|
return lastIndexOf(s, length - s.length);
|
|
}
|
|
|
|
public int lastIndexOf(String s, int lastIndex) {
|
|
if (s.length == 0) return lastIndex;
|
|
|
|
for (int i = Math.min(length - s.length, lastIndex); i >= 0; --i) {
|
|
int j = 0;
|
|
for (; j < s.length && i + j < length; ++j) {
|
|
if (charAt(i + j) != s.charAt(j)) {
|
|
break;
|
|
}
|
|
}
|
|
if (j == s.length) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public String replace(char oldChar, char newChar) {
|
|
if (data instanceof char[]) {
|
|
char[] buf = new char[length];
|
|
for (int i=0; i < length; i++) {
|
|
if (charAt(i) == oldChar) {
|
|
buf[i] = newChar;
|
|
} else {
|
|
buf[i] = charAt(i);
|
|
}
|
|
}
|
|
return new String(buf, 0, length, false);
|
|
} else {
|
|
byte[] buf = new byte[length];
|
|
byte[] orig = (byte[])data;
|
|
byte oldByte = (byte)oldChar;
|
|
byte newByte = (byte)newChar;
|
|
for (int i=0; i < length; i++) {
|
|
if (orig[i+offset] == oldByte) {
|
|
buf[i] = newByte;
|
|
} else {
|
|
buf[i] = orig[i+offset];
|
|
}
|
|
}
|
|
return new String(buf, 0, length, false);
|
|
}
|
|
}
|
|
|
|
public String substring(int start) {
|
|
return substring(start, length);
|
|
}
|
|
|
|
public String substring(int start, int end) {
|
|
if (start < 0)
|
|
throw new StringIndexOutOfBoundsException(start);
|
|
else if (end > length)
|
|
throw new StringIndexOutOfBoundsException(end);
|
|
int newLen = end - start;
|
|
if (newLen < 0)
|
|
throw new StringIndexOutOfBoundsException(newLen);
|
|
|
|
if (start == 0 && end == length)
|
|
return this;
|
|
else if (end - start == 0)
|
|
return "";
|
|
else
|
|
return new String(data, offset + start, newLen, false);
|
|
}
|
|
|
|
public boolean startsWith(String s) {
|
|
if (length >= s.length) {
|
|
return substring(0, s.length).compareTo(s) == 0;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean startsWith(String s, int start) {
|
|
if (length >= s.length + start) {
|
|
return substring(start, s.length).compareTo(s) == 0;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean endsWith(String s) {
|
|
if (length >= s.length) {
|
|
return substring(length - s.length).compareTo(s) == 0;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public String concat(String s) {
|
|
if (s.length() == 0) {
|
|
return this;
|
|
} else {
|
|
return this + s;
|
|
}
|
|
}
|
|
|
|
public void getBytes(int srcOffset, int srcLength,
|
|
byte[] dst, int dstOffset)
|
|
{
|
|
if (srcOffset < 0)
|
|
throw new StringIndexOutOfBoundsException(srcOffset);
|
|
else if (srcOffset + srcLength > length)
|
|
throw new StringIndexOutOfBoundsException(srcOffset + srcLength);
|
|
else if (srcLength < 0)
|
|
throw new StringIndexOutOfBoundsException(srcLength);
|
|
|
|
if (data instanceof char[]) {
|
|
char[] src = (char[]) data;
|
|
for (int i = 0; i < srcLength; ++i) {
|
|
dst[i + dstOffset] = (byte) src[i + offset + srcOffset];
|
|
}
|
|
} else {
|
|
byte[] src = (byte[]) data;
|
|
System.arraycopy(src, offset + srcOffset, dst, dstOffset, srcLength);
|
|
}
|
|
}
|
|
|
|
public byte[] getBytes() {
|
|
try {
|
|
return getBytes(DEFAULT_ENCODING);
|
|
} catch (java.io.UnsupportedEncodingException ex) {
|
|
throw new RuntimeException(
|
|
"Default '" + DEFAULT_ENCODING + "' encoding not handled", ex);
|
|
}
|
|
}
|
|
|
|
public byte[] getBytes(String format)
|
|
throws java.io.UnsupportedEncodingException
|
|
{
|
|
if(data instanceof byte[]) {
|
|
byte[] b = new byte[length];
|
|
getBytes(0, length, b, 0);
|
|
return b;
|
|
}
|
|
String fmt = format.trim().toUpperCase();
|
|
if (DEFAULT_ENCODING.equals(fmt)) {
|
|
return Utf8.encode((char[])data, offset, length);
|
|
} else if (ISO_8859_1_ENCODING.equals(fmt) || LATIN_1_ENCODING.equals(fmt)) {
|
|
return Iso88591.encode((char[])data, offset, length);
|
|
} else {
|
|
throw new java.io.UnsupportedEncodingException(
|
|
"Encoding " + format + " not supported");
|
|
}
|
|
}
|
|
|
|
public void getChars(int srcOffset, int srcEnd,
|
|
char[] dst, int dstOffset)
|
|
{
|
|
if (srcOffset < 0)
|
|
throw new StringIndexOutOfBoundsException(srcOffset);
|
|
else if (srcEnd > length)
|
|
throw new StringIndexOutOfBoundsException(srcEnd);
|
|
|
|
int srcLength = srcEnd-srcOffset;
|
|
if (data instanceof char[]) {
|
|
char[] src = (char[]) data;
|
|
System.arraycopy(src, offset + srcOffset, dst, dstOffset, srcLength);
|
|
} else {
|
|
byte[] src = (byte[]) data;
|
|
for (int i = 0; i < srcLength; ++i) {
|
|
dst[i + dstOffset] = (char) src[i + offset + srcOffset];
|
|
}
|
|
}
|
|
}
|
|
|
|
public char[] toCharArray() {
|
|
char[] b = new char[length];
|
|
getChars(0, length, b, 0);
|
|
return b;
|
|
}
|
|
|
|
@Override
|
|
public char charAt(int index) {
|
|
if (index < 0 || index > length) {
|
|
throw new StringIndexOutOfBoundsException(index);
|
|
}
|
|
|
|
if (data instanceof char[]) {
|
|
return ((char[]) data)[index + offset];
|
|
} else {
|
|
return (char) ((byte[]) data)[index + offset];
|
|
}
|
|
}
|
|
|
|
public String[] split(String regex) {
|
|
return split(regex, 0);
|
|
}
|
|
|
|
public String[] split(String regex, int limit) {
|
|
return Pattern.compile(regex).split(this, limit);
|
|
}
|
|
|
|
@Override
|
|
public CharSequence subSequence(int start, int end) {
|
|
return substring(start, end);
|
|
}
|
|
|
|
public boolean matches(String regex) {
|
|
return Pattern.matches(regex, this);
|
|
}
|
|
|
|
public String replaceFirst(String regex, String replacement) {
|
|
return Pattern.compile(regex).matcher(this).replaceFirst(replacement);
|
|
}
|
|
|
|
public String replaceAll(String regex, String replacement) {
|
|
return Pattern.compile(regex).matcher(this).replaceAll(replacement);
|
|
}
|
|
|
|
public String replace(CharSequence target, CharSequence replace) {
|
|
if (target.length() == 0) {
|
|
return this.infuse(replace.toString());
|
|
}
|
|
|
|
String targetString = target.toString();
|
|
String replaceString = replace.toString();
|
|
|
|
int targetSize = target.length();
|
|
|
|
StringBuilder returnValue = new StringBuilder();
|
|
String unhandled = this;
|
|
|
|
int index = -1;
|
|
while ((index = unhandled.indexOf(targetString)) != -1) {
|
|
returnValue.append(unhandled.substring(0, index)).append(replaceString);
|
|
unhandled = unhandled.substring(index + targetSize,
|
|
unhandled.length());
|
|
}
|
|
|
|
returnValue.append(unhandled);
|
|
return returnValue.toString();
|
|
}
|
|
|
|
private String infuse(String infuseWith) {
|
|
StringBuilder retVal = new StringBuilder();
|
|
|
|
String me = this;
|
|
for (int i = 0; i < me.length(); i++) {
|
|
retVal.append(infuseWith).append(me.substring(i, i + 1));
|
|
}
|
|
|
|
retVal.append(infuseWith);
|
|
return retVal.toString();
|
|
}
|
|
|
|
public native String intern();
|
|
|
|
public static String format(String fmt, Object... args) {
|
|
final Formatter formatter = new Formatter();
|
|
final String result = formatter.format(fmt, args).toString();
|
|
formatter.close();
|
|
return result;
|
|
}
|
|
|
|
public static String format(Locale l, String fmt, Object... args) {
|
|
final Formatter formatter = new Formatter();
|
|
final String result = formatter.format(l, fmt, args).toString();
|
|
formatter.close();
|
|
return result;
|
|
}
|
|
|
|
public static String valueOf(Object s) {
|
|
return s == null ? "null" : s.toString();
|
|
}
|
|
|
|
public static String valueOf(boolean v) {
|
|
return Boolean.toString(v);
|
|
}
|
|
|
|
public static String valueOf(byte v) {
|
|
return Byte.toString(v);
|
|
}
|
|
|
|
public static String valueOf(short v) {
|
|
return Short.toString(v);
|
|
}
|
|
|
|
public static String valueOf(char v) {
|
|
return Character.toString(v);
|
|
}
|
|
|
|
public static String valueOf(int v) {
|
|
return Integer.toString(v);
|
|
}
|
|
|
|
public static String valueOf(long v) {
|
|
return Long.toString(v);
|
|
}
|
|
|
|
public static String valueOf(float v) {
|
|
return Float.toString(v);
|
|
}
|
|
|
|
public static String valueOf(double v) {
|
|
return Double.toString(v);
|
|
}
|
|
|
|
public static String valueOf(char[] data, int offset, int length) {
|
|
return new String(data, offset, length);
|
|
}
|
|
|
|
public static String valueOf(char[] data) {
|
|
return valueOf(data, 0, data.length);
|
|
}
|
|
|
|
public int lastIndexOf(int ch, int lastIndex) {
|
|
if (lastIndex >= length) {
|
|
lastIndex = length - 1;
|
|
}
|
|
for (int i = lastIndex ; i >= 0; --i) {
|
|
if (charAt(i) == ch) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public boolean regionMatches(int thisOffset, String match, int matchOffset,
|
|
int length)
|
|
{
|
|
return regionMatches(false, thisOffset, match, matchOffset, length);
|
|
}
|
|
|
|
public boolean regionMatches(boolean ignoreCase, int thisOffset,
|
|
String match, int matchOffset, int length)
|
|
{
|
|
String a = substring(thisOffset, thisOffset + length);
|
|
String b = match.substring(matchOffset, matchOffset + length);
|
|
if (ignoreCase) {
|
|
return a.equalsIgnoreCase(b);
|
|
} else {
|
|
return a.equals(b);
|
|
}
|
|
}
|
|
|
|
public boolean isEmpty() {
|
|
return length == 0;
|
|
}
|
|
|
|
public boolean contains(CharSequence match) {
|
|
return indexOf(match.toString()) != -1;
|
|
}
|
|
|
|
public int codePointAt(int offset) {
|
|
return Character.codePointAt(this, offset);
|
|
}
|
|
|
|
public int codePointCount(int start, int end) {
|
|
return Character.codePointCount(this, start, end);
|
|
}
|
|
|
|
public String toUpperCase(Locale locale) {
|
|
if (locale == Locale.ENGLISH) {
|
|
return toUpperCase();
|
|
} else {
|
|
throw new UnsupportedOperationException("toUpperCase("+locale+')');
|
|
}
|
|
}
|
|
|
|
public String toLowerCase(Locale locale) {
|
|
if (locale == Locale.ENGLISH) {
|
|
return toLowerCase();
|
|
} else {
|
|
throw new UnsupportedOperationException("toLowerCase("+locale+')');
|
|
}
|
|
}
|
|
}
|