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

serp.util.Strings Maven / Gradle / Ivy

There is a newer version: 4.0.1
Show newest version
package serp.util;

import java.math.*;
import java.util.*;

/**
 * String utiltity methods.
 *
 * @author Abe White
 */
public class Strings {
    private static final Object[][] _codes = new Object[][] {
        { byte.class, "byte", "B" },
        { char.class, "char", "C" },
        { double.class, "double", "D" },
        { float.class, "float", "F" },
        { int.class, "int", "I" },
        { long.class, "long", "J" },
        { short.class, "short", "S" },
        { boolean.class, "boolean", "Z" },
        { void.class, "void", "V" }
    };

    /**
     * Replace all instances of from in str
     * with to.
     *
     * @param str the candidate string to replace
     * @param from the token to replace
     * @param to the new token
     * @return the string with all the replacements made
     */
    public static String replace(String str, String from, String to) {
        String[] split = split(str, from, Integer.MAX_VALUE);
        return join(split, to);
    }

    /**
     * Splits the given string on the given token. Follows the semantics
     * of the Java 1.4 {@link String#split(String,int)} method, but does
     * not treat the given token as a regular expression.
     */
    public static String[] split(String str, String token, int max) {
        if (str == null || str.length() == 0)
            return new String[0];
        if (token == null || token.length() == 0)
            throw new IllegalArgumentException("token: [" + token + "]");

        // split on token 
        LinkedList ret = new LinkedList();
        int start = 0;
        for (int split = 0; split != -1;) {
            split = str.indexOf(token, start);
            if (split == -1 && start >= str.length())
                ret.add("");
            else if (split == -1)
                ret.add(str.substring(start));
            else {
                ret.add(str.substring(start, split));
                start = split + token.length();
            }
        }

        // now take max into account; this isn't the most efficient way
        // of doing things since we split the maximum number of times
        // regardless of the given parameters, but it makes things easy
        if (max == 0) {
            // discard any trailing empty splits
            while (ret.getLast().equals(""))
                ret.removeLast();
        } else if (max > 0 && ret.size() > max) {
            // move all splits over max into the last split
            StringBuilder buf = new StringBuilder(ret.removeLast().toString());
            while (ret.size() >= max) {
                buf.insert(0, token);
                buf.insert(0, ret.removeLast());
            }
            ret.add(buf.toString());
        }
        return (String[]) ret.toArray(new String[ret.size()]);
    }

    /**
     * Joins the given strings, placing the given token between them.
     */
    public static String join(Object[] strings, String token) {
        if (strings == null)
            return null;

        StringBuilder buf = new StringBuilder(20 * strings.length);
        for (int i = 0; i < strings.length; i++) {
            if (i > 0)
                buf.append(token);
            if (strings[i] != null)
                buf.append(strings[i]);
        }
        return buf.toString();
    }

    /**
     * Return the class for the given string, correctly handling
     * primitive types. If the given class loader is null, the context
     * loader of the current thread will be used.
     *
     * @throws RuntimeException on load error
     */
    public static Class toClass(String str, ClassLoader loader) {
        return toClass(str, false, loader);
    }

    /**
     * Return the class for the given string, correctly handling
     * primitive types. If the given class loader is null, the context
     * loader of the current thread will be used.
     *
     * @throws RuntimeException on load error
     */
    public static Class toClass(String str, boolean resolve, 
        ClassLoader loader) {
        if (str == null)
            throw new NullPointerException("str == null");

        // array handling
        int dims = 0;
        while (str.endsWith("[]")) {
            dims++;
            str = str.substring(0, str.length() - 2);
        }

        // check against primitive types
        boolean primitive = false;
        if (str.indexOf('.') == -1) {
            for (int i = 0; !primitive && (i < _codes.length); i++) {
                if (_codes[i][1].equals(str)) {
                    if (dims == 0)
                        return (Class) _codes[i][0];
                    str = (String) _codes[i][2];
                    primitive = true;
                }
            }
        }

        if (dims > 0) {
            int size = str.length() + dims;
            if (!primitive)
                size += 2;

            StringBuilder buf = new StringBuilder(size);
            for (int i = 0; i < dims; i++)
                buf.append('[');
            if (!primitive)
                buf.append('L');
            buf.append(str);
            if (!primitive)
                buf.append(';');
            str = buf.toString();
        }

        if (loader == null)
            loader = Thread.currentThread().getContextClassLoader();
        try {
            return Class.forName(str, resolve, loader);
        } catch (Throwable t) {
            throw new IllegalArgumentException(t.toString());
        }
    }

    /**
     * Return only the class name, without package.
     */
    public static String getClassName(Class cls) {
        return (cls == null) ? null : getClassName(cls.getName());
    }

    /**
     * Return only the class name.
     */
    public static String getClassName(String fullName) {
        if (fullName == null)
            return null;

        // special case for arrays
        int dims = 0;
        for (int i = 0; i < fullName.length(); i++) {
            if (fullName.charAt(i) != '[') {
                dims = i;
                break;
            }
        }
        if (dims > 0)
            fullName = fullName.substring(dims);

        // check for primitives
        for (int i = 0; i < _codes.length; i++) {
            if (_codes[i][2].equals(fullName)) {
                fullName = (String) _codes[i][1];
                break;
            }
        }

        fullName = fullName.substring(fullName.lastIndexOf('.') + 1);
        for (int i = 0; i < dims; i++)
            fullName = fullName + "[]";
        return fullName;
    }

    /**
     * Return only the package, or empty string if none.
     */
    public static String getPackageName(Class cls) {
        return (cls == null) ? null : getPackageName(cls.getName());
    }

    /**
     * Return only the package, or empty string if none.
     */
    public static String getPackageName(String fullName) {
        if (fullName == null)
            return null;
        int dotIdx = fullName.lastIndexOf('.');
        return (dotIdx == -1) ? "" : fullName.substring(0, dotIdx);
    }

    /**
     * Return val as the type specified by
     * type. If type is a primitive, the
     * primitive wrapper type is created and returned, and
     * nulls are converted to the Java default for the
     * primitive type.
     *
     * @param val The string value to parse
     * @param type The type to parse. This must be a primitive or a
     * primitive wrapper, or one of {@link BigDecimal},
     * {@link BigInteger}, {@link String}, {@link Date}.
     * @throws IllegalArgumentException if type is not a
     * supported type, or if val cannot be
     * converted into an instance of type type.
     */
    public static Object parse(String val, Class type) {
        if (!canParse(type))
            throw new IllegalArgumentException("invalid type: " +
                type.getName());

        // deal with null value
        if (val == null) {
            if (!type.isPrimitive())
                return null;
            if (type == boolean.class)
                return Boolean.FALSE;
            if (type == byte.class)
                return new Byte((byte) 0);
            if (type == char.class)
                return new Character((char) 0);
            if (type == double.class)
                return new Double(0);
            if (type == float.class)
                return new Float(0);
            if (type == int.class)
                return Numbers.valueOf(0);
            if (type == long.class)
                return Numbers.valueOf(0L);
            if (type == short.class)
                return new Short((short) 0);
            throw new IllegalStateException("invalid type: " + type);
        }

        // deal with non-null value
        if (type == boolean.class || type == Boolean.class)
            return Boolean.valueOf(val);
        if (type == byte.class || type == Byte.class)
            return Byte.valueOf(val);
        if (type == char.class || type == Character.class) {
            if (val.length() == 0)
                return new Character((char) 0);
            if (val.length() == 1)
                return new Character(val.charAt(0));
            throw new IllegalArgumentException("'" + val + "' is longer than " 
                + "one character.");
        }
        if (type == double.class || type == Double.class)
            return Double.valueOf(val);
        if (type == float.class || type == Float.class)
            return Float.valueOf(val);
        if (type == int.class || type == Integer.class)
            return Integer.valueOf(val);
        if (type == long.class || type == Long.class)
            return Long.valueOf(val);
        if (type == short.class || type == Short.class)
            return Short.valueOf(val);
        if (type == String.class)
            return val;
        if (type == Date.class)
            return new Date(val);
        if (type == BigInteger.class)
            return new BigInteger(val);
        if (type == BigDecimal.class)
            return new BigDecimal(val);
        throw new IllegalArgumentException("Invalid type: " + type);
    }

    /**
     * Whether the given type is parsable via {@link #parse}.
     */
    public static boolean canParse(Class type) {
        return type.isPrimitive() || type == Boolean.class 
            || type == Byte.class || type == Character.class 
            || type == Short.class || type == Integer.class 
            || type == Long.class || type == Float.class 
            || type == Double.class || type == String.class 
            || type == Date.class || type == BigInteger.class 
            || type == BigDecimal.class;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy