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

com.helger.commons.string.StringHelper Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2014-2024 Philip Helger (www.helger.com)
 * philip[at]helger[dot]com
 *
 * 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.helger.commons.string;

import java.io.IOException;
import java.io.Writer;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.CharacterCodingException;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CharsetEncoder;
import java.nio.charset.CoderResult;
import java.nio.charset.CodingErrorAction;
import java.util.Arrays;
import java.util.Collection;
import java.util.Locale;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.IntConsumer;
import java.util.function.Predicate;
import java.util.function.Supplier;

import javax.annotation.CheckForSigned;
import javax.annotation.CheckReturnValue;
import javax.annotation.Nonnegative;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import javax.annotation.concurrent.Immutable;

import com.helger.commons.CGlobal;
import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.CodingStyleguideUnaware;
import com.helger.commons.annotation.Nonempty;
import com.helger.commons.annotation.PresentForCodeCoverage;
import com.helger.commons.annotation.ReturnsMutableCopy;
import com.helger.commons.annotation.ReturnsMutableObject;
import com.helger.commons.builder.IBuilder;
import com.helger.commons.collection.ArrayHelper;
import com.helger.commons.collection.impl.CommonsArrayList;
import com.helger.commons.collection.impl.CommonsHashSet;
import com.helger.commons.collection.impl.CommonsLinkedHashMap;
import com.helger.commons.collection.impl.CommonsLinkedHashSet;
import com.helger.commons.collection.impl.CommonsTreeSet;
import com.helger.commons.collection.impl.ICommonsList;
import com.helger.commons.collection.impl.ICommonsMap;
import com.helger.commons.functional.ICharConsumer;
import com.helger.commons.functional.ICharPredicate;
import com.helger.commons.functional.Predicates;
import com.helger.commons.math.MathHelper;

/**
 * Generic string transformation and helper methods. If you need to modify a
 * string, start looking in this class.
 *
 * @author Philip Helger
 */
@Immutable
public final class StringHelper
{
  /**
   * The constant to be returned if an String.indexOf call did not find a match!
   */
  public static final int STRING_NOT_FOUND = -1;

  private static final int [] SIZE_TABLE_INT = { 9,
                                                 99,
                                                 999,
                                                 9999,
                                                 99999,
                                                 999999,
                                                 9999999,
                                                 99999999,
                                                 999999999,
                                                 Integer.MAX_VALUE };
  private static final long [] SIZE_TABLE_LONG = { 9L,
                                                   99L,
                                                   999L,
                                                   9999L,
                                                   99999L,
                                                   999999L,
                                                   9999999L,
                                                   99999999L,
                                                   999999999L,
                                                   9999999999L,
                                                   99999999999L,
                                                   999999999999L,
                                                   9999999999999L,
                                                   99999999999999L,
                                                   999999999999999L,
                                                   9999999999999999L,
                                                   99999999999999999L,
                                                   999999999999999999L,
                                                   Long.MAX_VALUE };
  @PresentForCodeCoverage
  private static final StringHelper INSTANCE = new StringHelper ();

  private StringHelper ()
  {}

  /**
   * Check if the string is null or empty.
   *
   * @param aCS
   *        The character sequence to check. May be null.
   * @return true if the string is null or empty,
   *         false otherwise
   * @since 10.1.8
   */
  public static boolean isEmpty (@Nullable final CharSequence aCS)
  {
    return aCS == null || aCS.length () == 0;
  }

  /**
   * Check if the string is null or empty.
   *
   * @param sStr
   *        The string to check. May be null.
   * @return true if the string is null or empty,
   *         false otherwise
   * @since 10.1.8
   */
  public static boolean isEmpty (@Nullable final String sStr)
  {
    return sStr == null || sStr.isEmpty ();
  }

  /**
   * Check if the string is null or empty after trimming.
   *
   * @param s
   *        The string to check. May be null.
   * @return true if the string is null or empty or
   *         consists only of whitespaces, false otherwise
   * @since 10.1.8
   */
  public static boolean isEmptyAfterTrim (@Nullable final String s)
  {
    return s == null || s.trim ().isEmpty ();
  }

  /**
   * Check if the string contains any char.
   *
   * @param aCS
   *        The character sequence to check. May be null.
   * @return true if the string contains at least one,
   *         false otherwise
   * @since 10.1.8
   */
  public static boolean isNotEmpty (@Nullable final CharSequence aCS)
  {
    return aCS != null && aCS.length () > 0;
  }

  /**
   * Check if the string contains any char.
   *
   * @param sStr
   *        The string to check. May be null.
   * @return true if the string contains at least one char,
   *         false otherwise
   * @since 10.1.8
   */
  public static boolean isNotEmpty (@Nullable final String sStr)
  {
    return sStr != null && !sStr.isEmpty ();
  }

  /**
   * Check if the string neither null nor empty after trimming.
   *
   * @param s
   *        The string to check. May be null.
   * @return true if the string is neither null nor
   *         empty nor consists only of whitespaces, false
   *         otherwise
   * @since 10.1.8
   */
  public static boolean isNotEmptyAfterTrim (@Nullable final String s)
  {
    return s != null && !s.trim ().isEmpty ();
  }

  /**
   * Check if the string is null or empty.
   *
   * @param aCS
   *        The character sequence to check. May be null.
   * @return true if the string is null or empty,
   *         false otherwise
   */
  public static boolean hasNoText (@Nullable final CharSequence aCS)
  {
    return isEmpty (aCS);
  }

  /**
   * Check if the string is null or empty.
   *
   * @param sStr
   *        The string to check. May be null.
   * @return true if the string is null or empty,
   *         false otherwise
   */
  public static boolean hasNoText (@Nullable final String sStr)
  {
    return isEmpty (sStr);
  }

  /**
   * Check if the string is null or empty after trimming.
   *
   * @param s
   *        The string to check. May be null.
   * @return true if the string is null or empty or
   *         consists only of whitespaces, false otherwise
   */
  public static boolean hasNoTextAfterTrim (@Nullable final String s)
  {
    return isEmptyAfterTrim (s);
  }

  /**
   * Check if the string contains any char.
   *
   * @param aCS
   *        The character sequence to check. May be null.
   * @return true if the string contains at least one,
   *         false otherwise
   */
  public static boolean hasText (@Nullable final CharSequence aCS)
  {
    return isNotEmpty (aCS);
  }

  /**
   * Check if the string contains any char.
   *
   * @param sStr
   *        The string to check. May be null.
   * @return true if the string contains at least one char,
   *         false otherwise
   */
  public static boolean hasText (@Nullable final String sStr)
  {
    return isNotEmpty (sStr);
  }

  /**
   * Check if the string neither null nor empty after trimming.
   *
   * @param s
   *        The string to check. May be null.
   * @return true if the string is neither null nor
   *         empty nor consists only of whitespaces, false
   *         otherwise
   */
  public static boolean hasTextAfterTrim (@Nullable final String s)
  {
    return isNotEmptyAfterTrim (s);
  }

  /**
   * Check if the passed {@link CharSequence} contains any character matching
   * the provided filter.
   *
   * @param aCS
   *        String to check. May be null.
   * @param aFilter
   *        The filter to use. May be null.
   * @return true if the filter is null and the string
   *         is not empty. true if the filter is not
   *         null and at least one character of the string matches
   *         the filter. false otherwise.
   * @since 9.1.7
   */
  public static boolean containsAny (@Nullable final CharSequence aCS, @Nullable final ICharPredicate aFilter)
  {
    final int nLen = getLength (aCS);
    if (aFilter == null)
      return nLen > 0;

    if (nLen > 0)
      for (int i = 0; i < nLen; ++i)
        if (aFilter.test (aCS.charAt (i)))
          return true;
    return false;
  }

  /**
   * Check if the passed {@link String} contains any character matching the
   * provided filter.
   *
   * @param sStr
   *        String to check. May be null.
   * @param aFilter
   *        The filter to use. May be null.
   * @return true if the filter is null and the string
   *         is not empty. true if the filter is not
   *         null and at least one character of the string matches
   *         the filter. false otherwise.
   * @since 9.1.7
   */
  public static boolean containsAny (@Nullable final String sStr, @Nullable final ICharPredicate aFilter)
  {
    final int nLen = getLength (sStr);
    if (aFilter == null)
      return nLen > 0;

    if (nLen > 0)
      for (final char c : sStr.toCharArray ())
        if (aFilter.test (c))
          return true;
    return false;
  }

  /**
   * Check if the passed {@link CharSequence} contains no character matching the
   * provided filter.
   *
   * @param aCS
   *        String to check. May be null.
   * @param aFilter
   *        The filter to use. May be null.
   * @return true if the filter is null and the string
   *         is empty. true if the filter is not null
   *         and no character of the string matches the filter.
   *         false otherwise.
   * @since 9.1.7
   */
  public static boolean containsNone (@Nullable final CharSequence aCS, @Nullable final ICharPredicate aFilter)
  {
    final int nLen = getLength (aCS);
    if (aFilter == null)
      return nLen == 0;

    for (int i = 0; i < nLen; ++i)
      if (aFilter.test (aCS.charAt (i)))
        return false;
    return true;
  }

  /**
   * Check if the passed {@link String} contains no character matching the
   * provided filter.
   *
   * @param sStr
   *        String to check. May be null.
   * @param aFilter
   *        The filter to use. May be null.
   * @return true if the filter is null and the string
   *         is empty. true if the filter is not null
   *         and no character of the string matches the filter.
   *         false otherwise.
   * @since 9.1.7
   */
  public static boolean containsNone (@Nullable final String sStr, @Nullable final ICharPredicate aFilter)
  {
    final int nLen = getLength (sStr);
    if (aFilter == null)
      return nLen == 0;

    if (nLen > 0)
      for (final char c : sStr.toCharArray ())
        if (aFilter.test (c))
          return false;
    return true;
  }

  /**
   * Check if the passed {@link CharSequence} contains only characters matching
   * the provided filter.
   *
   * @param aCS
   *        String to check. May be null.
   * @param aFilter
   *        The filter to use. May be null.
   * @return true if the filter is null and the string
   *         is not empty. true if the filter is not
   *         null and the string has at least one character and all
   *         characters of the string match the filter. false
   *         otherwise.
   * @since 9.1.7
   */
  public static boolean containsOnly (@Nullable final CharSequence aCS, @Nullable final ICharPredicate aFilter)
  {
    final int nLen = getLength (aCS);
    if (nLen == 0)
      return false;

    if (aFilter == null)
      return true;

    for (int i = 0; i < nLen; ++i)
      if (!aFilter.test (aCS.charAt (i)))
        return false;
    return true;
  }

  /**
   * Check if the passed {@link String} contains only characters matching the
   * provided filter.
   *
   * @param sStr
   *        String to check. May be null.
   * @param aFilter
   *        The filter to use. May be null.
   * @return true if the filter is null and the string
   *         is not empty. true if the filter is not
   *         null and the string has at least one character and all
   *         characters of the string match the filter. false
   *         otherwise.
   * @since 9.1.7
   */
  public static boolean containsOnly (@Nullable final String sStr, @Nullable final ICharPredicate aFilter)
  {
    final int nLen = getLength (sStr);
    if (nLen == 0)
      return false;

    if (aFilter == null)
      return true;

    for (final char c : sStr.toCharArray ())
      if (!aFilter.test (c))
        return false;
    return true;
  }

  /**
   * Check if the passed character sequence is only whitespace or not.
   *
   * @param s
   *        The character sequence to be checked. May be null.
   * @return true if the passed sequence is empty or if only
   *         whitespace characters are contained.
   * @see Character#isWhitespace(char)
   */
  public static boolean isAllWhitespace (@Nullable final CharSequence s)
  {
    return containsOnly (s, Character::isWhitespace);
  }

  @Nullable
  public static String getLeadingZero (@Nullable final Byte aValue, final int nChars)
  {
    return aValue == null ? null : getLeadingZero (aValue.byteValue (), nChars);
  }

  @Nullable
  public static String getLeadingZero (@Nullable final Integer aValue, final int nChars)
  {
    return aValue == null ? null : getLeadingZero (aValue.longValue (), nChars);
  }

  @Nullable
  public static String getLeadingZero (@Nullable final Long aValue, final int nChars)
  {
    return aValue == null ? null : getLeadingZero (aValue.longValue (), nChars);
  }

  @Nullable
  public static String getLeadingZero (@Nullable final Short aValue, final int nChars)
  {
    return aValue == null ? null : getLeadingZero (aValue.shortValue (), nChars);
  }

  @Nonnull
  public static String getLeadingZero (final int nValue, final int nChars)
  {
    final boolean bNeg = nValue < 0;
    final String sValue = Integer.toString (MathHelper.abs (nValue));
    if (sValue.length () >= nChars)
      return bNeg ? '-' + sValue : sValue;

    // prepend '0's
    final StringBuilder aSB = new StringBuilder ((bNeg ? 1 : 0) + nChars);
    if (bNeg)
      aSB.append ('-');
    for (int i = 0; i < nChars - sValue.length (); ++i)
      aSB.append ('0');
    return aSB.append (sValue).toString ();
  }

  @Nonnull
  public static String getLeadingZero (final long nValue, final int nChars)
  {
    final boolean bNeg = nValue < 0;
    final String sValue = Long.toString (MathHelper.abs (nValue));
    if (sValue.length () >= nChars)
      return bNeg ? '-' + sValue : sValue;

    // prepend '0's
    final StringBuilder aSB = new StringBuilder ((bNeg ? 1 : 0) + nChars);
    if (bNeg)
      aSB.append ('-');
    for (int i = 0; i < nChars - sValue.length (); ++i)
      aSB.append ('0');
    return aSB.append (sValue).toString ();
  }

  @Nonnull
  public static String getLeadingZero (@Nonnull final String sValue, final int nChars)
  {
    return getWithLeading (sValue, nChars, '0');
  }

  /**
   * Get the result string with at least the desired length, and fill the lead
   * or trail with the provided char
   *
   * @param sSrc
   *        Source string. May be null or empty.
   * @param nMinLen
   *        The destination minimum length. Should be > 0 to have an impact.
   * @param cGap
   *        The character to use to fill the gap
   * @param bLeading
   *        true to fill at the front (like "00" in "007") or at
   *        the end (like "00" in "700")
   * @return Never null.
   */
  @Nonnull
  private static String _getWithLeadingOrTrailing (@Nullable final String sSrc,
                                                   @Nonnegative final int nMinLen,
                                                   final char cGap,
                                                   final boolean bLeading)
  {
    if (nMinLen <= 0)
    {
      // Requested length is too short - return as is
      return getNotNull (sSrc, "");
    }

    final int nSrcLen = getLength (sSrc);
    if (nSrcLen == 0)
    {
      // Input string is empty
      return getRepeated (cGap, nMinLen);
    }

    final int nCharsToAdd = nMinLen - nSrcLen;
    if (nCharsToAdd <= 0)
    {
      // Input string is already longer than requested minimum length
      return sSrc;
    }

    final StringBuilder aSB = new StringBuilder (nMinLen);
    if (!bLeading)
      aSB.append (sSrc);
    for (int i = 0; i < nCharsToAdd; ++i)
      aSB.append (cGap);
    if (bLeading)
      aSB.append (sSrc);
    return aSB.toString ();
  }

  /**
   * Get a string that is filled at the beginning with the passed character
   * until the minimum length is reached. If the input string is empty, the
   * result is a string with the provided len only consisting of the passed
   * characters. If the input String is longer than the provided length, it is
   * returned unchanged.
   *
   * @param sSrc
   *        Source string. May be null.
   * @param nMinLen
   *        Minimum length. Should be > 0.
   * @param cFront
   *        The character to be used at the beginning
   * @return A non-null string that has at least nLen chars
   */
  @Nonnull
  public static String getWithLeading (@Nullable final String sSrc, @Nonnegative final int nMinLen, final char cFront)
  {
    return _getWithLeadingOrTrailing (sSrc, nMinLen, cFront, true);
  }

  /**
   * Get a string that is filled at the beginning with the passed character
   * until the minimum length is reached. If the input String is longer than the
   * provided length, it is returned unchanged.
   *
   * @param nValue
   *        Source string. May be null.
   * @param nMinLen
   *        Minimum length. Should be > 0.
   * @param cFront
   *        The character to be used at the beginning
   * @return A non-null string that has at least nLen chars
   * @see #getWithLeading(String, int, char)
   */
  @Nonnull
  public static String getWithLeading (final int nValue, @Nonnegative final int nMinLen, final char cFront)
  {
    return _getWithLeadingOrTrailing (Integer.toString (nValue), nMinLen, cFront, true);
  }

  /**
   * Get a string that is filled at the beginning with the passed character
   * until the minimum length is reached. If the input String is longer than the
   * provided length, it is returned unchanged.
   *
   * @param nValue
   *        Source string. May be null.
   * @param nMinLen
   *        Minimum length. Should be > 0.
   * @param cFront
   *        The character to be used at the beginning
   * @return A non-null string that has at least nLen chars
   * @see #getWithLeading(String, int, char)
   */
  @Nonnull
  public static String getWithLeading (final long nValue, @Nonnegative final int nMinLen, final char cFront)
  {
    return _getWithLeadingOrTrailing (Long.toString (nValue), nMinLen, cFront, true);
  }

  /**
   * Get a string that is filled at the end with the passed character until the
   * minimum length is reached. If the input string is empty, the result is a
   * string with the provided len only consisting of the passed characters. If
   * the input String is longer than the provided length, it is returned
   * unchanged.
   *
   * @param sSrc
   *        Source string. May be null.
   * @param nMinLen
   *        Minimum length. Should be > 0.
   * @param cEnd
   *        The character to be used at the end
   * @return A non-null string that has at least nLen chars
   */
  @Nonnull
  public static String getWithTrailing (@Nullable final String sSrc, @Nonnegative final int nMinLen, final char cEnd)
  {
    return _getWithLeadingOrTrailing (sSrc, nMinLen, cEnd, false);
  }

  /**
   * Get the matching hex digit as a lower case character.
   *
   * @param n
   *        The value to get the hex digit from. Must be between 0 and 15.
   * @return The hex character (one of 0-9 or a-f), or '\0' if the value could
   *         not be converted
   */
  public static char getHexChar (final int n)
  {
    return Character.forDigit (n, CGlobal.HEX_RADIX);
  }

  /**
   * Get the matching hex digit as an upper case character.
   *
   * @param n
   *        The value to get the hex digit from. Must be between 0 and 15.
   * @return The hex character (one of 0-9 or A-F), or '\0' if the value could
   *         not be converted
   */
  public static char getHexCharUpperCase (final int n)
  {
    return Character.toUpperCase (getHexChar (n));
  }

  /**
   * Convert a string to a byte array and than to a hexadecimal encoded string.
   *
   * @param sInput
   *        The source string. May not be null.
   * @param aCharset
   *        The charset to use. May not be null.
   * @return The String representation of the byte array of the string.
   */
  @Nonnull
  public static String getHexEncoded (@Nonnull final String sInput, @Nonnull final Charset aCharset)
  {
    ValueEnforcer.notNull (sInput, "Input");
    ValueEnforcer.notNull (aCharset, "Charset");

    return getHexEncoded (sInput.getBytes (aCharset));
  }

  /**
   * Convert a byte array to a hexadecimal encoded string.
   *
   * @param aInput
   *        The byte array to be converted to a String. May not be
   *        null.
   * @return The String representation of the byte array.
   */
  @Nonnull
  public static String getHexEncoded (@Nonnull final byte [] aInput)
  {
    ValueEnforcer.notNull (aInput, "Input");

    return getHexEncoded (aInput, 0, aInput.length);
  }

  /**
   * Convert a byte array to a hexadecimal encoded string.
   *
   * @param aInput
   *        The byte array to be converted to a String. May not be
   *        null.
   * @param nOfs
   *        Byte array offset
   * @param nLen
   *        Number of bytes to encode
   * @return The String representation of the byte array.
   */
  @Nonnull
  public static String getHexEncoded (@Nonnull final byte [] aInput, final int nOfs, final int nLen)
  {
    ValueEnforcer.isArrayOfsLen (aInput, nOfs, nLen);

    final StringBuilder aSB = new StringBuilder (nLen * 2);
    for (int i = nOfs; i < (nOfs + nLen); ++i)
    {
      final byte b = aInput[i];
      final char c1 = getHexChar ((b & 0xf0) >> 4);
      final char c2 = getHexChar (b & 0x0f);
      aSB.append (c1).append (c2);
    }
    return aSB.toString ();
  }

  /**
   * Get the decimal value of the passed hex character
   *
   * @param c
   *        The hex char to convert
   * @return A value between 0 and 15, or -1 if the input character is not a hex
   *         char!
   */
  @CheckForSigned
  public static int getHexValue (@Nonnegative final char c)
  {
    return Character.digit (c, CGlobal.HEX_RADIX);
  }

  /**
   * @param cHigh
   *        High hex part
   * @param cLow
   *        Low hex part
   * @return A value between 0 and 255, or -1 if any input character is not a
   *         hex char!
   */
  public static int getHexByte (@Nonnegative final char cHigh, @Nonnegative final char cLow)
  {
    final int nHex1 = getHexValue (cHigh);
    final int nHex2 = getHexValue (cLow);
    return nHex1 < 0 || nHex2 < 0 ? -1 : (nHex1 << 4) | nHex2;
  }

  @Nonnull
  @ReturnsMutableCopy
  public static byte [] getHexDecoded (@Nonnull final String sInput)
  {
    ValueEnforcer.notNull (sInput, "Input");

    return getHexDecoded (sInput.toCharArray (), 0, sInput.length ());
  }

  @Nonnull
  @ReturnsMutableCopy
  public static byte [] getHexDecoded (@Nonnull final char [] aInput)
  {
    ValueEnforcer.notNull (aInput, "Input");

    return getHexDecoded (aInput, 0, aInput.length);
  }

  @Nonnull
  @ReturnsMutableCopy
  public static byte [] getHexDecoded (@Nonnull final char [] aInput,
                                       @Nonnegative final int nOfs,
                                       @Nonnegative final int nLen)
  {
    ValueEnforcer.isArrayOfsLen (aInput, nOfs, nLen);
    ValueEnforcer.isTrue ((nLen % 2) == 0, () -> "Passed chars have no even length: " + nLen);

    final byte [] ret = new byte [nLen / 2];
    int nRetIdx = 0;
    for (int i = 0; i < nLen; i += 2)
    {
      final char c0 = aInput[nOfs + i];
      final char c1 = aInput[nOfs + i + 1];
      final int nHexByte = getHexByte (c0, c1);
      if (nHexByte == -1)
        throw new IllegalArgumentException ("Failed to convert '" + c0 + "' and '" + c1 + "' to a hex value!");
      ret[nRetIdx++] = (byte) nHexByte;
    }
    return ret;
  }

  @Nonnull
  public static String getHexString (final byte nValue)
  {
    // Bytes are always handled unsigned
    return Integer.toString (nValue & 0xff, CGlobal.HEX_RADIX);
  }

  @Nonnull
  public static String getHexStringLeadingZero (final byte nValue, final int nDigits)
  {
    return getLeadingZero (getHexString (nValue), nDigits);
  }

  @Nonnull
  public static String getHexStringLeadingZero2 (final byte nValue)
  {
    final String ret = getHexString (nValue);
    return ret.length () >= 2 ? ret : '0' + ret;
  }

  @Nonnull
  public static String getHexString (final int nValue)
  {
    return Integer.toString (nValue, CGlobal.HEX_RADIX);
  }

  @Nonnull
  public static String getHexStringLeadingZero (final int nValue, final int nDigits)
  {
    if (nValue < 0)
      return "-" + getLeadingZero (getHexString (-nValue), nDigits - 1);
    return getLeadingZero (getHexString (nValue), nDigits);
  }

  @Nonnull
  public static String getHexString (final long nValue)
  {
    return Long.toString (nValue, CGlobal.HEX_RADIX);
  }

  @Nonnull
  public static String getHexStringLeadingZero (final long nValue, final int nDigits)
  {
    if (nValue < 0)
      return "-" + getLeadingZero (getHexString (-nValue), nDigits - 1);
    return getLeadingZero (getHexString (nValue), nDigits);
  }

  @Nonnull
  public static String getHexString (final short nValue)
  {
    return Integer.toString (nValue & 0xffff, CGlobal.HEX_RADIX);
  }

  @Nonnull
  public static String getHexStringLeadingZero (final short nValue, final int nDigits)
  {
    // Short are always handled unsigned
    return getLeadingZero (getHexString (nValue), nDigits);
  }

  /**
   * Get the number of leading white spaces according to
   * {@link Character#isWhitespace(char)}
   *
   * @param s
   *        The string to be parsed. May be null.
   * @return Always ≥ 0.
   */
  @Nonnegative
  public static int getLeadingWhitespaceCount (@Nullable final String s)
  {
    int ret = 0;
    if (s != null)
    {
      final int nMax = s.length ();
      while (ret < nMax && Character.isWhitespace (s.charAt (ret)))
        ++ret;
    }
    return ret;
  }

  /**
   * Get the number of trailing white spaces according to
   * {@link Character#isWhitespace(char)}
   *
   * @param s
   *        The string to be parsed. May be null.
   * @return Always ≥ 0.
   */
  @Nonnegative
  public static int getTrailingWhitespaceCount (@Nullable final String s)
  {
    int ret = 0;
    if (s != null)
    {
      int nLast = s.length () - 1;
      while (nLast >= 0 && Character.isWhitespace (s.charAt (nLast)))
      {
        ++ret;
        --nLast;
      }
    }
    return ret;
  }

  /**
   * Get the number of specified chars, the passed string starts with.
   *
   * @param s
   *        The string to be parsed. May be null.
   * @param c
   *        The char to be searched.
   * @return Always ≥ 0.
   */
  @Nonnegative
  public static int getLeadingCharCount (@Nullable final String s, final char c)
  {
    int ret = 0;
    if (s != null)
    {
      final int nMax = s.length ();
      while (ret < nMax && s.charAt (ret) == c)
        ++ret;
    }
    return ret;
  }

  /**
   * Get the number of specified chars, the passed string ends with.
   *
   * @param s
   *        The string to be parsed. May be null.
   * @param c
   *        The char to be searched.
   * @return Always ≥ 0.
   */
  @Nonnegative
  public static int getTrailingCharCount (@Nullable final String s, final char c)
  {
    int ret = 0;
    if (s != null)
    {
      int nLast = s.length () - 1;
      while (nLast >= 0 && s.charAt (nLast) == c)
      {
        ++ret;
        --nLast;
      }
    }
    return ret;
  }

  /**
   * A simple builder that allows to implode collections of arguments with a lot
   * of customization. It used by all the "getImploded*" overloads and fulfills
   * the requests of other use cases as well.
   *
   * @author Philip Helger
   * @since 10.0.0
   */
  public static class ImploderBuilder implements IBuilder 
  {
    private ICommonsList  m_aSource;
    private String m_sSeparator;
    private int m_nOffset = 0;
    private int m_nLength = -1;
    private Predicate  m_aFilter;

    public ImploderBuilder ()
    {}

    @Nullable
    private static String _valueOf (@Nullable final Object o)
    {
      return o == null ? null : o.toString ();
    }

    @Nonnull
    public ImploderBuilder source (@Nullable final Iterable  a)
    {
      return source (a, ImploderBuilder::_valueOf);
    }

    @Nonnull
    public  ImploderBuilder source (@Nullable final Iterable  a,
                                       @Nonnull final Function  aMapper)
    {
      ValueEnforcer.notNull (aMapper, "Mapper");
      m_aSource = a == null ? null : new CommonsArrayList <> (a, aMapper);
      return this;
    }

    @Nonnull
    public ImploderBuilder source (@Nullable final String... a)
    {
      m_aSource = a == null ? null : new CommonsArrayList <> (a);
      return this;
    }

    @Nonnull
    @SafeVarargs
    public final  ImploderBuilder source (@Nullable final T... a)
    {
      return source (a, ImploderBuilder::_valueOf);
    }

    @Nonnull
    public  ImploderBuilder source (@Nullable final T [] a, @Nonnull final Function  aMapper)
    {
      ValueEnforcer.notNull (aMapper, "Mapper");
      m_aSource = a == null ? null : new CommonsArrayList <> (a, aMapper);
      return this;
    }

    @Nonnull
    public ImploderBuilder separator (final char c)
    {
      return separator (Character.toString (c));
    }

    @Nonnull
    public ImploderBuilder separator (@Nullable final String s)
    {
      m_sSeparator = s;
      return this;
    }

    /**
     * Set an offset of the source from which to start. Only values > 0 have
     * an effect.
     *
     * @param n
     *        The offset to use
     * @return this for chaining
     */
    @Nonnull
    public ImploderBuilder offset (final int n)
    {
      m_nOffset = n;
      return this;
    }

    /**
     * Set the number of source items to iterate, depending on the source
     * offset. By default all elements from start to end are used. Only values
     * > 0 have an effect.
     *
     * @param n
     *        The length to use
     * @return this for chaining
     */
    @Nonnull
    public ImploderBuilder length (final int n)
    {
      m_nLength = n;
      return this;
    }

    @Nonnull
    public ImploderBuilder filterNonEmpty ()
    {
      return filter (StringHelper::hasText);
    }

    @Nonnull
    public ImploderBuilder filter (@Nullable final Predicate  a)
    {
      m_aFilter = a;
      return this;
    }

    @Nonnull
    public String build ()
    {
      final ICommonsList  aSource = m_aSource;
      if (aSource == null || aSource.isEmpty () || m_nLength == 0)
        return "";

      final StringBuilder aSB = new StringBuilder ();

      // Avoid constant this access for speed
      final String sSep = m_sSeparator;
      final Predicate  aFilter = m_aFilter == null ? Predicates.all () : m_aFilter;

      if (m_nOffset <= 0 && m_nLength < 0)
      {
        // Iterate all
        int nElementsAdded = 0;
        for (final String sElement : aSource)
        {
          if (aFilter.test (sElement))
          {
            if (nElementsAdded > 0 && sSep != null)
              aSB.append (sSep);
            aSB.append (sElement);
            nElementsAdded++;
          }
        }
      }
      else
      {
        // Start as early as possible
        final int nStart = Math.max (0, m_nOffset);

        // End of iteration?
        int nEnd = aSource.size ();
        if (m_nLength > 0)
          nEnd = Math.min (nEnd, nStart + m_nLength);

        int nElementsAdded = 0;
        for (int i = nStart; i < nEnd; ++i)
        {
          final String sElement = aSource.get (i);
          if (aFilter.test (sElement))
          {
            if (nElementsAdded > 0 && sSep != null)
              aSB.append (sSep);
            aSB.append (sElement);
            nElementsAdded++;
          }
        }
      }
      return aSB.toString ();
    }
  }

  /**
   * @return A new {@link ImploderBuilder}.
   * @since 10.0.0
   */
  @Nonnull
  public static ImploderBuilder imploder ()
  {
    return new ImploderBuilder ();
  }

  /**
   * Get a concatenated String from all elements of the passed container,
   * without a separator. Even null elements are added.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImploded (@Nullable final Iterable  aElements)
  {
    return imploder ().source (aElements).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed container,
   * without a separator. Even null elements are added.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Iterable element type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nullable final Iterable  aElements,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed container,
   * separated by the specified separator string. Even null
   * elements are added.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImploded (@Nonnull final String sSep, @Nullable final Iterable  aElements)
  {
    return imploder ().source (aElements).separator (sSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed container,
   * separated by the specified separator char. Even null elements
   * are added.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImploded (final char cSep, @Nullable final Iterable  aElements)
  {
    return imploder ().source (aElements).separator (cSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed container,
   * separated by the specified separator string. Even null
   * elements are added.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The element type of the collection to be imploded
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nonnull final String sSep,
                                                        @Nullable final Iterable  aElements,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (sSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed container,
   * separated by the specified separator char. Even null elements
   * are added.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The element type of the collection to be imploded
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (final char cSep,
                                                        @Nullable final Iterable  aElements,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (cSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, without a
   * separator.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   */
  @Nonnull
  @SafeVarargs
  public static  String getImploded (@Nullable final ELEMENTTYPE... aElements)
  {
    return imploder ().source (aElements).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, without a
   * separator.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   */
  @Nonnull
  public static  String getImploded (@Nullable final ELEMENTTYPE [] aElements,
                                                  @Nonnegative final int nOfs,
                                                  @Nonnegative final int nLen)
  {
    return imploder ().source (aElements).offset (nOfs).length (nLen).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, without a
   * separator.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nullable final ELEMENTTYPE [] aElements,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, without a
   * separator.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nullable final ELEMENTTYPE [] aElements,
                                                        @Nonnegative final int nOfs,
                                                        @Nonnegative final int nLen,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).offset (nOfs).length (nLen).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   */
  @Nonnull
  @SafeVarargs
  public static  String getImploded (@Nonnull final String sSep, @Nullable final ELEMENTTYPE... aElements)
  {
    return imploder ().source (aElements).separator (sSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   */
  @Nonnull
  @SafeVarargs
  public static  String getImploded (final char cSep, @Nullable final ELEMENTTYPE... aElements)
  {
    return imploder ().source (aElements).separator (cSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   */
  @Nonnull
  public static  String getImploded (@Nonnull final String sSep,
                                                  @Nullable final ELEMENTTYPE [] aElements,
                                                  @Nonnegative final int nOfs,
                                                  @Nonnegative final int nLen)
  {
    return imploder ().source (aElements).offset (nOfs).length (nLen).separator (sSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nonnull final String sSep,
                                                        @Nullable final ELEMENTTYPE [] aElements,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (sSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (final char cSep,
                                                        @Nullable final ELEMENTTYPE [] aElements,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (cSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (final char cSep,
                                                        @Nullable final ELEMENTTYPE [] aElements,
                                                        @Nonnegative final int nOfs,
                                                        @Nonnegative final int nLen,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).offset (nOfs).length (nLen).separator (cSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nonnull final String sSep,
                                                        @Nullable final ELEMENTTYPE [] aElements,
                                                        @Nonnegative final int nOfs,
                                                        @Nonnegative final int nLen,
                                                        @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).offset (nOfs).length (nLen).separator (sSep).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @return The concatenated string.
   * @param 
   *        The type of elements to be imploded.
   */
  @Nonnull
  public static  String getImploded (final char cSep,
                                                  @Nullable final ELEMENTTYPE [] aElements,
                                                  @Nonnegative final int nOfs,
                                                  @Nonnegative final int nLen)
  {
    return imploder ().source (aElements).offset (nOfs).length (nLen).separator (cSep).build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container without a separator string. This the very
   * generic version of {@link #getConcatenatedOnDemand(String, String)} for an
   * arbitrary number of elements.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (@Nullable final Iterable  aElements)
  {
    return imploder ().source (aElements).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container without a separator string. This the very
   * generic version of {@link #getConcatenatedOnDemand(String, String)} for an
   * arbitrary number of elements.
   *
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Iterable element type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (@Nullable final Iterable  aElements,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container without a separator string. This the very
   * generic version of {@link #getConcatenatedOnDemand(String, String)} for an
   * arbitrary number of elements.
   *
   * @param aElements
   *        The array to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Array component type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (@Nullable final ELEMENTTYPE [] aElements,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container, separated by the specified separator
   * string. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (@Nonnull final String sSep, @Nullable final Iterable  aElements)
  {
    return imploder ().source (aElements).separator (sSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container, separated by the specified separator
   * char. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (final char cSep, @Nullable final Iterable  aElements)
  {
    return imploder ().source (aElements).separator (cSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container, separated by the specified separator
   * string. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The element type of the collection to be imploded
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (@Nonnull final String sSep,
                                                                @Nullable final Iterable  aElements,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (sSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all non-null and non empty
   * elements of the passed container, separated by the specified separator
   * char. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        The element type of the collection to be imploded
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (final char cSep,
                                                                @Nullable final Iterable  aElements,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (cSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (@Nonnull final String sSep, @Nullable final String... aElements)
  {
    return imploder ().source (aElements).separator (sSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (final char cSep, @Nullable final String... aElements)
  {
    return imploder ().source (aElements).separator (cSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Array component type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (@Nonnull final String sSep,
                                                                @Nullable final ELEMENTTYPE [] aElements,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (sSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Array component type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (final char cSep,
                                                                @Nullable final ELEMENTTYPE [] aElements,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper).separator (cSep).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (@Nonnull final String sSep,
                                            @Nullable final String [] aElements,
                                            @Nonnegative final int nOfs,
                                            @Nonnegative final int nLen)
  {
    return imploder ().source (aElements).separator (sSep).offset (nOfs).length (nLen).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getImplodedNonEmpty (final char cSep,
                                            @Nullable final String [] aElements,
                                            @Nonnegative final int nOfs,
                                            @Nonnegative final int nLen)
  {
    return imploder ().source (aElements).separator (cSep).offset (nOfs).length (nLen).filterNonEmpty ().build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator string. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Array component type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (@Nonnull final String sSep,
                                                                @Nullable final ELEMENTTYPE [] aElements,
                                                                @Nonnegative final int nOfs,
                                                                @Nonnegative final int nLen,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper)
                      .separator (sSep)
                      .offset (nOfs)
                      .length (nLen)
                      .filterNonEmpty ()
                      .build ();
  }

  /**
   * Get a concatenated String from all elements of the passed array, separated
   * by the specified separator char. This the very generic version of
   * {@link #getConcatenatedOnDemand(String, String, String)} for an arbitrary
   * number of elements.
   *
   * @param cSep
   *        The separator to use.
   * @param aElements
   *        The container to convert. May be null or empty.
   * @param nOfs
   *        The offset to start from.
   * @param nLen
   *        The number of elements to implode.
   * @param aMapper
   *        The mapping function to convert from ELEMENTTYPE to String. May not
   *        be null.
   * @return The concatenated string.
   * @param 
   *        Array component type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMappedNonEmpty (final char cSep,
                                                                @Nullable final ELEMENTTYPE [] aElements,
                                                                @Nonnegative final int nOfs,
                                                                @Nonnegative final int nLen,
                                                                @Nonnull final Function  aMapper)
  {
    return imploder ().source (aElements, aMapper)
                      .separator (cSep)
                      .offset (nOfs)
                      .length (nLen)
                      .filterNonEmpty ()
                      .build ();
  }

  /**
   * A simple builder that allows to implode maps of arguments with a lot of
   * customization. It used by all the "getImploded*" overloads and fulfills the
   * requests of other use cases as well.
   *
   * @author Philip Helger
   * @since 10.0.0
   */
  public static class ImploderBuilderMap implements IBuilder 
  {
    private ICommonsMap  m_aSource;
    private String m_sSeparatorOuter;
    private String m_sSeparatorInner;
    private Predicate  m_aFilterKey;
    private Predicate  m_aFilterValue;

    public ImploderBuilderMap ()
    {}

    @Nullable
    private static String _valueOf (@Nullable final Object o)
    {
      return o == null ? null : o.toString ();
    }

    @Nonnull
    public ImploderBuilderMap source (@Nullable final Map  a)
    {
      return source (a, ImploderBuilderMap::_valueOf, ImploderBuilderMap::_valueOf);
    }

    @Nonnull
    public  ImploderBuilderMap source (@Nullable final Map  a,
                                             @Nonnull final Function  aKeyMapper,
                                             @Nonnull final Function  aValueMapper)
    {
      ValueEnforcer.notNull (aKeyMapper, "KeyMapper");
      ValueEnforcer.notNull (aValueMapper, "ValueMapper");
      m_aSource = a == null ? null : new CommonsLinkedHashMap <> (a, aKeyMapper, aValueMapper);
      return this;
    }

    @Nonnull
    public ImploderBuilderMap separatorOuter (final char c)
    {
      return separatorOuter (Character.toString (c));
    }

    @Nonnull
    public ImploderBuilderMap separatorOuter (@Nullable final String s)
    {
      m_sSeparatorOuter = s;
      return this;
    }

    @Nonnull
    public ImploderBuilderMap separatorInner (final char c)
    {
      return separatorInner (Character.toString (c));
    }

    @Nonnull
    public ImploderBuilderMap separatorInner (@Nullable final String s)
    {
      m_sSeparatorInner = s;
      return this;
    }

    @Nonnull
    public ImploderBuilderMap filterKeyNonEmpty ()
    {
      return filterKey (StringHelper::hasText);
    }

    @Nonnull
    public ImploderBuilderMap filterKey (@Nullable final Predicate  a)
    {
      m_aFilterKey = a;
      return this;
    }

    @Nonnull
    public ImploderBuilderMap filterValueNonEmpty ()
    {
      return filterValue (StringHelper::hasText);
    }

    @Nonnull
    public ImploderBuilderMap filterValue (@Nullable final Predicate  a)
    {
      m_aFilterValue = a;
      return this;
    }

    @Nonnull
    public String build ()
    {
      final ICommonsMap  aSource = m_aSource;
      if (aSource == null || aSource.isEmpty ())
        return "";

      final StringBuilder aSB = new StringBuilder ();

      // Avoid constant this access for speed
      final String sSepOuter = m_sSeparatorOuter;
      final String sSepInner = m_sSeparatorInner;
      final Predicate  aFilterKey = m_aFilterKey == null ? Predicates.all () : m_aFilterKey;
      final Predicate  aFilterValue = m_aFilterValue == null ? Predicates.all () : m_aFilterValue;

      // Iterate all
      int nElementsAdded = 0;
      for (final Map.Entry  aEntry : aSource.entrySet ())
      {
        final String sKey = aEntry.getKey ();
        final String sValue = aEntry.getValue ();
        if (aFilterKey.test (sKey) && aFilterValue.test (sValue))
        {
          if (nElementsAdded > 0 && sSepOuter != null)
            aSB.append (sSepOuter);
          aSB.append (sKey);
          if (sSepInner != null)
            aSB.append (sSepInner);
          aSB.append (sValue);
          nElementsAdded++;
        }
      }
      return aSB.toString ();
    }
  }

  /**
   * @return A new {@link ImploderBuilderMap}.
   * @since 10.0.0
   */
  @Nonnull
  public static ImploderBuilderMap imploderMap ()
  {
    return new ImploderBuilderMap ();
  }

  /**
   * Get a concatenated String from all elements of the passed map, separated by
   * the specified separator strings.
   *
   * @param sSepOuter
   *        The separator to use for separating the map entries. May not be
   *        null.
   * @param sSepInner
   *        The separator to use for separating the key from the value. May not
   *        be null.
   * @param aElements
   *        The map to convert. May be null or empty.
   * @return The concatenated string.
   * @param 
   *        Map key type
   * @param 
   *        Map value type
   */
  @Nonnull
  public static  String getImploded (@Nonnull final String sSepOuter,
                                                         @Nonnull final String sSepInner,
                                                         @Nullable final Map  aElements)
  {
    return imploderMap ().source (aElements).separatorOuter (sSepOuter).separatorInner (sSepInner).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed map, separated by
   * the specified separator chars.
   *
   * @param cSepOuter
   *        The separator to use for separating the map entries.
   * @param cSepInner
   *        The separator to use for separating the key from the value.
   * @param aElements
   *        The map to convert. May be null or empty.
   * @return The concatenated string.
   * @param 
   *        Map key type
   * @param 
   *        Map value type
   */
  @Nonnull
  public static  String getImploded (final char cSepOuter,
                                                         final char cSepInner,
                                                         @Nullable final Map  aElements)
  {
    return imploderMap ().source (aElements).separatorOuter (cSepOuter).separatorInner (cSepInner).build ();
  }

  /**
   * Get a concatenated String from all elements of the passed map, separated by
   * the specified separator strings.
   *
   * @param sSepOuter
   *        The separator to use for separating the map entries. May not be
   *        null.
   * @param sSepInner
   *        The separator to use for separating the key from the value. May not
   *        be null.
   * @param aElements
   *        The map to convert. May be null or empty.
   * @param aKeyMapper
   *        The mapping function to convert from KEYTYPE to String. May not be
   *        null.
   * @param aValueMapper
   *        The mapping function to convert from VALUETYPE to String. May not be
   *        null.
   * @return The concatenated string.
   * @param 
   *        Map key type
   * @param 
   *        Map value type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (@Nonnull final String sSepOuter,
                                                               @Nonnull final String sSepInner,
                                                               @Nullable final Map  aElements,
                                                               @Nonnull final Function  aKeyMapper,
                                                               @Nonnull final Function  aValueMapper)
  {
    return imploderMap ().source (aElements, aKeyMapper, aValueMapper)
                         .separatorOuter (sSepOuter)
                         .separatorInner (sSepInner)
                         .build ();
  }

  /**
   * Get a concatenated String from all elements of the passed map, separated by
   * the specified separator chars.
   *
   * @param cSepOuter
   *        The separator to use for separating the map entries.
   * @param cSepInner
   *        The separator to use for separating the key from the value.
   * @param aElements
   *        The map to convert. May be null or empty.
   * @param aKeyMapper
   *        The mapping function to convert from KEYTYPE to String. May not be
   *        null.
   * @param aValueMapper
   *        The mapping function to convert from VALUETYPE to String. May not be
   *        null.
   * @return The concatenated string.
   * @param 
   *        Map key type
   * @param 
   *        Map value type
   * @since 8.5.6
   */
  @Nonnull
  public static  String getImplodedMapped (final char cSepOuter,
                                                               final char cSepInner,
                                                               @Nullable final Map  aElements,
                                                               @Nonnull final Function  aKeyMapper,
                                                               @Nonnull final Function  aValueMapper)
  {
    return imploderMap ().source (aElements, aKeyMapper, aValueMapper)
                         .separatorOuter (cSepOuter)
                         .separatorInner (cSepInner)
                         .build ();
  }

  /**
   * Take a concatenated String and return the passed String array of all
   * elements in the passed string, using specified separator char.
   *
   * @param cSep
   *        The separator to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @return The passed collection and never null.
   */
  @Nonnull
  public static String [] getExplodedArray (final char cSep,
                                            @Nullable final String sElements,
                                            @CheckForSigned final int nMaxItems)
  {
    if (nMaxItems == 1)
      return new String [] { sElements };
    if (hasNoText (sElements))
      return ArrayHelper.EMPTY_STRING_ARRAY;

    final int nMaxResultElements = 1 + getCharCount (sElements, cSep);
    if (nMaxResultElements == 1)
    {
      // Separator not found
      return new String [] { sElements };
    }
    final String [] ret = new String [nMaxItems < 1 ? nMaxResultElements : Math.min (nMaxResultElements, nMaxItems)];

    // Do not use RegExCache.stringReplacePattern because of package
    // dependencies
    // Do not use String.split because it trims empty tokens from the end
    int nStartIndex = 0;
    int nItemsAdded = 0;
    while (true)
    {
      final int nMatchIndex = sElements.indexOf (cSep, nStartIndex);
      if (nMatchIndex < 0)
        break;

      ret[nItemsAdded++] = sElements.substring (nStartIndex, nMatchIndex);
      // 1 == length of separator char
      nStartIndex = nMatchIndex + 1;
      if (nMaxItems > 0 && nItemsAdded == nMaxItems - 1)
      {
        // We have exactly one item the left: the rest of the string
        break;
      }
    }
    ret[nItemsAdded++] = sElements.substring (nStartIndex);
    if (nItemsAdded != ret.length)
      throw new IllegalStateException ("Added " + nItemsAdded + " but expected " + ret.length);
    return ret;
  }

  /**
   * Take a concatenated String and return the passed String array of all
   * elements in the passed string, using specified separator char.
   *
   * @param cSep
   *        The separator to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @return The passed collection and never null.
   */
  @Nonnull
  public static String [] getExplodedArray (final char cSep, @Nullable final String sElements)
  {
    return getExplodedArray (cSep, sElements, -1);
  }

  /**
   * Take a concatenated String and return the passed Collection of all elements
   * in the passed string, using specified separator string.
   *
   * @param 
   *        The collection type to be passed and returned
   * @param cSep
   *        The separator to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @param aCollection
   *        The non-null target collection that should be filled
   *        with the exploded elements
   * @return The passed collection and never null.
   */
  @Nonnull
  @ReturnsMutableObject ("The passed parameter")
  @CodingStyleguideUnaware
  public static > COLLTYPE getExploded (final char cSep,
                                                                             @Nullable final String sElements,
                                                                             final int nMaxItems,
                                                                             @Nonnull final COLLTYPE aCollection)
  {
    explode (cSep, sElements, nMaxItems, aCollection::add);
    return aCollection;
  }

  /**
   * Split the provided string by the provided separator and invoke the consumer
   * for each matched element. The number of returned items is unlimited.
   *
   * @param cSep
   *        The separator to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param aConsumer
   *        The non-null consumer that is invoked for each exploded
   *        element
   */
  public static void explode (final char cSep,
                              @Nullable final String sElements,
                              @Nonnull final Consumer  aConsumer)
  {
    explode (cSep, sElements, -1, aConsumer);
  }

  /**
   * Split the provided string by the provided separator and invoke the consumer
   * for each matched element. The maximum number of elements can be specified.
   *
   * @param cSep
   *        The separator to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @param aConsumer
   *        The non-null consumer that is invoked for each exploded
   *        element
   */
  public static void explode (final char cSep,
                              @Nullable final String sElements,
                              final int nMaxItems,
                              @Nonnull final Consumer  aConsumer)
  {
    ValueEnforcer.notNull (aConsumer, "Consumer");

    if (nMaxItems == 1)
      aConsumer.accept (sElements);
    else
      if (hasText (sElements))
      {
        // Do not use RegExPool.stringReplacePattern because of package
        // dependencies
        // Do not use String.split because it trims empty tokens from the end
        int nStartIndex = 0;
        int nMatchIndex;
        int nItemsAdded = 0;
        while ((nMatchIndex = sElements.indexOf (cSep, nStartIndex)) >= 0)
        {
          aConsumer.accept (sElements.substring (nStartIndex, nMatchIndex));
          // 1 == length of separator char
          nStartIndex = nMatchIndex + 1;
          ++nItemsAdded;
          if (nMaxItems > 0 && nItemsAdded == nMaxItems - 1)
          {
            // We have exactly one item the left: the rest of the string
            break;
          }
        }
        aConsumer.accept (sElements.substring (nStartIndex));
      }
  }

  /**
   * Take a concatenated String and return a {@link ICommonsList} of all
   * elements in the passed string, using specified separator string.
   *
   * @param cSep
   *        The separator character to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @return The {@link ICommonsList} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static ICommonsList  getExploded (final char cSep, @Nullable final String sElements)
  {
    return getExploded (cSep, sElements, -1);
  }

  /**
   * Take a concatenated String and return a {@link ICommonsList} of all
   * elements in the passed string, using specified separator string.
   *
   * @param cSep
   *        The separator character to use.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @return The {@link ICommonsList} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static ICommonsList  getExploded (final char cSep,
                                                   @Nullable final String sElements,
                                                   final int nMaxItems)
  {
    return getExploded (cSep,
                        sElements,
                        nMaxItems,
                        nMaxItems >= 1 ? new CommonsArrayList <> (nMaxItems) : new CommonsArrayList <> ());
  }

  /**
   * Take a concatenated String and return the passed Collection of all elements
   * in the passed string, using specified separator string.
   *
   * @param 
   *        The collection type to be used and returned
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @param aCollection
   *        The non-null target collection that should be filled
   *        with the exploded elements
   * @return The passed collection and never null.
   */
  @Nonnull
  @CodingStyleguideUnaware
  public static > COLLTYPE getExploded (@Nonnull final String sSep,
                                                                             @Nullable final String sElements,
                                                                             final int nMaxItems,
                                                                             @Nonnull final COLLTYPE aCollection)
  {
    explode (sSep, sElements, nMaxItems, aCollection::add);
    return aCollection;
  }

  /**
   * Split the provided string by the provided separator and invoke the consumer
   * for each matched element.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param aConsumer
   *        The non-null consumer that is invoked for each exploded
   *        element
   */
  public static void explode (@Nonnull final String sSep,
                              @Nullable final String sElements,
                              @Nonnull final Consumer  aConsumer)
  {
    explode (sSep, sElements, -1, aConsumer);
  }

  /**
   * Split the provided string by the provided separator and invoke the consumer
   * for each matched element. The maximum number of elements can be specified.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @param aConsumer
   *        The non-null consumer that is invoked for each exploded
   *        element
   */
  public static void explode (@Nonnull final String sSep,
                              @Nullable final String sElements,
                              final int nMaxItems,
                              @Nonnull final Consumer  aConsumer)
  {
    ValueEnforcer.notNull (sSep, "Separator");
    ValueEnforcer.notNull (aConsumer, "Collection");

    if (sSep.length () == 1)
    {
      // If the separator consists only of a single character, use the
      // char-optimized version for better performance
      // Note: do it before the "hasText (sElements)" check, because the same
      // check is performed in the char version as well
      explode (sSep.charAt (0), sElements, nMaxItems, aConsumer);
    }
    else
    {
      if (nMaxItems == 1)
        aConsumer.accept (sElements);
      else
      {
        if (hasText (sElements))
        {
          // Do not use RegExPool.stringReplacePattern because of package
          // dependencies
          // Do not use String.split because it trims empty tokens from the end
          int nStartIndex = 0;
          int nItemsAdded = 0;
          while (true)
          {
            final int nMatchIndex = sElements.indexOf (sSep, nStartIndex);
            if (nMatchIndex < 0)
              break;
            aConsumer.accept (sElements.substring (nStartIndex, nMatchIndex));
            nStartIndex = nMatchIndex + sSep.length ();
            ++nItemsAdded;
            if (nMaxItems > 0 && nItemsAdded == nMaxItems - 1)
            {
              // We have exactly one item the left: the rest of the string
              break;
            }
          }
          aConsumer.accept (sElements.substring (nStartIndex));
        }
      }
    }
  }

  /**
   * Take a concatenated String and return a {@link ICommonsList} of all
   * elements in the passed string, using specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @return The {@link ICommonsList} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static ICommonsList  getExploded (@Nonnull final String sSep, @Nullable final String sElements)
  {
    return getExploded (sSep, sElements, -1);
  }

  /**
   * Take a concatenated String and return a {@link ICommonsList} of all
   * elements in the passed string, using specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @param nMaxItems
   *        The maximum number of items to explode. If the passed value is ≤
   *        0 all items are used. If max items is 1, than the result string is
   *        returned as is. If max items is larger than the number of elements
   *        found, it has no effect.
   * @return The {@link ICommonsList} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static ICommonsList  getExploded (@Nonnull final String sSep,
                                                   @Nullable final String sElements,
                                                   final int nMaxItems)
  {
    return getExploded (sSep,
                        sElements,
                        nMaxItems,
                        nMaxItems >= 1 ? new CommonsArrayList <> (nMaxItems) : new CommonsArrayList <> ());
  }

  /**
   * Take a concatenated String and return a {@link Set} of all elements in the
   * passed string, using specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @return The {@link Set} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static CommonsHashSet  getExplodedToSet (@Nonnull final String sSep, @Nullable final String sElements)
  {
    return getExploded (sSep, sElements, -1, new CommonsHashSet <> ());
  }

  /**
   * Take a concatenated String and return an ordered
   * {@link CommonsLinkedHashSet} of all elements in the passed string, using
   * specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @return The ordered {@link Set} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static CommonsLinkedHashSet  getExplodedToOrderedSet (@Nonnull final String sSep,
                                                                       @Nullable final String sElements)
  {
    return getExploded (sSep, sElements, -1, new CommonsLinkedHashSet <> ());
  }

  /**
   * Take a concatenated String and return a sorted {@link CommonsTreeSet} of
   * all elements in the passed string, using specified separator string.
   *
   * @param sSep
   *        The separator to use. May not be null.
   * @param sElements
   *        The concatenated String to convert. May be null or
   *        empty.
   * @return The sorted {@link Set} represented by the passed string. Never
   *         null. If the passed input string is null
   *         or "" an empty list is returned.
   */
  @Nonnull
  @ReturnsMutableCopy
  public static CommonsTreeSet  getExplodedToSortedSet (@Nonnull final String sSep,
                                                                @Nullable final String sElements)
  {
    return getExploded (sSep, sElements, -1, new CommonsTreeSet <> ());
  }

  /**
   * Get the passed string element repeated for a certain number of times. Each
   * string element is simply appended at the end of the string.
   *
   * @param cElement
   *        The character to get repeated.
   * @param nRepeats
   *        The number of repetitions to retrieve. May not be < 0.
   * @return A non-null string containing the string element for
   *         the given number of times.
   */
  @Nonnull
  public static String getRepeated (final char cElement, @Nonnegative final int nRepeats)
  {
    ValueEnforcer.isGE0 (nRepeats, "Repeats");

    if (nRepeats == 0)
      return "";
    if (nRepeats == 1)
      return Character.toString (cElement);

    final char [] aElement = new char [nRepeats];
    Arrays.fill (aElement, cElement);
    return new String (aElement);
  }

  /**
   * Get the passed string element repeated for a certain number of times. Each
   * string element is simply appended at the end of the string.
   *
   * @param sElement
   *        The string to get repeated. May not be null.
   * @param nRepeats
   *        The number of repetitions to retrieve. May not be < 0.
   * @return A non-null string containing the string element for
   *         the given number of times.
   */
  @Nonnull
  public static String getRepeated (@Nonnull final String sElement, @Nonnegative final int nRepeats)
  {
    ValueEnforcer.notNull (sElement, "Element");
    ValueEnforcer.isGE0 (nRepeats, "Repeats");

    final int nElementLength = sElement.length ();

    // Check if result length would exceed int range
    if ((long) nElementLength * nRepeats > Integer.MAX_VALUE)
      throw new IllegalArgumentException ("Resulting string exceeds the maximum integer length");

    if (nElementLength == 0 || nRepeats == 0)
      return "";
    if (nRepeats == 1)
      return sElement;

    // use character version
    if (nElementLength == 1)
      return getRepeated (sElement.charAt (0), nRepeats);

    // combine via StringBuilder
    final StringBuilder ret = new StringBuilder (nElementLength * nRepeats);
    for (int i = 0; i < nRepeats; ++i)
      ret.append (sElement);
    return ret.toString ();
  }

  /**
   * Concatenate the strings sFront and sEnd. If either front or back is
   * null or empty only the other element is returned. If both
   * strings are null or empty and empty String is returned.
   *
   * @param sFront
   *        Front string. May be null.
   * @param sEnd
   *        May be null.
   * @return The concatenated string. Never null.
   */
  @Nonnull
  public static String getConcatenatedOnDemand (@Nullable final String sFront, @Nullable final String sEnd)
  {
    if (sFront == null)
      return sEnd == null ? "" : sEnd;
    if (sEnd == null)
      return sFront;
    return sFront + sEnd;
  }

  /**
   * Concatenate the strings sFront and sEnd by the "sSep" string. If either
   * front or back is null or empty, the separator is not applied.
   *
   * @param sFront
   *        Front string. May be null.
   * @param sSep
   *        Separator string. May be null.
   * @param sEnd
   *        May be null.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getConcatenatedOnDemand (@Nullable final String sFront,
                                                @Nullable final String sSep,
                                                @Nullable final String sEnd)
  {
    final StringBuilder aSB = new StringBuilder ();
    if (hasText (sFront))
    {
      aSB.append (sFront);
      if (hasText (sSep) && hasText (sEnd))
        aSB.append (sSep);
    }
    if (hasText (sEnd))
      aSB.append (sEnd);
    return aSB.toString ();
  }

  /**
   * Concatenate the strings sFront and sEnd by the "cSep" separator. If either
   * front or back is null or empty, the separator is not applied.
   *
   * @param sFront
   *        Front string. May be null.
   * @param cSep
   *        Separator character.
   * @param sEnd
   *        May be null.
   * @return The concatenated string.
   */
  @Nonnull
  public static String getConcatenatedOnDemand (@Nullable final String sFront,
                                                final char cSep,
                                                @Nullable final String sEnd)
  {
    final StringBuilder aSB = new StringBuilder ();
    if (hasText (sFront))
    {
      aSB.append (sFront);
      if (hasText (sEnd))
        aSB.append (cSep);
    }
    if (hasText (sEnd))
      aSB.append (sEnd);
    return aSB.toString ();
  }

  /**
   * Get the provided string quoted or unquoted if it is null.
   *
   * @param sSource
   *        Source string. May be null.
   * @return The String "null" if the source is null,
   *         "'" + sSource + "'" otherwise.
   * @since 9.2.0
   */
  @Nonnull
  public static String getQuoted (@Nullable final String sSource)
  {
    return sSource == null ? "null" : "'" + sSource + "'";
  }

  /**
   * Append the provided string quoted or unquoted if it is null.
   *
   * @param aTarget
   *        The target to write to. May not be null.
   * @param sSource
   *        Source string. May be null.
   * @see #getQuoted(String)
   * @since 9.2.0
   */
  public static void appendQuoted (@Nonnull final StringBuilder aTarget, @Nullable final String sSource)
  {
    if (sSource == null)
      aTarget.append ("null");
    else
      aTarget.append ('\'').append (sSource).append ('\'');
  }

  /**
   * Append the provided string quoted or unquoted if it is null.
   *
   * @param aTarget
   *        The target to write to. May not be null.
   * @param sSource
   *        Source string. May be null.
   * @throws IOException
   *         in case of IO error
   * @see #getQuoted(String)
   * @since 9.2.0
   */
  public static void appendQuoted (@Nonnull final Appendable aTarget, @Nullable final String sSource) throws IOException
  {
    if (sSource == null)
      aTarget.append ("null");
    else
      aTarget.append ('\'').append (sSource).append ('\'');
  }

  public static boolean startsWith (@Nullable final CharSequence aCS, final char c)
  {
    return hasText (aCS) && aCS.charAt (0) == c;
  }

  public static boolean startsWithAny (@Nullable final CharSequence aCS, @Nullable final char [] aChars)
  {
    if (hasText (aCS) && aChars != null)
      if (ArrayHelper.contains (aChars, aCS.charAt (0)))
        return true;
    return false;
  }

  public static boolean startsWithIgnoreCase (@Nullable final CharSequence aCS, final char c)
  {
    return hasText (aCS) && Character.toLowerCase (aCS.charAt (0)) == Character.toLowerCase (c);
  }

  public static boolean startsWith (@Nullable final CharSequence aCS, @Nullable final CharSequence aSearch)
  {
    if (aCS == null || aSearch == null)
      return false;
    final int nSearchLength = aSearch.length ();
    if (nSearchLength == 0)
      return true;
    final int nCSLength = aCS.length ();
    if (nCSLength < nSearchLength)
      return false;
    return aCS.subSequence (0, nSearchLength).equals (aSearch);
  }

  public static boolean startsWith (@Nullable final String sStr, @Nullable final String sSearch)
  {
    if (sStr == null || sSearch == null)
      return false;
    final int nSearchLength = sSearch.length ();
    if (nSearchLength == 0)
      return true;
    final int nStrLength = sStr.length ();
    if (nStrLength < nSearchLength)
      return false;

    if (nSearchLength == 1)
      return sStr.charAt (0) == sSearch.charAt (0);

    return sStr.subSequence (0, nSearchLength).equals (sSearch);
  }

  public static boolean startsWithIgnoreCase (@Nullable final String sStr, @Nullable final String sSearch)
  {
    if (sStr == null || sSearch == null)
      return false;
    final int nSearchLength = sSearch.length ();
    if (nSearchLength == 0)
      return true;
    final int nStrLength = sStr.length ();
    if (nStrLength < nSearchLength)
      return false;
    return sStr.substring (0, nSearchLength).equalsIgnoreCase (sSearch);
  }

  public static boolean endsWith (@Nullable final CharSequence aCS, final char c)
  {
    return hasText (aCS) && getLastChar (aCS) == c;
  }

  public static boolean endsWithAny (@Nullable final CharSequence aCS, @Nullable final char [] aChars)
  {
    if (hasText (aCS) && aChars != null)
      if (ArrayHelper.contains (aChars, getLastChar (aCS)))
        return true;
    return false;
  }

  public static boolean endsWith (@Nullable final CharSequence aCS, @Nullable final CharSequence aSearch)
  {
    if (aCS == null || aSearch == null)
      return false;
    final int nSearchLength = aSearch.length ();
    if (nSearchLength == 0)
      return true;
    final int nCSLength = aCS.length ();
    if (nCSLength < nSearchLength)
      return false;

    if (nSearchLength == 1)
      return aCS.charAt (nCSLength - 1) == aSearch.charAt (0);

    return aCS.subSequence (nCSLength - nSearchLength, nCSLength).equals (aSearch);
  }

  public static boolean endsWith (@Nullable final String sStr, @Nullable final String sSearch)
  {
    if (sStr == null || sSearch == null)
      return false;
    final int nSearchLength = sSearch.length ();
    if (nSearchLength == 0)
      return true;
    final int nStrLength = sStr.length ();
    if (nStrLength < nSearchLength)
      return false;

    if (nSearchLength == 1)
      return sStr.charAt (nStrLength - 1) == sSearch.charAt (0);

    return sStr.startsWith (sSearch, nStrLength - nSearchLength);
  }

  public static boolean endsWithIgnoreCase (@Nullable final CharSequence aCS, final char c)
  {
    return hasText (aCS) && Character.toLowerCase (getLastChar (aCS)) == Character.toLowerCase (c);
  }

  public static boolean endsWithIgnoreCase (@Nullable final String sStr, @Nullable final String sSearch)
  {
    if (sStr == null || sSearch == null)
      return false;
    final int nSearchLength = sSearch.length ();
    if (nSearchLength == 0)
      return true;
    final int nStrLength = sStr.length ();
    if (nStrLength < nSearchLength)
      return false;
    return sStr.substring (nStrLength - nSearchLength, nStrLength).equalsIgnoreCase (sSearch);
  }

  /**
   * Get the first index of sSearch within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(String)
   */
  public static int getIndexOf (@Nullable final String sText, @Nullable final String sSearch)
  {
    return sText != null && sSearch != null && sText.length () >= sSearch.length () ? sText.indexOf (sSearch)
                                                                                    : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of sSearch within sText starting at index nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param sSearch
   *        The text to search for. May be null.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(String,int)
   */
  public static int getIndexOf (@Nullable final String sText,
                                @Nonnegative final int nFromIndex,
                                @Nullable final String sSearch)
  {
    return sText != null &&
           sSearch != null &&
           (sText.length () - nFromIndex) >= sSearch.length () ? sText.indexOf (sSearch, nFromIndex) : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of sSearch within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(String)
   */
  public static int getLastIndexOf (@Nullable final String sText, @Nullable final String sSearch)
  {
    return sText != null && sSearch != null && sText.length () >= sSearch.length () ? sText.lastIndexOf (sSearch)
                                                                                    : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of sSearch within sText starting at index nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param sSearch
   *        The text to search for. May be null.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(String,int)
   */
  public static int getLastIndexOf (@Nullable final String sText,
                                    @Nonnegative final int nFromIndex,
                                    @Nullable final String sSearch)
  {
    return sText != null &&
           sSearch != null &&
           (sText.length () - nFromIndex) >= sSearch.length () ? sText.lastIndexOf (sSearch, nFromIndex)
                                                               : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of cSearch within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The character to search for. May be null.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if cSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(int)
   */
  public static int getIndexOf (@Nullable final String sText, final char cSearch)
  {
    return sText != null && sText.length () >= 1 ? sText.indexOf (cSearch) : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of cSearch within sText starting at index nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param cSearch
   *        The character to search for. May be null.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if cSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(int,int)
   */
  public static int getIndexOf (@Nullable final String sText, @Nonnegative final int nFromIndex, final char cSearch)
  {
    return sText != null && (sText.length () - nFromIndex) >= 1 ? sText.indexOf (cSearch, nFromIndex)
                                                                : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of cSearch within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The character to search for. May be null.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if cSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(int)
   */
  public static int getLastIndexOf (@Nullable final String sText, final char cSearch)
  {
    return sText != null && sText.length () >= 1 ? sText.lastIndexOf (cSearch) : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of cSearch within sText starting at index nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param cSearch
   *        The character to search for. May be null.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if cSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(int,int)
   */
  public static int getLastIndexOf (@Nullable final String sText, @Nonnegative final int nFromIndex, final char cSearch)
  {
    return sText != null && (sText.length () - nFromIndex) >= 1 ? sText.lastIndexOf (cSearch, nFromIndex)
                                                                : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of sSearch within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(String)
   */
  public static int getIndexOfIgnoreCase (@Nullable final String sText,
                                          @Nullable final String sSearch,
                                          @Nonnull final Locale aSortLocale)
  {
    return sText != null &&
           sSearch != null &&
           sText.length () >= sSearch.length () ? sText.toLowerCase (aSortLocale).indexOf (sSearch.toLowerCase (aSortLocale)) : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of sSearch within sText ignoring case starting at index
   * nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param sSearch
   *        The text to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(String)
   */
  public static int getIndexOfIgnoreCase (@Nullable final String sText,
                                          @Nonnegative final int nFromIndex,
                                          @Nullable final String sSearch,
                                          @Nonnull final Locale aSortLocale)
  {
    return sText != null &&
           sSearch != null &&
           (sText.length () - nFromIndex) >= sSearch.length ()
                                                               ? sText.toLowerCase (aSortLocale)
                                                                      .indexOf (sSearch.toLowerCase (aSortLocale),
                                                                                nFromIndex)
                                                               : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of sSearch within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(String)
   */
  public static int getLastIndexOfIgnoreCase (@Nullable final String sText,
                                              @Nullable final String sSearch,
                                              @Nonnull final Locale aSortLocale)
  {
    return sText != null &&
           sSearch != null &&
           sText.length () >= sSearch.length () ? sText.toLowerCase (aSortLocale).lastIndexOf (sSearch.toLowerCase (aSortLocale)) : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of sSearch within sText ignoring case starting at index
   * nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param sSearch
   *        The text to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(String)
   */
  public static int getLastIndexOfIgnoreCase (@Nullable final String sText,
                                              @Nonnegative final int nFromIndex,
                                              @Nullable final String sSearch,
                                              @Nonnull final Locale aSortLocale)
  {
    return sText != null &&
           sSearch != null &&
           (sText.length () - nFromIndex) >= sSearch.length ()
                                                               ? sText.toLowerCase (aSortLocale)
                                                                      .lastIndexOf (sSearch.toLowerCase (aSortLocale),
                                                                                    nFromIndex)
                                                               : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of cSearch within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The char to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(int)
   */
  public static int getIndexOfIgnoreCase (@Nullable final String sText,
                                          final char cSearch,
                                          @Nonnull final Locale aSortLocale)
  {
    return sText != null && sText.length () >= 1 ? sText.toLowerCase (aSortLocale)
                                                        .indexOf (Character.toLowerCase (cSearch))
                                                 : STRING_NOT_FOUND;
  }

  /**
   * Get the first index of cSearch within sText ignoring case starting at index
   * nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param cSearch
   *        The char to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The first index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#indexOf(int)
   */
  public static int getIndexOfIgnoreCase (@Nullable final String sText,
                                          @Nonnegative final int nFromIndex,
                                          final char cSearch,
                                          @Nonnull final Locale aSortLocale)
  {
    return sText != null && (sText.length () - nFromIndex) >= 1 ? sText.toLowerCase (aSortLocale)
                                                                       .indexOf (Character.toLowerCase (cSearch),
                                                                                 nFromIndex)
                                                                : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of cSearch within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The char to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(int)
   */
  public static int getLastIndexOfIgnoreCase (@Nullable final String sText,
                                              final char cSearch,
                                              @Nonnull final Locale aSortLocale)
  {
    return sText != null && sText.length () >= 1
                                                 ? sText.toLowerCase (aSortLocale)
                                                        .lastIndexOf (Character.toLowerCase (cSearch))
                                                 : STRING_NOT_FOUND;
  }

  /**
   * Get the last index of cSearch within sText ignoring case starting at index
   * nFromIndex.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param nFromIndex
   *        The index to start searching in the source string
   * @param cSearch
   *        The char to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return The last index of sSearch within sText or
   *         {@value #STRING_NOT_FOUND} if sSearch was not found or if any
   *         parameter was null.
   * @see String#lastIndexOf(int)
   */
  public static int getLastIndexOfIgnoreCase (@Nullable final String sText,
                                              @Nonnegative final int nFromIndex,
                                              final char cSearch,
                                              @Nonnull final Locale aSortLocale)
  {
    return sText != null && (sText.length () - nFromIndex) >= 1 ? sText.toLowerCase (aSortLocale)
                                                                       .lastIndexOf (Character.toLowerCase (cSearch),
                                                                                     nFromIndex)
                                                                : STRING_NOT_FOUND;
  }

  /**
   * Check if sSearch is contained within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @return true if sSearch is contained in sText,
   *         false otherwise.
   * @see String#contains(CharSequence)
   */
  public static boolean contains (@Nullable final String sText, @Nullable final String sSearch)
  {
    return getIndexOf (sText, sSearch) != STRING_NOT_FOUND;
  }

  /**
   * Check if cSearch is contained within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The character to search for. May be null.
   * @return true if cSearch is contained in sText,
   *         false otherwise.
   * @see String#contains(CharSequence)
   */
  public static boolean contains (@Nullable final String sText, final char cSearch)
  {
    return getIndexOf (sText, cSearch) != STRING_NOT_FOUND;
  }

  /**
   * Check if sSearch is contained within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return true if sSearch is contained in sText,
   *         false otherwise.
   * @see String#contains(CharSequence)
   */
  public static boolean containsIgnoreCase (@Nullable final String sText,
                                            @Nullable final String sSearch,
                                            @Nonnull final Locale aSortLocale)
  {
    return getIndexOfIgnoreCase (sText, sSearch, aSortLocale) != STRING_NOT_FOUND;
  }

  /**
   * Check if cSearch is contained within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The char to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return true if sSearch is contained in sText,
   *         false otherwise.
   * @see String#indexOf(int)
   */
  public static boolean containsIgnoreCase (@Nullable final String sText,
                                            final char cSearch,
                                            @Nonnull final Locale aSortLocale)
  {
    return getIndexOfIgnoreCase (sText, cSearch, aSortLocale) != STRING_NOT_FOUND;
  }

  /**
   * Check if any of the passed searched characters is contained in the input
   * char array.
   *
   * @param aInput
   *        The input char array. May be null.
   * @param aSearchChars
   *        The char array to search. May not be null.
   * @return true if at least any of the search char is contained
   *         in the input char array, false otherwise.
   */
  public static boolean containsAny (@Nullable final char [] aInput, @Nonnull final char [] aSearchChars)
  {
    ValueEnforcer.notNull (aSearchChars, "SearchChars");

    if (aInput != null)
      for (final char cIn : aInput)
        if (ArrayHelper.contains (aSearchChars, cIn))
          return true;
    return false;
  }

  /**
   * Check if any of the passed searched characters in contained in the input
   * string.
   *
   * @param sInput
   *        The input string. May be null.
   * @param aSearchChars
   *        The char array to search. May not be null.
   * @return true if at least any of the search char is contained
   *         in the input char array, false otherwise.
   */
  public static boolean containsAny (@Nullable final String sInput, @Nonnull final char [] aSearchChars)
  {
    return sInput != null && containsAny (sInput.toCharArray (), aSearchChars);
  }

  /**
   * Count the number of occurrences of sSearch within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @return A non-negative number of occurrences.
   */
  @Nonnegative
  public static int getOccurrenceCount (@Nullable final String sText, @Nullable final String sSearch)
  {
    int ret = 0;
    final int nTextLength = getLength (sText);
    final int nSearchLength = getLength (sSearch);
    if (nSearchLength > 0 && nTextLength >= nSearchLength)
    {
      int nLastIndex = 0;
      int nIndex;
      do
      {
        // Start searching from the last result
        nIndex = getIndexOf (sText, nLastIndex, sSearch);
        if (nIndex != STRING_NOT_FOUND)
        {
          // Match found
          ++ret;

          // Identify the next starting position (relative index + number of
          // search strings)
          nLastIndex = nIndex + nSearchLength;
        }
      } while (nIndex != STRING_NOT_FOUND);
    }
    return ret;
  }

  /**
   * Count the number of occurrences of sSearch within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param sSearch
   *        The text to search for. May be null.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return A non-negative number of occurrences.
   */
  @Nonnegative
  public static int getOccurrenceCountIgnoreCase (@Nullable final String sText,
                                                  @Nullable final String sSearch,
                                                  @Nonnull final Locale aSortLocale)
  {
    return sText != null && sSearch != null ? getOccurrenceCount (sText.toLowerCase (aSortLocale),
                                                                  sSearch.toLowerCase (aSortLocale))
                                            : 0;
  }

  /**
   * Count the number of occurrences of cSearch within sText.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The character to search for.
   * @return A non-negative number of occurrences.
   */
  @Nonnegative
  public static int getOccurrenceCount (@Nullable final String sText, final char cSearch)
  {
    int ret = 0;
    final int nTextLength = getLength (sText);
    if (nTextLength >= 1)
    {
      int nLastIndex = 0;
      int nIndex;
      do
      {
        // Start searching from the last result
        nIndex = getIndexOf (sText, nLastIndex, cSearch);
        if (nIndex != STRING_NOT_FOUND)
        {
          // Match found
          ++ret;

          // Identify the next starting position (relative index + number of
          // search strings)
          nLastIndex = nIndex + 1;
        }
      } while (nIndex != STRING_NOT_FOUND);
    }
    return ret;
  }

  /**
   * Count the number of occurrences of cSearch within sText ignoring case.
   *
   * @param sText
   *        The text to search in. May be null.
   * @param cSearch
   *        The character to search for.
   * @param aSortLocale
   *        The locale to be used for case unifying.
   * @return A non-negative number of occurrences.
   */
  @Nonnegative
  public static int getOccurrenceCountIgnoreCase (@Nullable final String sText,
                                                  final char cSearch,
                                                  @Nonnull final Locale aSortLocale)
  {
    return sText != null ? getOccurrenceCount (sText.toLowerCase (aSortLocale), Character.toLowerCase (cSearch)) : 0;
  }

  /**
   * Remove any leading whitespaces from the passed string.
   *
   * @param s
   *        the String to be trimmed
   * @return the original String with all leading whitespaces removed
   */
  @Nullable
  @CheckReturnValue
  public static String trimLeadingWhitespaces (@Nullable final String s)
  {
    final int n = getLeadingWhitespaceCount (s);
    return n == 0 ? s : s.substring (n, s.length ());
  }

  /**
   * Remove any trailing whitespaces from the passed string.
   *
   * @param s
   *        the String to be cut
   * @return the original String with all trailing whitespaces removed
   */
  @Nullable
  @CheckReturnValue
  public static String trimTrailingWhitespaces (@Nullable final String s)
  {
    final int n = getTrailingWhitespaceCount (s);
    return n == 0 ? s : s.substring (0, s.length () - n);
  }

  /**
   * Trim the passed lead from the source value. If the source value does not
   * start with the passed lead, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param sLead
   *        The string to be trimmed of the beginning
   * @return The trimmed string, or the original input string, if the lead was
   *         not found
   * @see #trimEnd(String, String)
   * @see #trimStartAndEnd(String, String)
   * @see #trimStartAndEnd(String, String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimStart (@Nullable final String sSrc, @Nullable final String sLead)
  {
    return startsWith (sSrc, sLead) ? sSrc.substring (sLead.length (), sSrc.length ()) : sSrc;
  }

  @Nullable
  @CheckReturnValue
  public static String trimStartRepeatedly (@Nullable final String sSrc, @Nullable final String sLead)
  {
    if (hasNoText (sSrc))
      return sSrc;

    final int nLeadLength = getLength (sLead);
    if (nLeadLength == 0)
      return sSrc;

    String ret = sSrc;
    while (startsWith (ret, sLead))
      ret = ret.substring (nLeadLength, ret.length ());
    return ret;
  }

  /**
   * Trim the passed lead from the source value. If the source value does not
   * start with the passed lead, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param cLead
   *        The char to be trimmed of the beginning
   * @return The trimmed string, or the original input string, if the lead was
   *         not found
   * @see #trimEnd(String, String)
   * @see #trimStartAndEnd(String, String)
   * @see #trimStartAndEnd(String, String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimStart (@Nullable final String sSrc, final char cLead)
  {
    return startsWith (sSrc, cLead) ? sSrc.substring (1, sSrc.length ()) : sSrc;
  }

  @Nullable
  @CheckReturnValue
  public static String trimStartRepeatedly (@Nullable final String sSrc, final char cLead)
  {
    if (hasNoText (sSrc))
      return sSrc;

    String ret = sSrc;
    while (startsWith (ret, cLead))
      ret = ret.substring (1, ret.length ());
    return ret;
  }

  /**
   * Trim the passed lead from the source value. If the source value does not
   * start with the passed lead, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param nCount
   *        The number of characters to trim at the end.
   * @return The trimmed string, or an empty string if nCount is ≥ the length
   *         of the source string
   */
  @Nullable
  @CheckReturnValue
  public static String trimStart (@Nullable final String sSrc, @Nonnegative final int nCount)
  {
    if (nCount <= 0)
      return sSrc;
    return getLength (sSrc) <= nCount ? "" : sSrc.substring (nCount, sSrc.length ());
  }

  /**
   * Trim the passed tail from the source value. If the source value does not
   * end with the passed tail, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param sTail
   *        The string to be trimmed of the end
   * @return The trimmed string, or the original input string, if the tail was
   *         not found
   * @see #trimStart(String, String)
   * @see #trimStartAndEnd(String, String)
   * @see #trimStartAndEnd(String, String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimEnd (@Nullable final String sSrc, @Nullable final String sTail)
  {
    return endsWith (sSrc, sTail) ? sSrc.substring (0, sSrc.length () - sTail.length ()) : sSrc;
  }

  @Nullable
  @CheckReturnValue
  public static String trimEndRepeatedly (@Nullable final String sSrc, @Nullable final String sTail)
  {
    if (hasNoText (sSrc))
      return sSrc;

    final int nTailLength = getLength (sTail);
    if (nTailLength == 0)
      return sSrc;

    String ret = sSrc;
    while (endsWith (ret, sTail))
      ret = ret.substring (0, ret.length () - nTailLength);
    return ret;
  }

  /**
   * Trim the passed tail from the source value. If the source value does not
   * end with the passed tail, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param cTail
   *        The char to be trimmed of the end
   * @return The trimmed string, or the original input string, if the tail was
   *         not found
   * @see #trimStart(String, String)
   * @see #trimStartAndEnd(String, String)
   * @see #trimStartAndEnd(String, String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimEnd (@Nullable final String sSrc, final char cTail)
  {
    return endsWith (sSrc, cTail) ? sSrc.substring (0, sSrc.length () - 1) : sSrc;
  }

  @Nullable
  @CheckReturnValue
  public static String trimEndRepeatedly (@Nullable final String sSrc, final char cTail)
  {
    if (hasNoText (sSrc))
      return sSrc;

    String ret = sSrc;
    while (endsWith (ret, cTail))
      ret = ret.substring (0, ret.length () - 1);
    return ret;
  }

  /**
   * Trim the passed tail from the source value. If the source value does not
   * end with the passed tail, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param nCount
   *        The number of characters to trim at the end.
   * @return The trimmed string, or an empty string if nCount is ≥ the length
   *         of the source string
   */
  @Nullable
  @CheckReturnValue
  public static String trimEnd (@Nullable final String sSrc, @Nonnegative final int nCount)
  {
    if (nCount <= 0)
      return sSrc;
    return getLength (sSrc) <= nCount ? "" : sSrc.substring (0, sSrc.length () - nCount);
  }

  /**
   * Trim the passed lead and tail from the source value. If the source value
   * does not start with the passed trimmed value, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param sValueToTrim
   *        The string to be trimmed of the beginning and the end
   * @return The trimmed string, or the original input string, if the value to
   *         trim was not found
   * @see #trimStart(String, String)
   * @see #trimEnd(String, String)
   * @see #trimStartAndEnd(String, String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimStartAndEnd (@Nullable final String sSrc, @Nullable final String sValueToTrim)
  {
    return trimStartAndEnd (sSrc, sValueToTrim, sValueToTrim);
  }

  /**
   * Trim the passed lead and tail from the source value. If the source value
   * does not start with the passed lead and does not end with the passed tail,
   * nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param sLead
   *        The string to be trimmed of the beginning
   * @param sTail
   *        The string to be trimmed of the end
   * @return The trimmed string, or the original input string, if the lead and
   *         the tail were not found
   * @see #trimStart(String, String)
   * @see #trimEnd(String, String)
   * @see #trimStartAndEnd(String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimStartAndEnd (@Nullable final String sSrc,
                                        @Nullable final String sLead,
                                        @Nullable final String sTail)
  {
    final String sInbetween = trimStart (sSrc, sLead);
    return trimEnd (sInbetween, sTail);
  }

  /**
   * Trim the passed lead and tail from the source value. If the source value
   * does not start with the passed trimmed value, nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param cValueToTrim
   *        The char to be trimmed of the beginning and the end
   * @return The trimmed string, or the original input string, if the value to
   *         trim was not found
   * @see #trimStart(String, String)
   * @see #trimEnd(String, String)
   * @see #trimStartAndEnd(String, String, String)
   */
  @Nullable
  @CheckReturnValue
  public static String trimStartAndEnd (@Nullable final String sSrc, final char cValueToTrim)
  {
    return trimStartAndEnd (sSrc, cValueToTrim, cValueToTrim);
  }

  /**
   * Trim the passed lead and tail from the source value. If the source value
   * does not start with the passed lead and does not end with the passed tail,
   * nothing happens.
   *
   * @param sSrc
   *        The input source string
   * @param cLead
   *        The char to be trimmed of the beginning
   * @param cTail
   *        The char to be trimmed of the end
   * @return The trimmed string, or the original input string, if the lead and
   *         the tail were not found
   * @see #trimStart(String, char)
   * @see #trimEnd(String, char)
   * @see #trimStartAndEnd(String, char)
   */
  @Nullable
  @CheckReturnValue
  public static String trimStartAndEnd (@Nullable final String sSrc, final char cLead, final char cTail)
  {
    final String sInbetween = trimStart (sSrc, cLead);
    return trimEnd (sInbetween, cTail);
  }

  /**
   * Trim the passed string, if it is not null.
   *
   * @param s
   *        The string to be trimmed. May be null.
   * @return null if the input string was null, the
   *         non-null trimmed string otherwise.
   * @see String#trim()
   */
  @Nullable
  @CheckReturnValue
  public static String trim (@Nullable final String s)
  {
    return hasNoText (s) ? s : s.trim ();
  }

  /**
   * Get the first character of the passed character sequence
   *
   * @param aCS
   *        The source character sequence
   * @return {@link CGlobal#ILLEGAL_CHAR} if the passed sequence was empty
   */
  public static char getFirstChar (@Nullable final CharSequence aCS)
  {
    return hasText (aCS) ? aCS.charAt (0) : CGlobal.ILLEGAL_CHAR;
  }

  /**
   * Get the first character of the passed array
   *
   * @param aChars
   *        The character array
   * @return {@link CGlobal#ILLEGAL_CHAR} if the passed array was empty
   */
  public static char getFirstChar (@Nullable final char [] aChars)
  {
    return ArrayHelper.getFirst (aChars, CGlobal.ILLEGAL_CHAR);
  }

  /**
   * Get the last character of the passed character sequence
   *
   * @param aCS
   *        The source character sequence
   * @return {@link CGlobal#ILLEGAL_CHAR} if the passed sequence was empty
   */
  public static char getLastChar (@Nullable final CharSequence aCS)
  {
    final int nLength = getLength (aCS);
    return nLength > 0 ? aCS.charAt (nLength - 1) : CGlobal.ILLEGAL_CHAR;
  }

  /**
   * Get the last character of the passed array
   *
   * @param aChars
   *        The character array
   * @return {@link CGlobal#ILLEGAL_CHAR} if the passed array was empty
   */
  public static char getLastChar (@Nullable final char [] aChars)
  {
    return ArrayHelper.getLast (aChars, CGlobal.ILLEGAL_CHAR);
  }

  @Nonnegative
  public static int getCharCount (@Nullable final String s, final char cSearch)
  {
    return s == null ? 0 : getCharCount (s.toCharArray (), cSearch);
  }

  @Nonnegative
  public static int getCharCount (@Nullable final char [] aChars, final char cSearch)
  {
    int ret = 0;
    if (aChars != null)
      for (final char c : aChars)
        if (c == cSearch)
          ++ret;
    return ret;
  }

  @Nonnegative
  public static int getLineCount (@Nullable final String s)
  {
    return getLineCount (s, '\n');
  }

  @Nonnegative
  public static int getLineCount (@Nullable final String s, final char cLineSep)
  {
    return 1 + getCharCount (s, cLineSep);
  }

  /**
   * Get the number of characters the passed value would occupy in a string
   * representation.
* Copied from java.lang.Integer#StringSize * * @param nValue * The integer value to check. May be be positive or negative. * @return Number of characters required. Alyways > 0. */ @Nonnegative public static int getCharacterCount (final int nValue) { // index is always one character less than the real size; that's why nPrefix // is 1 more! final int nPrefix = nValue < 0 ? 2 : 1; final int nRealValue = MathHelper.abs (nValue); for (int nIndex = 0;; nIndex++) if (nRealValue <= SIZE_TABLE_INT[nIndex]) return nPrefix + nIndex; } /** * Get the number of characters the passed value would occupy in a string * representation. * * @param nValue * The long value to check. May be be positive or negative. * @return Number of characters required. Always > 0. */ @Nonnegative public static int getCharacterCount (final long nValue) { // index is always one character less than the real size; that's why nPrefix // is 1 more! final int nPrefix = nValue < 0 ? 2 : 1; final long nRealValue = MathHelper.abs (nValue); for (int nIndex = 0;; nIndex++) if (nRealValue <= SIZE_TABLE_LONG[nIndex]) return nPrefix + nIndex; } @Nonnull public static String getCutAfterLength (@Nonnull final String sValue, @Nonnegative final int nMaxLength) { return getCutAfterLength (sValue, nMaxLength, null); } @Nonnull public static String getCutAfterLength (@Nonnull final String sValue, @Nonnegative final int nMaxLength, @Nullable final String sNewSuffix) { ValueEnforcer.notNull (sValue, "Value"); ValueEnforcer.isGE0 (nMaxLength, "MaxLength"); if (sValue.length () <= nMaxLength) return sValue; if (hasNoText (sNewSuffix)) return sValue.substring (0, nMaxLength); return sValue.substring (0, nMaxLength) + sNewSuffix; } /** * Same as {@link #replaceAll(String, String, CharSequence)} but allowing for * a null new-value, which is than interpreted as an empty string * instead. * * @param sInputString * The input string where the text should be replace. If this parameter * is null or empty, no replacement is done. * @param sSearchText * The string to be replaced. May neither be null nor * empty. * @param aReplacementText * The string with the replacement. May be null or empty. * @return The input string as is, if the input string is empty or if the * string to be replaced is not contained. */ public static String replaceAllSafe (@Nullable final String sInputString, @Nonnull final String sSearchText, @Nullable final CharSequence aReplacementText) { return replaceAll (sInputString, sSearchText, getNotNull (aReplacementText, "")); } /** * This is a fast replacement for * {@link String#replace(CharSequence, CharSequence)}. The problem with the * mentioned {@link String} method is, that is uses internally regular * expressions which use a synchronized block to compile the patterns. This * method is inherently thread safe since {@link String} is immutable and * we're operating on different temporary {@link StringBuilder} objects. * * @param sInputString * The input string where the text should be replace. If this parameter * is null or empty, no replacement is done. * @param sSearchText * The string to be replaced. May neither be null nor * empty. * @param aReplacementText * The string with the replacement. May not be null but * may be empty. * @return The input string as is, if the input string is empty or if the * search pattern and the replacement are equal or if the string to be * replaced is not contained. */ @Nullable public static String replaceAll (@Nullable final String sInputString, @Nonnull final String sSearchText, @Nonnull final CharSequence aReplacementText) { ValueEnforcer.notEmpty (sSearchText, "SearchText"); ValueEnforcer.notNull (aReplacementText, "ReplacementText"); // Is input string empty? if (hasNoText (sInputString)) return sInputString; // Replace old with the same new? final int nOldLength = sSearchText.length (); final int nNewLength = aReplacementText.length (); if (nOldLength == nNewLength) { // Any change? if (sSearchText.equals (aReplacementText)) return sInputString; if (nOldLength == 1) { // Use char version which is more efficient return replaceAll (sInputString, sSearchText.charAt (0), aReplacementText.charAt (0)); } } // Does the old text occur anywhere? int nIndex = sInputString.indexOf (sSearchText, 0); if (nIndex == STRING_NOT_FOUND) return sInputString; // build output buffer final StringBuilder ret = new StringBuilder (nOldLength >= nNewLength ? sInputString.length () : sInputString.length () * 2); int nOldIndex = 0; do { ret.append (sInputString, nOldIndex, nIndex).append (aReplacementText); nIndex += nOldLength; nOldIndex = nIndex; nIndex = sInputString.indexOf (sSearchText, nIndex); } while (nIndex != STRING_NOT_FOUND); ret.append (sInputString, nOldIndex, sInputString.length ()); return ret.toString (); } /** * This is a fast replacement for {@link String#replace(char, char)} for * characters. The problem with the mentioned String method is, that is uses * internally regular expressions which use a synchronized block to compile * the patterns. This method is inherently thread safe since {@link String} is * immutable and we're operating on different temporary {@link StringBuilder} * objects. * * @param sInputString * The input string where the text should be replace. If this parameter * is null or empty, no replacement is done. * @param cSearchChar * The character to be replaced. * @param cReplacementChar * The character with the replacement. * @return The input string as is, if the input string is empty or if the * search pattern and the replacement are equal or if the string to be * replaced is not contained. */ @Nullable public static String replaceAll (@Nullable final String sInputString, final char cSearchChar, final char cReplacementChar) { // Is input string empty? if (hasNoText (sInputString)) return sInputString; // Replace old with the same new? if (cSearchChar == cReplacementChar) return sInputString; // Does the old text occur anywhere? int nIndex = sInputString.indexOf (cSearchChar, 0); if (nIndex == STRING_NOT_FOUND) return sInputString; // build output buffer final StringBuilder ret = new StringBuilder (sInputString.length ()); int nOldIndex = 0; do { ret.append (sInputString, nOldIndex, nIndex).append (cReplacementChar); nIndex++; nOldIndex = nIndex; nIndex = sInputString.indexOf (cSearchChar, nIndex); } while (nIndex != STRING_NOT_FOUND); ret.append (sInputString, nOldIndex, sInputString.length ()); return ret.toString (); } /** * Just calls replaceAll as long as there are still replacements * found * * @param sInputString * The input string where the text should be replace. If this parameter * is null or empty, no replacement is done. * @param sSearchText * The string to be replaced. May neither be null nor * empty. * @param sReplacementText * The string with the replacement. May not be null but * may be empty. * @return The input string as is, if the input string is empty or if the * string to be replaced is not contained. */ @Nullable public static String replaceAllRepeatedly (@Nullable final String sInputString, @Nonnull final String sSearchText, @Nonnull final String sReplacementText) { ValueEnforcer.notEmpty (sSearchText, "SearchText"); ValueEnforcer.notNull (sReplacementText, "ReplacementText"); ValueEnforcer.isFalse (sReplacementText.contains (sSearchText), "Loop detection: replacementText must not contain searchText"); // Is input string empty? if (hasNoText (sInputString)) return sInputString; String sRet = sInputString; String sLastLiteral; do { sLastLiteral = sRet; sRet = replaceAll (sRet, sSearchText, sReplacementText); } while (!sLastLiteral.equals (sRet)); return sRet; } /** * Get the result length (in characters) when replacing all patterns with the * replacements on the passed input array. * * @param aInputString * Input char array. May not be null. * @param aSearchChars * The one-character search patterns. May not be null. * @param aReplacementStrings * The replacements to be performed. May not be null. The * first dimension of this array must have exactly the same amount of * elements as the patterns parameter array. * @return {@link CGlobal#ILLEGAL_UINT} if no replacement was needed, and * therefore the length of the input array could be used. */ public static int getReplaceMultipleResultLength (@Nonnull final char [] aInputString, @Nonnull @Nonempty final char [] aSearchChars, @Nonnull @Nonempty final char [] [] aReplacementStrings) { int nResultLen = 0; boolean bAnyReplacement = false; for (final char cInput : aInputString) { // In case no replacement is found use a single char int nReplacementLength = 1; for (int nIndex = 0; nIndex < aSearchChars.length; nIndex++) if (cInput == aSearchChars[nIndex]) { nReplacementLength = aReplacementStrings[nIndex].length; bAnyReplacement = true; break; } nResultLen += nReplacementLength; } return bAnyReplacement ? nResultLen : CGlobal.ILLEGAL_UINT; } /** * Optimized replace method that replaces a set of characters with a set of * strings. This method was created for efficient XML special character * replacements! * * @param sInputString * The input string. * @param aSearchChars * The characters to replace. * @param aReplacementStrings * The new strings to be inserted instead. Must have the same array * length as aPatterns. * @return The replaced version of the string or an empty char array if the * input string was null. */ @Nonnull public static char [] replaceMultiple (@Nullable final String sInputString, @Nonnull final char [] aSearchChars, @Nonnull final char [] [] aReplacementStrings) { // Any input text? if (hasNoText (sInputString)) return ArrayHelper.EMPTY_CHAR_ARRAY; return replaceMultiple (sInputString.toCharArray (), aSearchChars, aReplacementStrings); } /** * Optimized replace method that replaces a set of characters with a set of * strings. This method was created for efficient XML special character * replacements! * * @param aInput * The input string. * @param aSearchChars * The characters to replace. * @param aReplacementStrings * The new strings to be inserted instead. Must have the same array * length as aPatterns. * @return The replaced version of the string or an empty char array if the * input string was null. */ @Nonnull public static char [] replaceMultiple (@Nullable final char [] aInput, @Nonnull final char [] aSearchChars, @Nonnull final char [] [] aReplacementStrings) { ValueEnforcer.notNull (aSearchChars, "SearchChars"); ValueEnforcer.notNull (aReplacementStrings, "ReplacementStrings"); ValueEnforcer.isEqual (aSearchChars.length, aReplacementStrings.length, "array length mismatch"); // Any input text? if (aInput == null || aInput.length == 0) return ArrayHelper.EMPTY_CHAR_ARRAY; // Any replacement patterns? if (aSearchChars.length == 0) return aInput; // get result length final int nResultLen = getReplaceMultipleResultLength (aInput, aSearchChars, aReplacementStrings); // nothing to replace in here? if (nResultLen == CGlobal.ILLEGAL_UINT) return aInput; // build result final char [] aOutput = new char [nResultLen]; int nOutputIndex = 0; // For all input chars for (final char cInput : aInput) { boolean bFoundReplacement = false; for (int nPatternIndex = 0; nPatternIndex < aSearchChars.length; nPatternIndex++) { if (cInput == aSearchChars[nPatternIndex]) { final char [] aReplacement = aReplacementStrings[nPatternIndex]; final int nReplacementLength = aReplacement.length; System.arraycopy (aReplacement, 0, aOutput, nOutputIndex, nReplacementLength); nOutputIndex += nReplacementLength; bFoundReplacement = true; break; } } if (!bFoundReplacement) { // copy char as is aOutput[nOutputIndex++] = cInput; } } return aOutput; } /** * Specialized version of {@link #replaceMultiple(String, char[], char[][])} * where the object where the output should be appended is passed in as a * parameter. This has the advantage, that not length calculation needs to * take place! * * @param sInputString * The input string. * @param aSearchChars * The characters to replace. * @param aReplacementStrings * The new strings to be inserted instead. Must have the same array * length as aPatterns. * @param aTarget * Where the replaced objects should be written to. May not be * null. * @return The number of replacements performed. Always ≥ 0. * @throws IOException * In case writing to the Writer fails */ @Nonnegative public static int replaceMultipleTo (@Nullable final String sInputString, @Nonnull final char [] aSearchChars, @Nonnull final char [] [] aReplacementStrings, @Nonnull final Writer aTarget) throws IOException { if (hasNoText (sInputString)) return 0; return replaceMultipleTo (sInputString.toCharArray (), aSearchChars, aReplacementStrings, aTarget); } /** * Specialized version of {@link #replaceMultiple(String, char[], char[][])} * where the object where the output should be appended is passed in as a * parameter. This has the advantage, that not length calculation needs to * take place! * * @param aInput * The input char array. May not be null. * @param aSearchChars * The characters to replace. * @param aReplacementStrings * The new strings to be inserted instead. Must have the same array * length as aPatterns. * @param aTarget * Where the replaced objects should be written to. May not be * null. * @return The number of replacements performed. Always ≥ 0. * @throws IOException * In case writing to the Writer fails */ @Nonnegative public static int replaceMultipleTo (@Nullable final char [] aInput, @Nonnull final char [] aSearchChars, @Nonnull final char [] [] aReplacementStrings, @Nonnull final Writer aTarget) throws IOException { return aInput == null ? 0 : replaceMultipleTo (aInput, 0, aInput.length, aSearchChars, aReplacementStrings, aTarget); } /** * Specialized version of {@link #replaceMultiple(String, char[], char[][])} * where the object where the output should be appended is passed in as a * parameter. This has the advantage, that not length calculation needs to * take place! * * @param aInput * The input char array. May be null. * @param nOfs * Offset into input array. Must be ≥ 0. * @param nLen * Number of characters from input array. Must be ≥ 0. * @param aSearchChars * The characters to replace. * @param aReplacementStrings * The new strings to be inserted instead. Must have the same array * length as aPatterns. * @param aTarget * Where the replaced objects should be written to. May not be * null. * @return The number of replacements performed. Always ≥ 0. * @throws IOException * In case writing to the Writer fails */ @Nonnegative public static int replaceMultipleTo (@Nullable final char [] aInput, @Nonnegative final int nOfs, @Nonnegative final int nLen, @Nonnull final char [] aSearchChars, @Nonnull final char [] [] aReplacementStrings, @Nonnull final Writer aTarget) throws IOException { if (aInput != null) ValueEnforcer.isArrayOfsLen (aInput, nOfs, nLen); ValueEnforcer.notNull (aSearchChars, "SearchChars"); ValueEnforcer.notNull (aReplacementStrings, "ReplacementStrings"); ValueEnforcer.isEqual (aSearchChars.length, aReplacementStrings.length, "array length mismatch"); ValueEnforcer.notNull (aTarget, "Target"); if (aInput == null || aInput.length == 0 || nLen == 0) return 0; if (aSearchChars.length == 0) { // No modifications required aTarget.write (aInput, nOfs, nLen); return 0; } // for all input string characters int nFirstNonReplace = nOfs; int nInputIndex = nOfs; int nTotalReplacements = 0; final int nMaxSearchChars = aSearchChars.length; for (int i = 0; i < nLen; ++i) { final char cInput = aInput[nOfs + i]; for (int nPatternIndex = 0; nPatternIndex < nMaxSearchChars; nPatternIndex++) { if (cInput == aSearchChars[nPatternIndex]) { if (nFirstNonReplace < nInputIndex) aTarget.write (aInput, nFirstNonReplace, nInputIndex - nFirstNonReplace); nFirstNonReplace = nInputIndex + 1; aTarget.write (aReplacementStrings[nPatternIndex]); ++nTotalReplacements; break; } } nInputIndex++; } if (nFirstNonReplace < nInputIndex) aTarget.write (aInput, nFirstNonReplace, nInputIndex - nFirstNonReplace); return nTotalReplacements; } /** * Optimized replace method that replaces a set of characters with another * character. This method was created for efficient unsafe character * replacements! * * @param sInputString * The input string. * @param aSearchChars * The characters to replace. * @param cReplacementChar * The new char to be used instead of the search chars. * @return The replaced version of the string or an empty char array if the * input string was null. */ @Nonnull public static char [] replaceMultiple (@Nullable final String sInputString, @Nonnull final char [] aSearchChars, final char cReplacementChar) { ValueEnforcer.notNull (aSearchChars, "SearchChars"); // Any input text? if (hasNoText (sInputString)) return ArrayHelper.EMPTY_CHAR_ARRAY; // Get char array final char [] aInput = sInputString.toCharArray (); // Any replacement patterns? if (aSearchChars.length == 0) return aInput; // build result final char [] aOutput = new char [aInput.length]; int nOutputIndex = 0; for (final char c : aInput) { if (ArrayHelper.contains (aSearchChars, c)) aOutput[nOutputIndex] = cReplacementChar; else aOutput[nOutputIndex] = c; nOutputIndex++; } return aOutput; } /** * Optimized replace method that replaces a set of characters with another * character. This method was created for efficient unsafe character * replacements! * * @param sInputString * The input string. * @param aSearchChars * The characters to replace. * @param cReplacementChar * The new char to be used instead of the search chars. * @param aTarget * The target StringBuilder to write the result to. May not be * null. */ public static void replaceMultipleTo (@Nullable final String sInputString, @Nonnull final char [] aSearchChars, final char cReplacementChar, @Nonnull final StringBuilder aTarget) { ValueEnforcer.notNull (aSearchChars, "SearchChars"); ValueEnforcer.notNull (aTarget, "Target"); // Any input text? if (hasText (sInputString)) { // Any search chars? if (aSearchChars.length == 0) { aTarget.append (sInputString); } else { // Perform the replacement for (final char c : sInputString.toCharArray ()) { if (ArrayHelper.contains (aSearchChars, c)) aTarget.append (cReplacementChar); else aTarget.append (c); } } } } /** * Optimized replace method that replaces a set of characters with another * character. This method was created for efficient unsafe character * replacements! * * @param sInputString * The input string. * @param aSearchChars * The characters to replace. * @param cReplacementChar * The new char to be used instead of the search chars. * @param aTarget * The target writer to write the result to. May not be * null. * @throws IOException * in case writing to the Writer fails * @since 8.6.3 */ public static void replaceMultipleTo (@Nullable final String sInputString, @Nonnull final char [] aSearchChars, final char cReplacementChar, @Nonnull final Writer aTarget) throws IOException { ValueEnforcer.notNull (aSearchChars, "SearchChars"); ValueEnforcer.notNull (aTarget, "Target"); // Any input text? if (hasText (sInputString)) { // Any search chars? if (aSearchChars.length == 0) { aTarget.write (sInputString); } else { // Perform the replacement for (final char c : sInputString.toCharArray ()) { if (ArrayHelper.contains (aSearchChars, c)) aTarget.write (cReplacementChar); else aTarget.write (c); } } } } /** * Optimized replace method that replaces a set of characters with another * character. This method was created for efficient unsafe character * replacements! * * @param sInputString * The input string. * @param aSearchChars * The characters to replace. * @param cReplacementChar * The new char to be used instead of the search chars. * @return The replaced version of the string or an empty char array if the * input string was null. * @since 8.6.3 */ @Nonnull public static String replaceMultipleAsString (@Nullable final String sInputString, @Nonnull final char [] aSearchChars, final char cReplacementChar) { ValueEnforcer.notNull (aSearchChars, "SearchChars"); if (hasNoText (sInputString)) return ""; final StringBuilder aSB = new StringBuilder (sInputString.length ()); replaceMultipleTo (sInputString, aSearchChars, cReplacementChar, aSB); return aSB.toString (); } /** * Perform all string replacements on the input string as defined by the * passed map. All replacements are done using * {@link #replaceAll(String,String,CharSequence)} which is ok. * * @param sInputString * The input string where the text should be replaced. May be * null. * @param aTransTable * The map with the replacements to execute. If null is * passed, the input string is not altered. * @return null if the input string was null. */ @Nullable public static String replaceMultiple (@Nullable final String sInputString, @Nullable final Map aTransTable) { if (hasNoText (sInputString) || aTransTable == null || aTransTable.isEmpty ()) return sInputString; String sOutput = sInputString; for (final Entry aEntry : aTransTable.entrySet ()) sOutput = replaceAll (sOutput, aEntry.getKey (), aEntry.getValue ()); return sOutput; } /** * Perform all string replacements on the input string as defined by the * passed map. All replacements are done using * {@link #replaceAll(String,String,CharSequence)} which is ok. * * @param sInputString * The input string where the text should be replaced. May be * null. * @param aSearchTexts * The texts to be searched. If null is passed, the input * string is not altered. * @param aReplacementTexts * The texts to be used as the replacements. This array must have * exactly the same number of elements than the searched texts! If * null is passed, the input string is not altered. * @return null if the input string was null. The * unmodified input string if no search/replace patterns where * provided. */ @Nullable public static String replaceMultiple (@Nullable final String sInputString, @Nullable final String [] aSearchTexts, @Nullable final String [] aReplacementTexts) { if (hasNoText (sInputString)) return sInputString; final int nSearchTextLength = aSearchTexts == null ? 0 : aSearchTexts.length; final int nReplacementTextLength = aReplacementTexts == null ? 0 : aReplacementTexts.length; if (nSearchTextLength != nReplacementTextLength) throw new IllegalArgumentException ("Array length mismatch!"); // Nothing to replace? if (nSearchTextLength == 0) return sInputString; String sOutput = sInputString; for (int nIndex = 0; nIndex < nSearchTextLength; ++nIndex) sOutput = replaceAll (sOutput, aSearchTexts[nIndex], aReplacementTexts[nIndex]); return sOutput; } /** * Remove all occurrences of the passed character from the specified input * string * * @param sInputString * The input string where the character should be removed. If this * parameter is null or empty, no removing is done. * @param cRemoveChar * The character to be removed. * @return The input string as is, if the input string is empty or if the * remove char is not contained. */ @Nullable public static String removeAll (@Nullable final String sInputString, final char cRemoveChar) { // Is input string empty? if (hasNoText (sInputString)) return sInputString; // Does the char occur anywhere? final int nFirstIndex = sInputString.indexOf (cRemoveChar, 0); if (nFirstIndex == STRING_NOT_FOUND) return sInputString; // build output buffer final char [] aChars = sInputString.toCharArray (); final int nMax = aChars.length; final StringBuilder aSB = new StringBuilder (nMax); // Copy the first chars where we know it is not contained aSB.append (aChars, 0, nFirstIndex); // Start searching after the first occurrence because we know that this is a // char to be removed for (int i = nFirstIndex; i < nMax; ++i) { final char c = aChars[i]; if (c != cRemoveChar) aSB.append (c); } return aSB.toString (); } /** * Remove all occurrences of the passed character from the specified input * string * * @param sInputString * The input string where the character should be removed. If this * parameter is null or empty, no removing is done. * @param sRemoveString * The String to be removed. May be null or empty in which * case nothing happens. * @return The input string as is, if the input string is empty or if the * remove string is empty or not contained. */ @Nullable public static String removeAll (@Nullable final String sInputString, @Nullable final String sRemoveString) { // Is input string empty? if (hasNoText (sInputString)) return sInputString; final int nRemoveLength = getLength (sRemoveString); if (nRemoveLength == 0) { // Nothing to be removed return sInputString; } if (nRemoveLength == 1) { // Shortcut to char version return removeAll (sInputString, sRemoveString.charAt (0)); } // Does the string occur anywhere? int nIndex = sInputString.indexOf (sRemoveString, 0); if (nIndex == STRING_NOT_FOUND) return sInputString; // build output buffer final StringBuilder ret = new StringBuilder (sInputString.length ()); int nOldIndex = 0; do { ret.append (sInputString, nOldIndex, nIndex); nOldIndex = nIndex + nRemoveLength; nIndex = sInputString.indexOf (sRemoveString, nOldIndex); } while (nIndex != STRING_NOT_FOUND); ret.append (sInputString, nOldIndex, sInputString.length ()); return ret.toString (); } /** * Get the length of the passed character sequence. * * @param aCS * The character sequence who's length is to be determined. May be * null. * @return 0 if the parameter is null, its length otherwise. * @see CharSequence#length() */ @Nonnegative public static int getLength (@Nullable final CharSequence aCS) { return aCS == null ? 0 : aCS.length (); } /** * Get the passed string but never return null. If the passed * parameter is null an empty string is returned. * * @param s * The parameter to be not null. * @return An empty string if the passed parameter is null, the * passed string otherwise. */ @Nonnull public static String getNotNull (@Nullable final String s) { return getNotNull (s, ""); } /** * Get the passed string but never return null. If the passed * parameter is null the second parameter is returned. * * @param s * The parameter to be not null. * @param sDefaultIfNull * The value to be used if the first parameter is null. * May be null but in this case the call to this method is * obsolete. * @return The passed default value if the string is null, * otherwise the input string. */ @Nullable public static String getNotNull (@Nullable final String s, @Nullable final String sDefaultIfNull) { return s == null ? sDefaultIfNull : s; } /** * Get the passed string but never return null. If the passed * parameter is null the second parameter is returned. * * @param s * The parameter to be not null. * @param aDefaultIfNull * The value supplier to be used if the first parameter is * null. May not be null. * @return The passed default value if the string is null, * otherwise the input string. * @since 10.2.0 */ @Nullable public static String getNotNull (@Nullable final String s, @Nonnull final Supplier aDefaultIfNull) { return s == null ? aDefaultIfNull.get () : s; } /** * Get the passed {@link CharSequence} but never return null. If * the passed parameter is null an empty string is returned. * * @param s * The parameter to be not null. * @return An empty string if the passed parameter is null, the * passed {@link CharSequence} otherwise. */ @Nonnull public static CharSequence getNotNull (@Nullable final CharSequence s) { return getNotNull (s, ""); } /** * Get the passed {@link CharSequence} but never return null. If * the passed parameter is null the second parameter is returned. * * @param s * The parameter to be not null. * @param sDefaultIfNull * The value to be used if the first parameter is null. * May be null but in this case the call to this method is * obsolete. * @return The passed default value if the string is null, * otherwise the input {@link CharSequence}. */ @Nullable public static CharSequence getNotNull (@Nullable final CharSequence s, @Nullable final CharSequence sDefaultIfNull) { return s == null ? sDefaultIfNull : s; } /** * Get the passed {@link CharSequence} but never return null. If * the passed parameter is null the second parameter is returned. * * @param s * The parameter to be not null. * @param aDefaultIfNull * The value supplier to be used if the first parameter is * null. May not be null. * @return The passed default value if the string is null, * otherwise the input {@link CharSequence}. * @since 10.2.0 */ @Nullable public static CharSequence getNotNull (@Nullable final CharSequence s, @Nonnull final Supplier aDefaultIfNull) { return s == null ? aDefaultIfNull.get () : s; } /** * Get the passed string but never return an empty string. If the passed * parameter is null or empty the second parameter is returned. * * @param s * The parameter to be not null nor empty. * @param sDefaultIfEmpty * The value to be used if the first parameter is null or * empty. May be null but in this case the call to this * method is obsolete. * @return The passed default value if the string is null or * empty, otherwise the input string. */ @Nullable public static String getNotEmpty (@Nullable final String s, @Nullable final String sDefaultIfEmpty) { return hasNoText (s) ? sDefaultIfEmpty : s; } /** * Get the passed string but never return an empty string. If the passed * parameter is null or empty the second parameter is returned. * * @param s * The parameter to be not null nor empty. * @param aDefaultIfEmpty * The value supplier to be used if the first parameter is * null or empty. May not be null. * @return The passed default value if the string is null or * empty, otherwise the input string. * @since 10.2.0 */ @Nullable public static String getNotEmpty (@Nullable final String s, @Nonnull final Supplier aDefaultIfEmpty) { return hasNoText (s) ? aDefaultIfEmpty.get () : s; } /** * Get the passed char sequence but never return an empty char sequence. If * the passed parameter is null or empty the second parameter is * returned. * * @param s * The parameter to be not null nor empty. * @param sDefaultIfEmpty * The value to be used if the first parameter is null or * empty. May be null but in this case the call to this * method is obsolete. * @return The passed default value if the char sequence is null * or empty, otherwise the input char sequence. */ @Nullable public static CharSequence getNotEmpty (@Nullable final CharSequence s, @Nullable final CharSequence sDefaultIfEmpty) { return hasNoText (s) ? sDefaultIfEmpty : s; } /** * Get the passed char sequence but never return an empty char sequence. If * the passed parameter is null or empty the second parameter is * returned. * * @param s * The parameter to be not null nor empty. * @param aDefaultIfEmpty * The value supplier to be used if the first parameter is * null or empty. May not be null. * @return The passed default value if the char sequence is null * or empty, otherwise the input char sequence. * @since 10.2.0 */ @Nullable public static CharSequence getNotEmpty (@Nullable final CharSequence s, @Nullable final Supplier aDefaultIfEmpty) { return hasNoText (s) ? aDefaultIfEmpty.get () : s; } /** * Convert the passed object to a string using the {@link Object#toString()} * method. * * @param aObject * The value to be converted. May be null. * @return An empty string in case the passed object was null. * Never null. * @see Object#toString() */ @Nonnull public static String getToString (@Nullable final Object aObject) { return getToString (aObject, ""); } /** * Convert the passed object to a string using the {@link Object#toString()} * method or otherwise return the passed default value. * * @param aObject * The value to be converted. May be null. * @param sNullValue * The value to be returned in case the passed object is * null. May be null itself. * @return The passed default value in case the passed object was * null or the result of {@link Object#toString()} on the * passed object. * @see Object#toString() */ @Nullable public static String getToString (@Nullable final Object aObject, @Nullable final String sNullValue) { return aObject == null ? sNullValue : aObject.toString (); } /** * Get the passed string without the first char. * * @param sStr * The source string. May be null. * @return An empty, non-null string if the passed string has a * length ≤ 1. */ @Nonnull public static String getWithoutLeadingChar (@Nullable final String sStr) { return getWithoutLeadingChars (sStr, 1); } /** * Get the passed string without the specified number of leading chars. * * @param sStr * The source string. May be null. * @param nCount * The number of chars to remove. * @return An empty, non-null string if the passed string has a * length ≤ nCount. */ @Nonnull public static String getWithoutLeadingChars (@Nullable final String sStr, @Nonnegative final int nCount) { ValueEnforcer.isGE0 (nCount, "Count"); if (nCount == 0) return sStr; return getLength (sStr) <= nCount ? "" : sStr.substring (nCount); } /** * Get the passed string without the last char. * * @param sStr * The source string. May be null. * @return An empty, non-null string if the passed string has a * length ≤ 1. */ @Nonnull public static String getWithoutTrailingChar (@Nullable final String sStr) { return getWithoutTrailingChars (sStr, 1); } /** * Get the passed string without the specified number of trailing chars. * * @param sStr * The source string. May be null. * @param nCount * The number of chars to remove. * @return An empty, non-null string if the passed string has a * length ≤ nCount. */ @Nonnull public static String getWithoutTrailingChars (@Nullable final String sStr, @Nonnegative final int nCount) { ValueEnforcer.isGE0 (nCount, "Count"); if (nCount == 0) return sStr; final int nLength = getLength (sStr); return nLength <= nCount ? "" : sStr.substring (0, nLength - nCount); } /** * Get the passed string where all spaces (white spaces or unicode spaces) * have been removed. * * @param sStr * The source string. May be null * @return A non-null string representing the passed string * without any spaces */ @Nonnull public static String getWithoutAnySpaces (@Nullable final String sStr) { if (sStr == null) return ""; // Trim first final char [] aChars = sStr.trim ().toCharArray (); final StringBuilder aResult = new StringBuilder (aChars.length); for (final char c : aChars) if (!Character.isWhitespace (c) && !Character.isSpaceChar (c)) aResult.append (c); return aResult.toString (); } @Nullable private static String _getUntilFirst (@Nullable final String sStr, final char cSearch, final boolean bIncludingSearchChar) { final int nIndex = getIndexOf (sStr, cSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (0, nIndex + (bIncludingSearchChar ? 1 : 0)); } /** * Get everything from the string up to and including the first passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getUntilFirstIncl (@Nullable final String sStr, final char cSearch) { return _getUntilFirst (sStr, cSearch, true); } /** * Get everything from the string up to and excluding first the passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getUntilFirstExcl (@Nullable final String sStr, final char cSearch) { return _getUntilFirst (sStr, cSearch, false); } @Nullable private static String _getUntilFirst (@Nullable final String sStr, @Nullable final String sSearch, final boolean bIncludingSearchChar) { if (hasNoText (sSearch)) return ""; final int nIndex = getIndexOf (sStr, sSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (0, nIndex + (bIncludingSearchChar ? sSearch.length () : 0)); } /** * Get everything from the string up to and including the first passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the empty string is * returned. */ @Nullable public static String getUntilFirstIncl (@Nullable final String sStr, @Nullable final String sSearch) { return _getUntilFirst (sStr, sSearch, true); } /** * Get everything from the string up to and excluding the first passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the empty string is * returned. */ @Nullable public static String getUntilFirstExcl (@Nullable final String sStr, @Nullable final String sSearch) { return _getUntilFirst (sStr, sSearch, false); } @Nullable private static String _getUntilLast (@Nullable final String sStr, final char cSearch, final boolean bIncludingSearchChar) { final int nIndex = getLastIndexOf (sStr, cSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (0, nIndex + (bIncludingSearchChar ? 1 : 0)); } /** * Get everything from the string up to and including the first passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getUntilLastIncl (@Nullable final String sStr, final char cSearch) { return _getUntilLast (sStr, cSearch, true); } /** * Get everything from the string up to and excluding first the passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getUntilLastExcl (@Nullable final String sStr, final char cSearch) { return _getUntilLast (sStr, cSearch, false); } @Nullable private static String _getUntilLast (@Nullable final String sStr, @Nullable final String sSearch, final boolean bIncludingSearchChar) { if (hasNoText (sSearch)) return ""; final int nIndex = getLastIndexOf (sStr, sSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (0, nIndex + (bIncludingSearchChar ? sSearch.length () : 0)); } /** * Get everything from the string up to and including the first passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the empty string is * returned. */ @Nullable public static String getUntilLastIncl (@Nullable final String sStr, @Nullable final String sSearch) { return _getUntilLast (sStr, sSearch, true); } /** * Get everything from the string up to and excluding the first passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the empty string is * returned. */ @Nullable public static String getUntilLastExcl (@Nullable final String sStr, @Nullable final String sSearch) { return _getUntilLast (sStr, sSearch, false); } @Nullable private static String _getFromFirst (@Nullable final String sStr, final char cSearch, final boolean bIncludingSearchChar) { final int nIndex = getIndexOf (sStr, cSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (nIndex + (bIncludingSearchChar ? 0 : 1)); } /** * Get everything from the string from and including the first passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getFromFirstIncl (@Nullable final String sStr, final char cSearch) { return _getFromFirst (sStr, cSearch, true); } /** * Get everything from the string from and excluding the first passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getFromFirstExcl (@Nullable final String sStr, final char cSearch) { return _getFromFirst (sStr, cSearch, false); } @Nullable private static String _getFromFirst (@Nullable final String sStr, @Nullable final String sSearch, final boolean bIncludingSearchString) { if (hasNoText (sSearch)) return sStr; final int nIndex = getIndexOf (sStr, sSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (nIndex + (bIncludingSearchString ? 0 : sSearch.length ())); } /** * Get everything from the string from and including the passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the input string is returned * unmodified. */ @Nullable public static String getFromFirstIncl (@Nullable final String sStr, @Nullable final String sSearch) { return _getFromFirst (sStr, sSearch, true); } /** * Get everything from the string from and excluding the passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the input string is returned * unmodified. */ @Nullable public static String getFromFirstExcl (@Nullable final String sStr, @Nullable final String sSearch) { return _getFromFirst (sStr, sSearch, false); } @Nullable private static String _getFromLast (@Nullable final String sStr, final char cSearch, final boolean bIncludingSearchChar) { final int nIndex = getLastIndexOf (sStr, cSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (nIndex + (bIncludingSearchChar ? 0 : 1)); } /** * Get everything from the string from and including the first passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getFromLastIncl (@Nullable final String sStr, final char cSearch) { return _getFromLast (sStr, cSearch, true); } /** * Get everything from the string from and excluding the first passed char. * * @param sStr * The source string. May be null. * @param cSearch * The character to search. * @return null if the passed string does not contain the search * character. */ @Nullable public static String getFromLastExcl (@Nullable final String sStr, final char cSearch) { return _getFromLast (sStr, cSearch, false); } @Nullable private static String _getFromLast (@Nullable final String sStr, @Nullable final String sSearch, final boolean bIncludingSearchString) { if (hasNoText (sSearch)) return sStr; final int nIndex = getLastIndexOf (sStr, sSearch); if (nIndex == STRING_NOT_FOUND) return null; return sStr.substring (nIndex + (bIncludingSearchString ? 0 : sSearch.length ())); } /** * Get everything from the string from and including the passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the input string is returned * unmodified. */ @Nullable public static String getFromLastIncl (@Nullable final String sStr, @Nullable final String sSearch) { return _getFromLast (sStr, sSearch, true); } /** * Get everything from the string from and excluding the passed string. * * @param sStr * The source string. May be null. * @param sSearch * The string to search. May be null. * @return null if the passed string does not contain the search * string. If the search string is empty, the input string is returned * unmodified. */ @Nullable public static String getFromLastExcl (@Nullable final String sStr, @Nullable final String sSearch) { return _getFromLast (sStr, sSearch, false); } /** * Get the first token up to (and excluding) the separating character. * * @param sStr * The string to search. May be null. * @param cSearch * The search character. * @return The passed string if no such separator token was found. */ @Nullable public static String getFirstToken (@Nullable final String sStr, final char cSearch) { final int nIndex = getIndexOf (sStr, cSearch); return nIndex == StringHelper.STRING_NOT_FOUND ? sStr : sStr.substring (0, nIndex); } /** * Get the first token up to (and excluding) the separating string. * * @param sStr * The string to search. May be null. * @param sSearch * The search string. May be null. * @return The passed string if no such separator token was found. */ @Nullable public static String getFirstToken (@Nullable final String sStr, @Nullable final String sSearch) { if (StringHelper.hasNoText (sSearch)) return sStr; final int nIndex = getIndexOf (sStr, sSearch); return nIndex == StringHelper.STRING_NOT_FOUND ? sStr : sStr.substring (0, nIndex); } /** * Get the last token from (and excluding) the separating character. * * @param sStr * The string to search. May be null. * @param cSearch * The search character. * @return The passed string if no such separator token was found. */ @Nullable public static String getLastToken (@Nullable final String sStr, final char cSearch) { final int nIndex = getLastIndexOf (sStr, cSearch); return nIndex == StringHelper.STRING_NOT_FOUND ? sStr : sStr.substring (nIndex + 1); } /** * Get the last token from (and excluding) the separating string. * * @param sStr * The string to search. May be null. * @param sSearch * The search string. May be null. * @return The passed string if no such separator token was found. */ @Nullable public static String getLastToken (@Nullable final String sStr, @Nullable final String sSearch) { if (StringHelper.hasNoText (sSearch)) return sStr; final int nIndex = getLastIndexOf (sStr, sSearch); return nIndex == StringHelper.STRING_NOT_FOUND ? sStr : sStr.substring (nIndex + getLength (sSearch)); } @Nullable public static String getReverse (@Nullable final String sStr) { if (sStr == null) return null; final char [] aChars = sStr.toCharArray (); if (aChars.length <= 1) return sStr; final char [] ret = new char [aChars.length]; int nSrc = aChars.length - 1; int nDst = 0; while (nSrc >= 0) { ret[nDst] = aChars[nSrc]; nSrc--; nDst++; } return new String (ret); } /** * Optimized remove method that removes a set of characters from an input * string! * * @param sInputString * The input string. * @param aRemoveChars * The characters to remove. May not be null. * @return The version of the string without the passed characters or an empty * String if the input string was null. */ @Nonnull public static String removeMultiple (@Nullable final String sInputString, @Nonnull final char [] aRemoveChars) { ValueEnforcer.notNull (aRemoveChars, "RemoveChars"); // Any input text? if (hasNoText (sInputString)) return ""; // Anything to remove? if (aRemoveChars.length == 0) return sInputString; final StringBuilder aSB = new StringBuilder (sInputString.length ()); iterateChars (sInputString, cInput -> { if (!ArrayHelper.contains (aRemoveChars, cInput)) aSB.append (cInput); }); return aSB.toString (); } /** * Iterate all characters and pass them to the provided consumer. * * @param sInputString * Input String to use. May be null or empty. * @param aConsumer * The consumer to be used. May not be null. */ public static void iterateChars (@Nullable final String sInputString, @Nonnull final ICharConsumer aConsumer) { ValueEnforcer.notNull (aConsumer, "Consumer"); if (sInputString != null) { final char [] aInput = sInputString.toCharArray (); for (final char cInput : aInput) aConsumer.accept (cInput); } } /** * Iterate all code points and pass them to the provided consumer. This * implementation is approximately 20% quicker than * CharSequence.codePoints().forEachOrdered(c) * * @param sInputString * Input String to use. May be null or empty. * @param aConsumer * The consumer to be used. May not be null. */ public static void iterateCodePoints (@Nullable final String sInputString, @Nonnull final IntConsumer aConsumer) { ValueEnforcer.notNull (aConsumer, "Consumer"); if (sInputString != null) { final int nStringLength = sInputString.length (); int nOfs = 0; while (nOfs < nStringLength) { final int nCodePoint = sInputString.codePointAt (nOfs); nOfs += Character.charCount (nCodePoint); aConsumer.accept (nCodePoint); } } } /** * Encode a char array to a byte array using the provided charset. This does * the same as new String (aCharArray).getBytes (aCharset) just * without the intermediate objects. * * @param aCharset * Charset to be used. May not be null. * @param aCharArray * The char array to be encoded. May not be null. * @return The created byte array. Never null. * @since 8.6.4 */ @Nonnull @ReturnsMutableCopy public static byte [] encodeCharToBytes (@Nonnull final char [] aCharArray, @Nonnull final Charset aCharset) { return encodeCharToBytes (aCharArray, 0, aCharArray.length, aCharset); } /** * Encode a char array to a byte array using the provided charset. This does * the same as new String (aCharArray).getBytes (aCharset) just * without the intermediate objects. * * @param aCharset * Charset to be used. May not be null. * @param aCharArray * The char array to be encoded. May not be null. * @param nOfs * Offset into char array. Must be ≥ 0. * @param nLen * Chars to encode. Must be ≥ 0. * @return The created byte array. Never null. * @since 8.6.4 */ @Nonnull @ReturnsMutableCopy public static byte [] encodeCharToBytes (@Nonnull final char [] aCharArray, @Nonnegative final int nOfs, @Nonnegative final int nLen, @Nonnull final Charset aCharset) { ValueEnforcer.isArrayOfsLen (aCharArray, nOfs, nLen); final CharsetEncoder aEncoder = aCharset.newEncoder (); // We need to perform double, not float, arithmetic; otherwise // we lose low order bits when nLen is larger than 2^24. final int nEncodedLen = (int) (nLen * (double) aEncoder.maxBytesPerChar ()); final byte [] aByteArray = new byte [nEncodedLen]; if (nLen == 0) return aByteArray; aEncoder.onMalformedInput (CodingErrorAction.REPLACE).onUnmappableCharacter (CodingErrorAction.REPLACE).reset (); final CharBuffer aSrcBuf = CharBuffer.wrap (aCharArray, nOfs, nLen); final ByteBuffer aDstBuf = ByteBuffer.wrap (aByteArray); try { CoderResult aRes = aEncoder.encode (aSrcBuf, aDstBuf, true); if (!aRes.isUnderflow ()) aRes.throwException (); aRes = aEncoder.flush (aDstBuf); if (!aRes.isUnderflow ()) aRes.throwException (); } catch (final CharacterCodingException x) { throw new IllegalStateException (x); } final int nDstLen = aDstBuf.position (); if (nDstLen == aByteArray.length) return aByteArray; return Arrays.copyOf (aByteArray, nDstLen); } /** * Decode a byte array to a char array using the provided charset. This does * the same as new String (aByteArray, aCharset) just without the * intermediate objects. * * @param aByteArray * The byte array to be decoded. May not be null. * @param aCharset * Charset to be used. May not be null. * @return The created char array. Never null. * @since 8.6.4 */ @Nonnull public static char [] decodeBytesToChars (@Nonnull final byte [] aByteArray, @Nonnull final Charset aCharset) { return decodeBytesToChars (aByteArray, 0, aByteArray.length, aCharset); } /** * Decode a byte array to a char array using the provided charset. This does * the same as new String (aByteArray, aCharset) just without the * intermediate objects. * * @param aByteArray * The byte array to be decoded. May not be null. * @param nOfs * Offset into byte array. Must be ≥ 0. * @param nLen * Bytes to encode. Must be ≥ 0. * @param aCharset * Charset to be used. May not be null. * @return The created char array. Never null. * @since 8.6.4 */ @Nonnull public static char [] decodeBytesToChars (@Nonnull final byte [] aByteArray, @Nonnegative final int nOfs, @Nonnegative final int nLen, @Nonnull final Charset aCharset) { final CharsetDecoder aDecoder = aCharset.newDecoder (); final int nDecodedLen = (int) (nLen * (double) aDecoder.maxCharsPerByte ()); final char [] aCharArray = new char [nDecodedLen]; if (nLen == 0) return aCharArray; aDecoder.onMalformedInput (CodingErrorAction.REPLACE).onUnmappableCharacter (CodingErrorAction.REPLACE).reset (); final ByteBuffer aSrcBuf = ByteBuffer.wrap (aByteArray, nOfs, nLen); final CharBuffer aDstBuf = CharBuffer.wrap (aCharArray); try { CoderResult aRes = aDecoder.decode (aSrcBuf, aDstBuf, true); if (!aRes.isUnderflow ()) aRes.throwException (); aRes = aDecoder.flush (aDstBuf); if (!aRes.isUnderflow ()) aRes.throwException (); } catch (final CharacterCodingException x) { // Substitution is always enabled, // so this shouldn't happen throw new IllegalStateException (x); } final int nDstLen = aDstBuf.position (); if (nDstLen == aCharArray.length) return aCharArray; return Arrays.copyOf (aCharArray, nDstLen); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy