Moving the CharSequence specific methods out of StringUtils and reintroducing the CharSequenceUtils class

git-svn-id: https://svn.apache.org/repos/asf/commons/proper/lang/trunk@1089751 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Henri Yandell 2011-04-07 06:28:21 +00:00
parent e5763ff6f0
commit 2541a62def
4 changed files with 296 additions and 213 deletions

View File

@ -0,0 +1,196 @@
/*
* 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 org.apache.commons.lang3;
/**
* <p>Operations on {@link java.lang.CharSequence} that are
* {@code null} safe.</p>
*
* @see java.lang.CharSequence
* @since 3.0
* @version $Id: StringUtils.java 1089734 2011-04-07 04:37:52Z bayard $
*/
public class CharSequenceUtils {
/**
* <p>{@code CharSequenceUtils} instances should NOT be constructed in
* standard programming. </p>
*
* <p>This constructor is public to permit tools that require a JavaBean
* instance to operate.</p>
*/
public CharSequenceUtils() {
super();
}
//-----------------------------------------------------------------------
/**
* <p>Returns a new {@code CharSequence} that is a subsequence of this
* sequence starting with the {@code char} value at the specified index.</p>
*
* <p>This provides the {@code CharSequence} equivalent to {@link String#substring(int)}.
* The length (in {@code char}) of the returned sequence is {@code length() - start},
* so if {@code start == end} then an empty sequence is returned.</p>
*
* @param cs the specified subsequence, null returns null
* @param start the start index, inclusive, valid
* @return a new subsequence, may be null
* @throws IndexOutOfBoundsException if {@code start} is negative or if
* {@code start} is greater than {@code length()}
*/
public static CharSequence subSequence(CharSequence cs, int start) {
return cs == null ? null : cs.subSequence(start, cs.length());
}
//-----------------------------------------------------------------------
/**
* Used by the indexOf(CharSequence methods) as a green implementation of
* indexOf.
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the char to be searched for
* @param start the start index
* @return the index where the search char was found
*/
static int indexOf(CharSequence cs, int searchChar, int start) {
if (cs instanceof String) {
return ((String) cs).indexOf(searchChar, start);
} else {
int sz = cs.length();
if ( start < 0 ) {
start = 0;
}
for ( int i=start; i < sz; i++ ) {
if ( cs.charAt(i) == searchChar) {
return i;
}
}
return -1;
}
}
/**
* Used by the indexOf(CharSequence methods) as a green implementation of indexOf.
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the {@code CharSequence} to be searched for
* @param start the start index
* @return the index where the search sequence was found
*/
static int indexOf(CharSequence cs, CharSequence searchChar, int start) {
if (cs instanceof String && searchChar instanceof String) {
// TODO: Do we assume searchChar is usually relatively small;
// If so then calling toString() on it is better than reverting to
// the green implementation in the else block
return ((String) cs).indexOf( (String) searchChar, start);
} else {
// TODO: Implement rather than convert to String
return cs.toString().indexOf(searchChar.toString(), start);
}
}
/**
* Used by the lastIndexOf(CharSequence methods) as a green implementation of lastIndexOf
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the char to be searched for
* @param start the start index
* @return the index where the search char was found
*/
static int lastIndexOf(CharSequence cs, int searchChar, int start) {
if (cs instanceof String) {
return ((String) cs).lastIndexOf(searchChar, start);
} else {
int sz = cs.length();
if ( start < 0 ) {
return -1;
}
if ( start >= sz ) {
start = sz - 1;
}
for ( int i=start; i >= 0; --i ) {
if ( cs.charAt(i) == searchChar) {
return i;
}
}
return -1;
}
}
/**
* Used by the lastIndexOf(CharSequence methods) as a green implementation of lastIndexOf
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the {@code CharSequence} to be searched for
* @param start the start index
* @return the index where the search sequence was found
*/
static int lastIndexOf(CharSequence cs, CharSequence searchChar, int start) {
if (cs instanceof String && searchChar instanceof String) {
// TODO: Do we assume searchChar is usually relatively small;
// If so then calling toString() on it is better than reverting to
// the green implementation in the else block
return ((String) cs).lastIndexOf( (String) searchChar, start);
} else {
// TODO: Implement rather than convert to String
return cs.toString().lastIndexOf(searchChar.toString(), start);
}
}
/**
* Green implementation of toCharArray.
*
* @param cs the {@code CharSequence} to be processed
* @return the resulting char array
*/
static char[] toCharArray(CharSequence cs) {
if (cs instanceof String) {
return ((String) cs).toCharArray();
} else {
int sz = cs.length();
char[] array = new char[cs.length()];
for (int i=0; i < sz; i++) {
array[i] = cs.charAt(i);
}
return array;
}
}
/**
* Green implementation of regionMatches.
*
* @param cs the {@code CharSequence} to be processed
* @param ignoreCase whether or not to be case insensitive
* @param thisStart the index to start on the {@code cs} CharSequence
* @param substring the {@code CharSequence} to be looked for
* @param start the index to start on the {@code substring} CharSequence
* @param length character length of the region
* @return whether the region matched
*/
static boolean regionMatches(CharSequence cs, boolean ignoreCase, int thisStart,
CharSequence substring, int start, int length)
{
if (cs instanceof String && substring instanceof String) {
return ((String) cs).regionMatches(ignoreCase, thisStart, ((String)substring), start, length);
} else {
// TODO: Implement rather than convert to String
return cs.toString().regionMatches(ignoreCase, thisStart, substring.toString(), start, length);
}
}
}

View File

@ -796,7 +796,7 @@ public class StringUtils {
if (str1 == null || str2 == null) { if (str1 == null || str2 == null) {
return str1 == str2; return str1 == str2;
} else { } else {
return regionMatchesSequence(str1, true, 0, str2, 0, Math.max(str1.length(), str2.length())); return CharSequenceUtils.regionMatches(str1, true, 0, str2, 0, Math.max(str1.length(), str2.length()));
} }
} }
@ -825,7 +825,7 @@ public class StringUtils {
if (isEmpty(seq)) { if (isEmpty(seq)) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.indexOfSequence(seq, searchChar, 0); return CharSequenceUtils.indexOf(seq, searchChar, 0);
} }
/** /**
@ -857,7 +857,7 @@ public class StringUtils {
if (isEmpty(seq)) { if (isEmpty(seq)) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.indexOfSequence(seq, searchChar, startPos); return CharSequenceUtils.indexOf(seq, searchChar, startPos);
} }
/** /**
@ -887,7 +887,7 @@ public class StringUtils {
if (seq == null || searchSeq == null) { if (seq == null || searchSeq == null) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.indexOfSequence(seq, searchSeq, 0); return CharSequenceUtils.indexOf(seq, searchSeq, 0);
} }
/** /**
@ -926,7 +926,7 @@ public class StringUtils {
if (seq == null || searchSeq == null) { if (seq == null || searchSeq == null) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.indexOfSequence(seq, searchSeq, startPos); return CharSequenceUtils.indexOf(seq, searchSeq, startPos);
} }
/** /**
@ -991,9 +991,9 @@ public class StringUtils {
int index = lastIndex ? str.length() : INDEX_NOT_FOUND; int index = lastIndex ? str.length() : INDEX_NOT_FOUND;
do { do {
if (lastIndex) { if (lastIndex) {
index = lastIndexOfSequence(str, searchStr, index - 1); index = CharSequenceUtils.lastIndexOf(str, searchStr, index - 1);
} else { } else {
index = indexOfSequence(str, searchStr, index + 1); index = CharSequenceUtils.indexOf(str, searchStr, index + 1);
} }
if (index < 0) { if (index < 0) {
return index; return index;
@ -1077,7 +1077,7 @@ public class StringUtils {
return startPos; return startPos;
} }
for (int i = startPos; i < endLimit; i++) { for (int i = startPos; i < endLimit; i++) {
if (regionMatchesSequence(str, true, i, searchStr, 0, searchStr.length())) { if (CharSequenceUtils.regionMatches(str, true, i, searchStr, 0, searchStr.length())) {
return i; return i;
} }
} }
@ -1109,7 +1109,7 @@ public class StringUtils {
if (isEmpty(seq)) { if (isEmpty(seq)) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.lastIndexOfSequence(seq, searchChar, seq.length()); return CharSequenceUtils.lastIndexOf(seq, searchChar, seq.length());
} }
/** /**
@ -1143,7 +1143,7 @@ public class StringUtils {
if (isEmpty(seq)) { if (isEmpty(seq)) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.lastIndexOfSequence(seq, searchChar, startPos); return CharSequenceUtils.lastIndexOf(seq, searchChar, startPos);
} }
/** /**
@ -1172,7 +1172,7 @@ public class StringUtils {
if (seq == null || searchSeq == null) { if (seq == null || searchSeq == null) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.lastIndexOfSequence(seq, searchSeq, seq.length()); return CharSequenceUtils.lastIndexOf(seq, searchSeq, seq.length());
} }
/** /**
@ -1244,7 +1244,7 @@ public class StringUtils {
if (seq == null || searchSeq == null) { if (seq == null || searchSeq == null) {
return INDEX_NOT_FOUND; return INDEX_NOT_FOUND;
} }
return StringUtils.lastIndexOfSequence(seq, searchSeq, startPos); return CharSequenceUtils.lastIndexOf(seq, searchSeq, startPos);
} }
/** /**
@ -1319,7 +1319,7 @@ public class StringUtils {
} }
for (int i = startPos; i >= 0; i--) { for (int i = startPos; i >= 0; i--) {
if (regionMatchesSequence(str, true, i, searchStr, 0, searchStr.length())) { if (CharSequenceUtils.regionMatches(str, true, i, searchStr, 0, searchStr.length())) {
return i; return i;
} }
} }
@ -1351,7 +1351,7 @@ public class StringUtils {
if (isEmpty(seq)) { if (isEmpty(seq)) {
return false; return false;
} }
return indexOfSequence(seq, searchChar, 0) >= 0; return CharSequenceUtils.indexOf(seq, searchChar, 0) >= 0;
} }
/** /**
@ -1379,7 +1379,7 @@ public class StringUtils {
if (seq == null || searchSeq == null) { if (seq == null || searchSeq == null) {
return false; return false;
} }
return indexOfSequence(seq, searchSeq, 0) >= 0; return CharSequenceUtils.indexOf(seq, searchSeq, 0) >= 0;
} }
/** /**
@ -1412,7 +1412,7 @@ public class StringUtils {
int len = searchStr.length(); int len = searchStr.length();
int max = str.length() - len; int max = str.length() - len;
for (int i = 0; i <= max; i++) { for (int i = 0; i <= max; i++) {
if (regionMatchesSequence(str, true, i, searchStr, 0, len)) { if (CharSequenceUtils.regionMatches(str, true, i, searchStr, 0, len)) {
return true; return true;
} }
} }
@ -1610,7 +1610,7 @@ public class StringUtils {
if (searchChars == null) { if (searchChars == null) {
return false; return false;
} }
return containsAny(cs, toCharArraySequence(searchChars)); return containsAny(cs, CharSequenceUtils.toCharArray(searchChars));
} }
// IndexOfAnyBut chars // IndexOfAnyBut chars
@ -1695,10 +1695,10 @@ public class StringUtils {
int strLen = seq.length(); int strLen = seq.length();
for (int i = 0; i < strLen; i++) { for (int i = 0; i < strLen; i++) {
char ch = seq.charAt(i); char ch = seq.charAt(i);
boolean chFound = indexOfSequence(searchChars, ch, 0) >= 0; boolean chFound = CharSequenceUtils.indexOf(searchChars, ch, 0) >= 0;
if (i + 1 < strLen && Character.isHighSurrogate(ch)) { if (i + 1 < strLen && Character.isHighSurrogate(ch)) {
char ch2 = seq.charAt(i + 1); char ch2 = seq.charAt(i + 1);
if (chFound && indexOfSequence(searchChars, ch2, 0) < 0) { if (chFound && CharSequenceUtils.indexOf(searchChars, ch2, 0) < 0) {
return i; return i;
} }
} else { } else {
@ -1906,7 +1906,7 @@ public class StringUtils {
if (search == null) { if (search == null) {
continue; continue;
} }
tmp = indexOfSequence(str, search, 0); tmp = CharSequenceUtils.indexOf(str, search, 0);
if (tmp == INDEX_NOT_FOUND) { if (tmp == INDEX_NOT_FOUND) {
continue; continue;
} }
@ -1956,7 +1956,7 @@ public class StringUtils {
if (search == null) { if (search == null) {
continue; continue;
} }
tmp = lastIndexOfSequence(str, search, str.length()); tmp = CharSequenceUtils.lastIndexOf(str, search, str.length());
if (tmp > ret) { if (tmp > ret) {
ret = tmp; ret = tmp;
} }
@ -5133,7 +5133,7 @@ public class StringUtils {
} }
int count = 0; int count = 0;
int idx = 0; int idx = 0;
while ((idx = indexOfSequence(str, sub, idx)) != INDEX_NOT_FOUND) { while ((idx = CharSequenceUtils.indexOf(str, sub, idx)) != INDEX_NOT_FOUND) {
count++; count++;
idx += sub.length(); idx += sub.length();
} }
@ -6194,7 +6194,7 @@ public class StringUtils {
if (prefix.length() > str.length()) { if (prefix.length() > str.length()) {
return false; return false;
} }
return regionMatchesSequence(str, ignoreCase, 0, prefix, 0, prefix.length()); return CharSequenceUtils.regionMatches(str, ignoreCase, 0, prefix, 0, prefix.length());
} }
/** /**
@ -6302,7 +6302,7 @@ public class StringUtils {
return false; return false;
} }
int strOffset = str.length() - suffix.length(); int strOffset = str.length() - suffix.length();
return regionMatchesSequence(str, ignoreCase, strOffset, suffix, 0, suffix.length()); return CharSequenceUtils.regionMatches(str, ignoreCase, strOffset, suffix, 0, suffix.length());
} }
/** /**
@ -6384,160 +6384,4 @@ public class StringUtils {
return false; return false;
} }
//-----------------------------------------------------------------------
/**
* <p>Returns a new {@code CharSequence} that is a subsequence of this
* sequence starting with the {@code char} value at the specified index.</p>
*
* <p>This provides the {@code CharSequence} equivalent to {@link String#substring(int)}.
* The length (in {@code char}) of the returned sequence is {@code length() - start},
* so if {@code start == end} then an empty sequence is returned.</p>
*
* @param cs the specified subsequence, null returns null
* @param start the start index, inclusive, valid
* @return a new subsequence, may be null
* @throws IndexOutOfBoundsException if {@code start} is negative or if
* {@code start} is greater than {@code length()}
*/
public static CharSequence subSequence(CharSequence cs, int start) {
return cs == null ? null : cs.subSequence(start, cs.length());
}
//-----------------------------------------------------------------------
/**
* Used by the indexOf(CharSequence methods) as a green implementation of
* indexOf.
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the char to be searched for
* @param start the start index
* @return the index where the search char was found
*/
static int indexOfSequence(CharSequence cs, int searchChar, int start) {
if (cs instanceof String) {
return ((String) cs).indexOf(searchChar, start);
} else {
int sz = cs.length();
if ( start < 0 ) {
start = 0;
}
for ( int i=start; i < sz; i++ ) {
if ( cs.charAt(i) == searchChar) {
return i;
}
}
return -1;
}
}
/**
* Used by the indexOf(CharSequence methods) as a green implementation of indexOf.
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the {@code CharSequence} to be searched for
* @param start the start index
* @return the index where the search sequence was found
*/
static int indexOfSequence(CharSequence cs, CharSequence searchChar, int start) {
if (cs instanceof String && searchChar instanceof String) {
// TODO: Do we assume searchChar is usually relatively small;
// If so then calling toString() on it is better than reverting to
// the green implementation in the else block
return ((String) cs).indexOf( (String) searchChar, start);
} else {
// TODO: Implement rather than convert to String
return cs.toString().indexOf(searchChar.toString(), start);
}
}
/**
* Used by the lastIndexOf(CharSequence methods) as a green implementation of lastIndexOf
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the char to be searched for
* @param start the start index
* @return the index where the search char was found
*/
static int lastIndexOfSequence(CharSequence cs, int searchChar, int start) {
if (cs instanceof String) {
return ((String) cs).lastIndexOf(searchChar, start);
} else {
int sz = cs.length();
if ( start < 0 ) {
return -1;
}
if ( start >= sz ) {
start = sz - 1;
}
for ( int i=start; i >= 0; --i ) {
if ( cs.charAt(i) == searchChar) {
return i;
}
}
return -1;
}
}
/**
* Used by the lastIndexOf(CharSequence methods) as a green implementation of lastIndexOf
*
* @param cs the {@code CharSequence} to be processed
* @param searchChar the {@code CharSequence} to be searched for
* @param start the start index
* @return the index where the search sequence was found
*/
static int lastIndexOfSequence(CharSequence cs, CharSequence searchChar, int start) {
if (cs instanceof String && searchChar instanceof String) {
// TODO: Do we assume searchChar is usually relatively small;
// If so then calling toString() on it is better than reverting to
// the green implementation in the else block
return ((String) cs).lastIndexOf( (String) searchChar, start);
} else {
// TODO: Implement rather than convert to String
return cs.toString().lastIndexOf(searchChar.toString(), start);
}
}
/**
* Green implementation of toCharArray.
*
* @param cs the {@code CharSequence} to be processed
* @return the resulting char array
*/
static char[] toCharArraySequence(CharSequence cs) {
if (cs instanceof String) {
return ((String) cs).toCharArray();
} else {
int sz = cs.length();
char[] array = new char[cs.length()];
for (int i=0; i < sz; i++) {
array[i] = cs.charAt(i);
}
return array;
}
}
/**
* Green implementation of regionMatches.
*
* @param cs the {@code CharSequence} to be processed
* @param ignoreCase whether or not to be case insensitive
* @param thisStart the index to start on the {@code cs} CharSequence
* @param substring the {@code CharSequence} to be looked for
* @param start the index to start on the {@code substring} CharSequence
* @param length character length of the region
* @return whether the region matched
*/
static boolean regionMatchesSequence(CharSequence cs, boolean ignoreCase, int thisStart,
CharSequence substring, int start, int length)
{
if (cs instanceof String && substring instanceof String) {
return ((String) cs).regionMatches(ignoreCase, thisStart, ((String)substring), start, length);
} else {
// TODO: Implement rather than convert to String
return cs.toString().regionMatches(ignoreCase, thisStart, substring.toString(), start, length);
}
}
} }

View File

@ -0,0 +1,76 @@
/*
* 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 org.apache.commons.lang3;
import java.lang.reflect.Constructor;
import java.lang.reflect.Modifier;
import junit.framework.Assert;
import junit.framework.TestCase;
/**
* Tests CharSequenceUtils
*
* @author Gary Gregory
* @version $Id: CharSequenceUtilsTest.java 1066341 2011-02-02 06:21:53Z bayard $
*/
public class CharSequenceUtilsTest extends TestCase {
//-----------------------------------------------------------------------
public void testConstructor() {
assertNotNull(new CharSequenceUtils());
Constructor<?>[] cons = CharSequenceUtils.class.getDeclaredConstructors();
assertEquals(1, cons.length);
assertEquals(true, Modifier.isPublic(cons[0].getModifiers()));
assertEquals(true, Modifier.isPublic(CharSequenceUtils.class.getModifiers()));
assertEquals(false, Modifier.isFinal(CharSequenceUtils.class.getModifiers()));
}
//-----------------------------------------------------------------------
public void testSubSequence() {
//
// null input
//
Assert.assertEquals(null, CharSequenceUtils.subSequence(null, -1));
Assert.assertEquals(null, CharSequenceUtils.subSequence(null, 0));
Assert.assertEquals(null, CharSequenceUtils.subSequence(null, 1));
//
// non-null input
//
Assert.assertEquals(StringUtils.EMPTY, CharSequenceUtils.subSequence(StringUtils.EMPTY, 0));
Assert.assertEquals("012", CharSequenceUtils.subSequence("012", 0));
Assert.assertEquals("12", CharSequenceUtils.subSequence("012", 1));
Assert.assertEquals("2", CharSequenceUtils.subSequence("012", 2));
Assert.assertEquals(StringUtils.EMPTY, CharSequenceUtils.subSequence("012", 3));
//
// Exception expected
//
try {
Assert.assertEquals(null, CharSequenceUtils.subSequence(StringUtils.EMPTY, -1));
Assert.fail("Expected " + IndexOutOfBoundsException.class.getName());
} catch (IndexOutOfBoundsException e) {
// Expected
}
try {
Assert.assertEquals(null, CharSequenceUtils.subSequence(StringUtils.EMPTY, 1));
Assert.fail("Expected " + IndexOutOfBoundsException.class.getName());
} catch (IndexOutOfBoundsException e) {
// Expected
}
}
}

View File

@ -311,37 +311,4 @@ public class StringUtilsSubstringTest extends TestCase {
StringUtils.countMatches("oooooooooooo", "ooo")); StringUtils.countMatches("oooooooooooo", "ooo"));
} }
//-----------------------------------------------------------------------
public void testSubSequence() {
//
// null input
//
Assert.assertEquals(null, StringUtils.subSequence(null, -1));
Assert.assertEquals(null, StringUtils.subSequence(null, 0));
Assert.assertEquals(null, StringUtils.subSequence(null, 1));
//
// non-null input
//
Assert.assertEquals(StringUtils.EMPTY, StringUtils.subSequence(StringUtils.EMPTY, 0));
Assert.assertEquals("012", StringUtils.subSequence("012", 0));
Assert.assertEquals("12", StringUtils.subSequence("012", 1));
Assert.assertEquals("2", StringUtils.subSequence("012", 2));
Assert.assertEquals(StringUtils.EMPTY, StringUtils.subSequence("012", 3));
//
// Exception expected
//
try {
Assert.assertEquals(null, StringUtils.subSequence(StringUtils.EMPTY, -1));
Assert.fail("Expected " + IndexOutOfBoundsException.class.getName());
} catch (IndexOutOfBoundsException e) {
// Expected
}
try {
Assert.assertEquals(null, StringUtils.subSequence(StringUtils.EMPTY, 1));
Assert.fail("Expected " + IndexOutOfBoundsException.class.getName());
} catch (IndexOutOfBoundsException e) {
// Expected
}
}
} }