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

com.landawn.abacus.util.Splitter Maven / Gradle / Ivy

/*
 * Copyright (C) 2016 HaiYang Li
 *
 * 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.landawn.abacus.util;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;

import com.landawn.abacus.type.Type;
import com.landawn.abacus.util.function.Supplier;

/**
 * 
 * @since 0.8
 * 
 * @author Haiyang Li
 */
public final class Splitter {
    private static final Splitter DEFAULT = new Splitter(Joiner.DEFAULT_DELIMITER, null);
    private final String delimiter;
    private final String delimiterRegex;
    private int max = Integer.MAX_VALUE;
    private boolean trim = false;

    Splitter(String delimiter, String delimiterRegex) {
        this.delimiter = delimiter;
        this.delimiterRegex = delimiterRegex;
    }

    /**
     * Returns the Splitter with the default delimiter: ", "
     * 
     * @return
     */
    public static Splitter defauLt() {
        return DEFAULT;
    }

    public static Splitter with(CharSequence delimiter) {
        if (N.isNullOrEmpty(delimiter)) {
            throw new IllegalArgumentException("'delimiter' can't be null or empty");
        }

        return new Splitter(delimiter.toString(), null);
    }

    public static Splitter pattern(CharSequence delimiterRegex) {
        if (N.isNullOrEmpty(delimiterRegex)) {
            throw new IllegalArgumentException("'delimiterRegex' can't be null or empty");
        }

        return new Splitter(null, delimiterRegex.toString());
    }

    public Splitter limit(int max) {
        if (max < 1) {
            throw new IllegalArgumentException("'max' must be greater than 0");
        }

        this.max = max;

        return this;
    }

    public Splitter trim(boolean trim) {
        this.trim = trim;

        return this;
    }

    public List split(final CharSequence source) {
        if (N.isNullOrEmpty(source)) {
            return new ArrayList<>();
        }

        return N.asList(split(source, delimiter, delimiterRegex, max, trim));
    }

    public  List split(Class targetType, final CharSequence source) {
        final Type type = N.typeOf(targetType);

        return split(type, source);
    }

    public  List split(Type type, final CharSequence source) {
        if (N.isNullOrEmpty(source)) {
            return new ArrayList<>();
        }

        final String[] strs = split(source, delimiter, delimiterRegex, max, trim);
        final List result = new ArrayList<>(strs.length);

        for (String str : strs) {
            result.add(type.valueOf(str));
        }

        return result;
    }

    public > C split(final C output, final CharSequence source) {
        final C result = output;

        if (N.isNullOrEmpty(source)) {
            return result;
        }

        final String[] strs = split(source, delimiter, delimiterRegex, max, trim);
        result.addAll(Arrays.asList(strs));

        return result;
    }

    public > C split(final C output, Class targetType, final CharSequence source) {
        final Type type = N.typeOf(targetType);

        return split(output, type, source);
    }

    public > C split(final C output, Type type, final CharSequence source) {
        final C result = output;

        if (N.isNullOrEmpty(source)) {
            return result;
        }

        final String[] strs = split(source, delimiter, delimiterRegex, max, trim);

        for (String str : strs) {
            result.add(type.valueOf(str));
        }

        return result;
    }

    public > C split(final CharSequence source, final Supplier supplier) {
        return this.split(supplier.get(), source);
    }

    public > C split(Class targetType, final CharSequence source, final Supplier supplier) {
        return this.split(supplier.get(), targetType, source);
    }

    public > C split(Type type, final CharSequence source, final Supplier supplier) {
        return this.split(supplier.get(), type, source);
    }

    public String[] splitToArray(final CharSequence source) {
        if (N.isNullOrEmpty(source)) {
            return N.EMPTY_STRING_ARRAY;
        }

        return Splitter.split(source, delimiter, delimiterRegex, max, trim);
    }

    public  T splitToArray(Class arrayType, final CharSequence source) {
        final Class eleCls = arrayType.getComponentType();

        if (N.isNullOrEmpty(source)) {
            return N.newArray(eleCls, 0);
        }

        final String[] strs = Splitter.split(source, delimiter, delimiterRegex, max, trim);

        if (eleCls.equals(String.class) || eleCls.equals(Object.class)) {
            return (T) strs;
        } else {
            final Type eleType = N.typeOf(eleCls);
            final Object a = N.newArray(eleCls, strs.length);

            if (Primitives.isPrimitiveType(eleCls)) {
                for (int i = 0, len = strs.length; i < len; i++) {
                    Array.set(a, i, eleType.valueOf(strs[i]));
                }
            } else {
                final Object[] objArray = (Object[]) a;

                for (int i = 0, len = strs.length; i < len; i++) {
                    objArray[i] = eleType.valueOf(strs[i]);
                }
            }

            return (T) a;
        }
    }

    //    public > C split(final Supplier supplier, final CharSequence source) {
    //        return this.split(supplier.get(), source);
    //    }
    //
    //    public > C split(final Supplier supplier, Class targetType, final CharSequence source) {
    //        return this.split(supplier.get(), targetType, source);
    //    }
    //
    //    public > C split(final Supplier supplier, Type type, final CharSequence source) {
    //        return this.split(supplier.get(), type, source);
    //    }
    //
    //    public > C split(final Supplier supplier, String typeName, final CharSequence source) {
    //        return this.split(supplier.get(), typeName, source);
    //    }

    public  T splitAndThen(final CharSequence source, Try.Function converter) throws E {
        return converter.apply(this.splitToArray(source));
    }

    static String[] split(final CharSequence source, String delimiter, String delimiterRegex, int max, boolean trim) {
        final String sourceStr = source.toString();
        String[] strs = null;

        if (N.notNullOrEmpty(delimiter)) {
            strs = StringUtil.split(sourceStr, delimiter, max, trim);
        } else {
            strs = sourceStr.split(delimiterRegex, max);

            if (trim) {
                for (int i = 0, len = strs.length; i < len; i++) {
                    strs[i] = strs[i].trim();
                }
            }
        }

        return strs;
    }

    public static final class MapSplitter {
        private static final MapSplitter DEFAULT = new MapSplitter(Joiner.DEFAULT_DELIMITER, Joiner.DEFAULT_KEY_VALUE_DELIMITER, null, null);

        private final String entryDelimiter;
        private final String keyValueDelimiter;
        private final String entryDelimiterRegex;
        private final String keyValueDelimiterRegex;
        private int max = Integer.MAX_VALUE;
        private boolean trim = false;

        MapSplitter(final String entryDelimiter, final String keyValueDelimiter, final String entryDelimiterRegex, final String keyValueDelimiterRegex) {
            this.entryDelimiter = entryDelimiter;
            this.keyValueDelimiter = keyValueDelimiter;
            this.entryDelimiterRegex = entryDelimiterRegex;
            this.keyValueDelimiterRegex = keyValueDelimiterRegex;
        }

        /**
         * Returns the Map Splitter with the default entry and key/value delimiter: ", " and "="
         * 
         * @return
         */
        public static MapSplitter defauLt() {
            return DEFAULT;
        }

        public static MapSplitter with(CharSequence entryDelimiter, CharSequence keyValueDelimiter) {
            if (N.isNullOrEmpty(entryDelimiter) || N.isNullOrEmpty(keyValueDelimiter)) {
                throw new IllegalArgumentException("'entryDelimiter' and 'keyValueDelimiter' can't be null or empty");
            }

            return new MapSplitter(entryDelimiter.toString(), keyValueDelimiter.toString(), null, null);
        }

        public static MapSplitter pattern(CharSequence entryDelimiterRegex, CharSequence keyValueDelimiterRegex) {
            if (N.isNullOrEmpty(entryDelimiterRegex) || N.isNullOrEmpty(keyValueDelimiterRegex)) {
                throw new IllegalArgumentException("'entryDelimiterRegex' and 'keyValueDelimiterRegex' can't be null or empty");
            }

            return new MapSplitter(null, null, entryDelimiterRegex.toString(), keyValueDelimiterRegex.toString());
        }

        public MapSplitter limit(int max) {
            if (max < 1) {
                throw new IllegalArgumentException("'max' must be greater than 0");
            }

            this.max = max;

            return this;
        }

        public MapSplitter trim(boolean trim) {
            this.trim = trim;

            return this;
        }

        public Map split(final CharSequence source) {
            if (N.isNullOrEmpty(source)) {
                return new LinkedHashMap<>();
            }

            final String[] strs = Splitter.split(source, entryDelimiter, entryDelimiterRegex, max, trim);
            final Map result = new LinkedHashMap<>(N.initHashCapacity(strs.length));
            String[] strEntry = null;

            if (N.notNullOrEmpty(keyValueDelimiter)) {
                for (String str : strs) {
                    strEntry = StringUtil.split(str, keyValueDelimiter, 2, trim);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    result.put(strEntry[0], strEntry[1]);
                }
            } else {
                for (String str : strs) {
                    strEntry = str.split(keyValueDelimiterRegex, 2);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    if (trim) {
                        strEntry[0] = strEntry[0].trim();
                        strEntry[1] = strEntry[1].trim();
                    }

                    result.put(strEntry[0], strEntry[1]);
                }
            }

            return result;
        }

        public  Map split(Class keyType, Class valueType, final CharSequence source) {
            final Type typeOfKey = N.typeOf(keyType);
            final Type typeOfValue = N.typeOf(valueType);

            return split(typeOfKey, typeOfValue, source);
        }

        public  Map split(Type keyType, Type valueType, final CharSequence source) {
            if (N.isNullOrEmpty(source)) {
                return new LinkedHashMap<>();
            }

            final String[] strs = Splitter.split(source, entryDelimiter, entryDelimiterRegex, max, trim);
            final Map result = new LinkedHashMap<>(N.initHashCapacity(strs.length));
            String[] strEntry = null;

            if (N.notNullOrEmpty(keyValueDelimiter)) {
                for (String str : strs) {
                    strEntry = StringUtil.split(str, keyValueDelimiter, 2, trim);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    result.put(keyType.valueOf(strEntry[0]), valueType.valueOf(strEntry[1]));
                }
            } else {
                for (String str : strs) {
                    strEntry = str.split(keyValueDelimiterRegex, 2);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    if (trim) {
                        strEntry[0] = strEntry[0].trim();
                        strEntry[1] = strEntry[1].trim();
                    }

                    result.put(keyType.valueOf(strEntry[0]), valueType.valueOf(strEntry[1]));
                }
            }

            return result;
        }

        public > M split(final M output, final CharSequence source) {
            final M result = output;

            if (N.isNullOrEmpty(source)) {
                return result;
            }

            final String[] strs = Splitter.split(source, entryDelimiter, entryDelimiterRegex, max, trim);
            String[] strEntry = null;

            if (N.notNullOrEmpty(keyValueDelimiter)) {
                for (String str : strs) {
                    strEntry = StringUtil.split(str, keyValueDelimiter, 2, trim);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    result.put(strEntry[0], strEntry[1]);
                }
            } else {
                for (String str : strs) {
                    strEntry = str.split(keyValueDelimiterRegex, 2);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    if (trim) {
                        strEntry[0] = strEntry[0].trim();
                        strEntry[1] = strEntry[1].trim();
                    }

                    result.put(strEntry[0], strEntry[1]);
                }
            }

            return result;
        }

        public > M split(final M output, Class keyType, Class valueType, final CharSequence source) {
            final Type typeOfKey = N.typeOf(keyType);
            final Type typeOfValue = N.typeOf(valueType);

            return split(output, typeOfKey, typeOfValue, source);
        }

        public > M split(final M output, Type keyType, Type valueType, final CharSequence source) {
            final M result = output;

            if (N.isNullOrEmpty(source)) {
                return result;
            }

            final String[] strs = Splitter.split(source, entryDelimiter, entryDelimiterRegex, max, trim);
            String[] strEntry = null;

            if (N.notNullOrEmpty(keyValueDelimiter)) {
                for (String str : strs) {
                    strEntry = StringUtil.split(str, keyValueDelimiter, 2, trim);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    result.put(keyType.valueOf(strEntry[0]), valueType.valueOf(strEntry[1]));
                }
            } else {
                for (String str : strs) {
                    strEntry = str.split(keyValueDelimiterRegex, 2);

                    if (strEntry.length != 2) {
                        throw new IllegalArgumentException("Invalid map entry String: " + N.toString(strEntry));
                    }

                    if (trim) {
                        strEntry[0] = strEntry[0].trim();
                        strEntry[1] = strEntry[1].trim();
                    }

                    result.put(keyType.valueOf(strEntry[0]), valueType.valueOf(strEntry[1]));
                }
            }

            return result;
        }

        public > M split(final CharSequence source, final Supplier supplier) {
            return this.split(supplier.get(), source);
        }

        public > M split(final Class keyType, final Class valueType, final CharSequence source, final Supplier supplier) {
            return this.split(supplier.get(), keyType, valueType, source);
        }

        public > M split(final Type keyType, final Type valueType, final CharSequence source, final Supplier supplier) {
            return this.split(supplier.get(), keyType, valueType, source);
        }

        public  T splitAndThen(final CharSequence source, Try.Function, T, E> converter) throws E {
            return converter.apply(this.split(source));
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy