All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.aspectran.core.util.StringUtils Maven / Gradle / Ivy

There is a newer version: 8.1.5
Show newest version
/*
 * Copyright (c) 2008-2019 The Aspectran Project
 *
 * Licensed 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 com.aspectran.core.util;

import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Locale;
import java.util.StringTokenizer;
import java.util.TimeZone;

/**
 * Static utility methods pertaining to {@code String} or {@code CharSequence} instances.
 */
public class StringUtils {

    /** The empty {@link String} */
    public static final String EMPTY = "";

    /**
     * Check that the given {@code CharSequence} is neither {@code null} nor
     * of length 0.
     * 

Note: this method returns {@code true} for a {@code CharSequence} * that purely consists of whitespace. *

     * StringUtils.hasLength(null) = false
     * StringUtils.hasLength("") = false
     * StringUtils.hasLength(" ") = true
     * StringUtils.hasLength("Hello") = true
     * 
* * @param str the {@code CharSequence} to check (may be {@code null}) * @return {@code true} if the {@code CharSequence} is not {@code null} and has length * @see #hasText(String) */ public static boolean hasLength(CharSequence str) { return (str != null && str.length() > 0); } /** * Check that the given {@code String} is neither {@code null} nor of length 0. *

Note: this method returns {@code true} for a {@code String} that * purely consists of whitespace. * * @param str the {@code String} to check (may be {@code null}) * @return {@code true} if the {@code String} is not {@code null} and has length * @see #hasLength(CharSequence) * @see #hasText(String) */ public static boolean hasLength(String str) { return hasLength((CharSequence)str); } /** * Check whether the given {@code CharSequence} contains actual text. *

More specifically, this method returns {@code true} if the * {@code CharSequence} is not {@code null}, its length is greater than * 0, and it contains at least one non-whitespace character. *

     * StringUtils.hasText(null) = false
     * StringUtils.hasText("") = false
     * StringUtils.hasText(" ") = false
     * StringUtils.hasText("12345") = true
     * StringUtils.hasText(" 12345 ") = true
     * 
* * @param str the {@code CharSequence} to check (may be {@code null}) * @return {@code true} if the {@code CharSequence} is not {@code null}, * its length is greater than 0, and it does not contain whitespace only * @see Character#isWhitespace */ public static boolean hasText(CharSequence str) { if (!hasLength(str)) { return false; } int strLen = str.length(); for (int i = 0; i < strLen; i++) { if (!Character.isWhitespace(str.charAt(i))) { return true; } } return false; } /** * Check whether the given {@code String} contains actual text. *

More specifically, this method returns {@code true} if the * {@code String} is not {@code null}, its length is greater than 0, * and it contains at least one non-whitespace character. * * @param str the {@code String} to check (may be {@code null}) * @return {@code true} if the {@code String} is not {@code null}, its * length is greater than 0, and it does not contain whitespace only * @see #hasText(CharSequence) */ public static boolean hasText(String str) { return hasText((CharSequence)str); } /** * Check whether the given {@code CharSequence} contains any whitespace characters. * * @param str the {@code CharSequence} to check (may be {@code null}) * @return {@code true} if the {@code CharSequence} is not empty and * contains at least 1 whitespace character * @see Character#isWhitespace */ public static boolean containsWhitespace(CharSequence str) { if (!hasLength(str)) { return false; } int strLen = str.length(); for (int i = 0; i < strLen; i++) { if (Character.isWhitespace(str.charAt(i))) { return true; } } return false; } /** * Check whether the given {@code String} contains any whitespace characters. * * @param str the {@code String} to check (may be {@code null}) * @return {@code true} if the {@code String} is not empty and * contains at least 1 whitespace character * @see #containsWhitespace(CharSequence) */ public static boolean containsWhitespace(String str) { return containsWhitespace((CharSequence)str); } /** * Trim leading and trailing whitespace from the given {@code String}. * * @param str the {@code String} to check * @return the trimmed {@code String} * @see java.lang.Character#isWhitespace */ public static String trimWhitespace(String str) { if (!hasLength(str)) { return str; } StringBuilder buf = new StringBuilder(str); while (buf.length() > 0 && Character.isWhitespace(buf.charAt(0))) { buf.deleteCharAt(0); } while (buf.length() > 0 && Character.isWhitespace(buf.charAt(buf.length() - 1))) { buf.deleteCharAt(buf.length() - 1); } return buf.toString(); } /** * Trim all whitespace from the given {@code String}: * leading, trailing, and in between characters. * * @param str the {@code String} to check * @return the trimmed {@code String} * @see java.lang.Character#isWhitespace */ public static String trimAllWhitespace(String str) { if (!hasLength(str)) { return str; } StringBuilder buf = new StringBuilder(str); int index = 0; while (buf.length() > index) { if (Character.isWhitespace(buf.charAt(index))) { buf.deleteCharAt(index); } else { index++; } } return buf.toString(); } /** * Trim leading whitespace from the given {@code String}. * * @param str the {@code String} to check * @return the trimmed {@code String} * @see java.lang.Character#isWhitespace */ public static String trimLeadingWhitespace(String str) { if (!hasLength(str)) { return str; } StringBuilder buf = new StringBuilder(str); while (buf.length() > 0 && Character.isWhitespace(buf.charAt(0))) { buf.deleteCharAt(0); } return buf.toString(); } /** * Trim trailing whitespace from the given {@code String}. * * @param str the {@code String} to check * @return the trimmed {@code String} * @see java.lang.Character#isWhitespace */ public static String trimTrailingWhitespace(String str) { if (!hasLength(str)) { return str; } StringBuilder buf = new StringBuilder(str); while (buf.length() > 0 && Character.isWhitespace(buf.charAt(buf.length() - 1))) { buf.deleteCharAt(buf.length() - 1); } return buf.toString(); } /** * Trim all occurrences of the supplied leading character from the given {@code String}. * * @param str the {@code String} to check * @param leadingCharacter the leading character to be trimmed * @return the trimmed {@code String} */ public static String trimLeadingCharacter(String str, char leadingCharacter) { if (!hasLength(str)) { return str; } StringBuilder buf = new StringBuilder(str); while (buf.length() > 0 && buf.charAt(0) == leadingCharacter) { buf.deleteCharAt(0); } return buf.toString(); } /** * Trim all occurrences of the supplied trailing character from the given {@code String}. * * @param str the {@code String} to check * @param trailingCharacter the trailing character to be trimmed * @return the trimmed {@code String} */ public static String trimTrailingCharacter(String str, char trailingCharacter) { if (!hasLength(str)) { return str; } StringBuilder buf = new StringBuilder(str); while (buf.length() > 0 && buf.charAt(buf.length() - 1) == trailingCharacter) { buf.deleteCharAt(buf.length() - 1); } return buf.toString(); } /** * Test if the given {@code String} starts with the specified prefix, * ignoring upper/lower case. * * @param str the {@code String} to check * @param prefix the prefix to look for * @return {@code true} if the {@code String} starts with the prefix, case insensitive, or both {@code null} * @see java.lang.String#startsWith */ public static boolean startsWithIgnoreCase(String str, String prefix) { return (str != null && prefix != null && str.length() >= prefix.length() && str.regionMatches(true, 0, prefix, 0, prefix.length())); } /** * Test if the given {@code String} ends with the specified suffix, * ignoring upper/lower case. * * @param str the {@code String} to check * @param suffix the suffix to look for * @return {@code true} if the {@code String} ends with the suffix, case insensitive, or both {@code null} * @see java.lang.String#endsWith */ public static boolean endsWithIgnoreCase(String str, String suffix) { return (str != null && suffix != null && str.length() >= suffix.length() && str.regionMatches(true, str.length() - suffix.length(), suffix, 0, suffix.length())); } /** * Returns {@code true} if the given string is null or is the empty string. * * @param str a string reference to check * @return {@code true} if the string is null or is the empty string */ public static boolean isEmpty(String str) { return (str == null || str.length() == 0); } /** * Returns the given string if it is non-null; the empty string otherwise. * * @param str the string to test and possibly return * @return {@code string} itself if it is non-null; {@code ""} if it is null */ public static String nullToEmpty(String str) { return (str != null ? str : EMPTY); } /** * Returns the given string if it is nonempty; {@code null} otherwise. * * @param str the string to test and possibly return * @return {@code string} itself if it is nonempty; {@code null} if it is empty or null */ public static String emptyToNull(String str) { return (str == null || str.length() == 0 ? null : str); } /** * Test if the given {@code String} starts with the specified prefix character. * * @param str the {@code String} to check * @param prefix the prefix character to look for * @return true if the string starts with the specified prefix; otherwise false * @see java.lang.String#startsWith */ public static boolean startsWith(String str, char prefix) { return (str != null && !str.isEmpty() && (str.charAt(0) == prefix)); } /** * Test if the given {@code String} ends with the specified prefix character. * * @param str the {@code String} to check * @param suffix the prefix character to look for * @return true if the string ends with the specified suffix; otherwise false * @see java.lang.String#endsWith */ public static boolean endsWith(String str, char suffix) { return (str != null && !str.isEmpty() && (str.charAt(str.length() - 1) == suffix)); } /** * Replace all occurrences of a substring within a string with another string. * * @param str {@code String} to examine * @param search {@code String} to replace * @param replace {@code String} to insert * @return a {@code String} with the replacements */ public static String replace(String str, String search, String replace) { if (str == null || search == null || replace == null) { return str; } StringBuilder sb = new StringBuilder(); int searchLen = search.length(); int stringLen = str.length(); int oldIndex = 0; int index; while ((index = str.indexOf(search, oldIndex)) >= 0) { sb.append(str, oldIndex, index); sb.append(replace); oldIndex = index + searchLen; } if (oldIndex < stringLen) { sb.append(str, oldIndex, stringLen); } return sb.toString(); } /** * Replace all occurrences of a substring within a string with another string. * * @param str {@code String} to examine * @param search {@code String} array to replace * @param replace {@code String} array to insert * @return a {@code String} with the replacements */ public static String replace(String str, String[] search, String[] replace) { if (str == null || search == null || replace == null) { return str; } StringBuilder sb = new StringBuilder(str); int loop = (search.length <= replace.length) ? search.length : replace.length; int start = 0; int end; int searchLen; int replaceLen; for (int i = 0; i < loop; i++) { if (search[i] == null || replace[i] == null) { continue; } searchLen = search[i].length(); replaceLen = replace[i].length(); while (true) { if (sb.length() == 0) { break; } start = sb.indexOf(search[i], start + replaceLen); if (start == -1) { break; } end = start + searchLen; sb.replace(start, end, replace[i]); } } return sb.toString(); } /** * Returns an array of strings separated by the delimiter string. * * @param str the string to be separated * @param delim the delimiter * @return an array, containing the splitted strings */ public static String[] split(String str, String delim) { if (isEmpty(str)) { return new String[0]; } int cnt = search(str, delim); String[] item = new String[cnt + 1]; if (cnt == 0) { item[0] = str; return item; } int idx = 0; int pos1 = 0; int pos2 = str.indexOf(delim); int delimLen = delim.length(); while (pos2 >= 0) { item[idx++] = (pos1 > pos2 - 1) ? EMPTY : str.substring(pos1, pos2); pos1 = pos2 + delimLen; pos2 = str.indexOf(delim, pos1); } if (pos1 < str.length()) { item[idx] = str.substring(pos1); } if (item[cnt] == null) { item[cnt] = EMPTY; } return item; } /** * Returns an array of strings separated by the delimiter string. * * @param str the string to be separated * @param delim the delimiter * @param size the size of the array * @return an array, containing the splitted strings */ public static String[] split(String str, String delim, int size) { String[] arr1 = new String[size]; String[] arr2 = split(str, delim); for (int i = 0; i < arr1.length; i++) { if (i < arr2.length) { arr1[i] = arr2[i]; } else { arr1[i] = EMPTY; } } return arr1; } /** * Returns an array of strings separated by the delimiter string. * * @param str the string to be separated * @param delim the delimiter * @return an array, containing the splitted strings */ public static String[] split(String str, char delim) { if (isEmpty(str)) { return new String[0]; } int cnt = search(str, delim); String[] item = new String[cnt + 1]; if (cnt == 0) { item[0] = str; return item; } int idx = 0; int pos1 = 0; int pos2 = str.indexOf(delim); while (pos2 >= 0) { item[idx++] = (pos1 > pos2 - 1) ? EMPTY : str.substring(pos1, pos2); pos1 = pos2 + 1; pos2 = str.indexOf(delim, pos1); } if (pos1 < str.length()) { item[idx] = str.substring(pos1); } if (item[cnt] == null) { item[cnt] = EMPTY; } return item; } /** * Returns an array of strings separated by the delimiter string. * * @param str the string to be separated * @param delim the delimiter * @param size the size of the array * @return an array, containing the splitted strings */ public static String[] split(String str, char delim, int size) { String[] arr1 = new String[size]; String[] arr2 = split(str, delim); for (int i = 0; i < arr1.length; i++) { if (i < arr2.length) { arr1[i] = arr2[i]; } else { arr1[i] = EMPTY; } } return arr1; } /** * Returns the number of times the specified string was found * in the target string, or 0 if there is no specified string. * * @param str the target string * @param keyw the string to find * @return the number of times the specified string was found */ public static int search(String str, String keyw) { int strLen = str.length(); int keywLen = keyw.length(); int pos = 0; int cnt = 0; if (keywLen == 0) { return 0; } while ((pos = str.indexOf(keyw, pos)) != -1) { pos += keywLen; cnt++; if (pos >= strLen) { break; } } return cnt; } /** * Returns the number of times the specified string was found * in the target string, or 0 if there is no specified string. * When searching for the specified string, it is not case-sensitive. * * @param str the target string * @param keyw the string to find * @return the number of times the specified string was found */ public static int searchIgnoreCase(String str, String keyw) { return search(str.toLowerCase(), keyw.toLowerCase()); } /** * Returns the number of times the specified character was found * in the target string, or 0 if there is no specified character. * * @param chars the target string * @param c the character to find * @return the number of times the specified character was found */ public static int search(CharSequence chars, char c) { int count = 0; for (int i = 0; i < chars.length(); i++) { if (chars.charAt(i) == c) { count++; } } return count; } /** * Returns the number of times the specified character was found * in the target string, or 0 if there is no specified character. * When searching for the specified character, it is not case-sensitive. * * @param chars the target string * @param c the character to find * @return the number of times the specified character was found */ public static int searchIgnoreCase(CharSequence chars, char c) { int count = 0; char cl = Character.toLowerCase(c); for (int i = 0; i < chars.length(); i++) { if (Character.toLowerCase(chars.charAt(i)) == cl) { count++; } } return count; } /** * Tokenize the given {@code String} into a String array via a StringTokenizer. * * @param str the {@code String} to tokenize * @param delimiters the delimiter characters * @return an array of the tokens */ public static String[] tokenize(String str, String delimiters) { return tokenize(str, delimiters, false); } /** * Tokenize the given {@code String} into a {@code String} array via a {@code StringTokenizer}. * * @param str the String to tokenize * @param delimiters the delimiter characters * @param trim trim the tokens via String's trim * @return an array of the tokens */ public static String[] tokenize(String str, String delimiters, boolean trim) { if (str == null) { return new String[0]; } StringTokenizer st = new StringTokenizer(str, delimiters); List tokens = new ArrayList<>(); while (st.hasMoreTokens()) { tokens.add(trim ? st.nextToken().trim() : st.nextToken()); } return tokens.toArray(new String[0]); } /** * Convert a {@code String} array into a delimited {@code String} (e.g. CSV). *

Useful for {@code toString()} implementations. * * @param arr the array to display * @param delim the delimiter to use (typically a ",") * @return the delimited {@code String} */ public static String toDelimitedString(Object[] arr, String delim) { if (arr == null || arr.length == 0) { return EMPTY; } if (arr.length == 1) { return (arr[0] == null) ? EMPTY : arr[0].toString(); } StringBuilder sb = new StringBuilder(); for (int i = 0; i < arr.length; i++) { if (i > 0) { sb.append(delim); } sb.append(arr[i]); } return sb.toString(); } /** * Convert a {@code Collection} into a delimited {@code String} (e.g. CSV). *

Useful for {@code toString()} implementations. * * @param list the collection * @param delim the delimiter to use (typically a ",") * @return the delimited {@code String} */ public static String toDelimitedString(Collection list, String delim) { if (list == null || list.isEmpty()) { return EMPTY; } StringBuilder sb = new StringBuilder(); boolean first = true; for (Object o : list) { if (!first) { sb.append(delim); } sb.append(o); first = false; } return sb.toString(); } /** * Convert a {@code String} array into a delimited {@code String} * by a system-dependent line separator. * * @param arr the array to display * @return the delimited {@code String} */ public static String toLineDelimitedString(Object[] arr) { return toDelimitedString(arr, System.lineSeparator()); } /** * Convert a {@code Collection} into a delimited {@code String} * by a system-dependent line separator. * * @param list the collection * @return the delimited {@code String} */ public static String toLineDelimitedString(Collection list) { return toDelimitedString(list, System.lineSeparator()); } /** * Convert a comma delimited list (e.g., a row from a CSV file) into an * array of strings. * * @param str the input {@code String} * @return an array of strings, or the empty array in case of empty input */ public static String[] splitCommaDelimitedString(String str) { return tokenize(str, ",", true); } /** * Convert a {@code String} array into a comma delimited {@code String} * (i.e., CSV). * * @param arr the array to display * @return the delimited {@code String} */ public static String joinCommaDelimitedList(String[] arr) { return toDelimitedString(arr, ", "); } /** * Convert a {@code Collection} into a comma delimited {@code String} * (i.e., CSV). * * @param list the collection * @return the delimited {@code String} */ public static String joinCommaDelimitedList(Collection list) { return toDelimitedString(list, ", "); } /** * Parse the given {@code localeString} value into a {@link Locale}. *

This is the inverse operation of {@link Locale#toString Locale's toString}. * * @param localeString the locale {@code String}, following {@code Locale's} * {@code toString()} format ("en", "en_UK", etc); * also accepts spaces as separators, as an alternative to underscores * @return a corresponding {@code Locale} instance * @throws IllegalArgumentException in case of an invalid locale specification */ public static Locale parseLocaleString(String localeString) { String[] parts = tokenize(localeString, "_ ", false); String language = (parts.length > 0 ? parts[0] : EMPTY); String country = (parts.length > 1 ? parts[1] : EMPTY); validateLocalePart(language); validateLocalePart(country); String variant = EMPTY; if (parts.length > 2) { // There is definitely a variant, and it is everything after the country // code sans the separator between the country code and the variant. int endIndexOfCountryCode = localeString.indexOf(country, language.length()) + country.length(); // Strip off any leading '_' and whitespace, what's left is the variant. variant = trimLeadingWhitespace(localeString.substring(endIndexOfCountryCode)); if (variant.startsWith("_")) { variant = trimLeadingCharacter(variant, '_'); } } return (language.length() > 0 ? new Locale(language, country, variant) : null); } private static void validateLocalePart(String localePart) { for (int i = 0; i < localePart.length(); i++) { char ch = localePart.charAt(i); if (ch != '_' && ch != ' ' && !Character.isLetterOrDigit(ch)) { throw new IllegalArgumentException( "Locale part \"" + localePart + "\" contains invalid characters"); } } } /** * Determine the RFC 3066 compliant language tag, * as used for the HTTP "Accept-Language" header. * @param locale the Locale to transform to a language tag * @return the RFC 3066 compliant language tag as {@code String} */ public static String toLanguageTag(Locale locale) { return locale.getLanguage() + (hasText(locale.getCountry()) ? "-" + locale.getCountry() : EMPTY); } /** * Parse the given {@code timeZoneString} value into a {@link TimeZone}. * @param timeZoneString the time zone {@code String}, following {@link TimeZone#getTimeZone(String)} * but throwing {@link IllegalArgumentException} in case of an invalid time zone specification * @return a corresponding {@link TimeZone} instance * @throws IllegalArgumentException in case of an invalid time zone specification */ public static TimeZone parseTimeZoneString(String timeZoneString) { TimeZone timeZone = TimeZone.getTimeZone(timeZoneString); if ("GMT".equals(timeZone.getID()) && !timeZoneString.startsWith("GMT")) { // We don't want that GMT fallback... throw new IllegalArgumentException("Invalid time zone specification '" + timeZoneString + "'"); } return timeZone; } /** * Convert byte size into human friendly format. * * @param size the number of bytes * @return a human friendly byte size (includes units) */ public static String convertToHumanFriendlyByteSize(long size) { if (size < 1024) { return size + " B"; } int z = (63 - Long.numberOfLeadingZeros(size)) / 10; double d = (double)size / (1L << (z * 10)); String format = (d % 1.0 == 0) ? "%.0f %sB" : "%.1f %sB"; return String.format(format, d, " KMGTPE".charAt(z)); } /** * Convert byte size into machine friendly format. * * @param size the human friendly byte size (includes units) * @return a number of bytes * @throws NumberFormatException if failed parse given size */ @SuppressWarnings("fallthrough") public static long convertToMachineFriendlyByteSize(String size) { double d; try { d = Double.parseDouble(size.replaceAll("[GMK]?[B]?$", "")); } catch (NumberFormatException e) { String msg = "Size must be specified as bytes (B), " + "kilobytes (KB), megabytes (MB), gigabytes (GB). " + "E.g. 1024, 1KB, 10M, 10MB, 100G, 100GB"; throw new NumberFormatException(msg + " " + e.getMessage()); } long l = Math.round(d * 1024 * 1024 * 1024L); int index = Math.max(0, size.length() - (size.endsWith("B") ? 2 : 1)); switch (size.charAt(index)) { default: l /= 1024; case 'K': l /= 1024; case 'M': l /= 1024; case 'G': return l; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy