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:
parent
e5763ff6f0
commit
2541a62def
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -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);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue