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 super String[], T, E> 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 super Map, T, E> converter) throws E {
return converter.apply(this.split(source));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy