AppendableJoiner.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */

  17. package org.apache.commons.lang3;

  18. import java.io.IOException;
  19. import java.util.Iterator;
  20. import java.util.StringJoiner;
  21. import java.util.function.Supplier;

  22. import org.apache.commons.lang3.exception.UncheckedException;
  23. import org.apache.commons.lang3.function.FailableBiConsumer;

  24. /**
  25.  * Joins an array or {@link Iterable} into an existing {@link Appendable} like a {@link StringBuilder}; with the goal for call sites to avoid creating
  26.  * intermediary Strings. This is like {@link String#join(CharSequence, CharSequence...)}, {@link String#join(CharSequence, Iterable)}, and {@link StringJoiner}.
  27.  * <p>
  28.  * Keep an instance in a (static) variable for efficient joining into an {@link Appendable} or {@link StringBuilder} without creating temporary Strings.
  29.  * </p>
  30.  * <p>
  31.  * Use the builder and instance methods to reuse the same kind of joining prefix, suffix, delimiter, and string conversion.
  32.  * </p>
  33.  * <p>
  34.  * For example:
  35.  * </p>
  36.  *
  37.  * <pre>{@code
  38.  * // A reuseable instance
  39.  * private static final AppendableJoiner<Object> JOINER = AppendableJoiner.builder()
  40.  *     .setPrefix("[")
  41.  *     .setSuffix("]")
  42.  *     .setDelimiter(", ")
  43.  *     .get();
  44.  * }
  45.  * ...
  46.  * // Builds straight into a StringBuilder:
  47.  * StringBuilder sbuilder = new StringBuilder("1");
  48.  * JOINER.join(sbuilder, "A", "B");
  49.  * sbuilder.append("2");
  50.  * JOINER.join(sbuilder, "C", "D");
  51.  * sbuilder.append("3");
  52.  * // Returns "1[A, B]2[C, D]3"
  53.  * return sbuilder.toString();
  54.  * }</pre>
  55.  * <p>
  56.  * To provide a custom Object element to {@link CharSequence} converter, call {@link Builder#setElementAppender(FailableBiConsumer)}, for example:
  57.  * </p>
  58.  *
  59.  * <pre>{@code
  60.  * private static final AppendableJoiner<Item> JOINER = AppendableJoiner.builder()
  61.  *     .setElementAppender(e -> (a, e) -> a.append(e.getFoo())
  62.  *                                        a.append(e.getBar())
  63.  *                                        a.append('!'))
  64.  *     ...
  65.  *     .get();
  66.  * }
  67.  * }</pre>
  68.  * <p>
  69.  * This class is immutable and thread-safe.
  70.  * </p>
  71.  *
  72.  * @param <T> the type of elements to join.
  73.  * @see Appendable
  74.  * @see StringBuilder
  75.  * @see String#join(CharSequence, CharSequence...)
  76.  * @see String#join(CharSequence, Iterable)
  77.  * @see StringJoiner
  78.  * @since 3.15.0
  79.  */
  80. public final class AppendableJoiner<T> {

  81.     /**
  82.      * Builds instances of {@link AppendableJoiner}.
  83.      *
  84.      * @param <T> the type of elements to join.
  85.      */
  86.     public static final class Builder<T> implements Supplier<AppendableJoiner<T>> {

  87.         /** The sequence of characters to be used at the beginning. */
  88.         private CharSequence prefix;

  89.         /** The sequence of characters to be used at the end. */
  90.         private CharSequence suffix;

  91.         /** The delimiter that separates each element. */
  92.         private CharSequence delimiter;

  93.         /** The consumer used to render each element of type {@code T} onto an {@link Appendable}. */
  94.         private FailableBiConsumer<Appendable, T, IOException> appender;

  95.         /**
  96.          * Constructs a new instance.
  97.          */
  98.         Builder() {
  99.             // empty
  100.         }

  101.         /**
  102.          * Gets a new instance of {@link AppendableJoiner}.
  103.          */
  104.         @Override
  105.         public AppendableJoiner<T> get() {
  106.             return new AppendableJoiner<>(prefix, suffix, delimiter, appender);
  107.         }

  108.         /**
  109.          * Sets the delimiter that separates each element.
  110.          *
  111.          * @param delimiter The delimiter that separates each element.
  112.          * @return this instance.
  113.          */
  114.         public Builder<T> setDelimiter(final CharSequence delimiter) {
  115.             this.delimiter = delimiter;
  116.             return this;
  117.         }

  118.         /**
  119.          * Sets the consumer used to render each element of type {@code T} onto an {@link Appendable}.
  120.          *
  121.          * @param appender The consumer used to render each element of type {@code T} onto an {@link Appendable}.
  122.          * @return this instance.
  123.          */
  124.         public Builder<T> setElementAppender(final FailableBiConsumer<Appendable, T, IOException> appender) {
  125.             this.appender = appender;
  126.             return this;
  127.         }

  128.         /**
  129.          * Sets the sequence of characters to be used at the beginning.
  130.          *
  131.          * @param prefix The sequence of characters to be used at the beginning.
  132.          * @return this instance.
  133.          */
  134.         public Builder<T> setPrefix(final CharSequence prefix) {
  135.             this.prefix = prefix;
  136.             return this;
  137.         }

  138.         /**
  139.          * Sets the sequence of characters to be used at the end.
  140.          *
  141.          * @param suffix The sequence of characters to be used at the end.
  142.          * @return this instance.
  143.          */
  144.         public Builder<T> setSuffix(final CharSequence suffix) {
  145.             this.suffix = suffix;
  146.             return this;
  147.         }

  148.     }

  149.     /**
  150.      * Creates a new builder.
  151.      *
  152.      * @param <T> The type of elements.
  153.      * @return a new builder.
  154.      */
  155.     public static <T> Builder<T> builder() {
  156.         return new Builder<>();
  157.     }

  158.     /** Could be public in the future, in some form. */
  159.     @SafeVarargs
  160.     static <A extends Appendable, T> A joinA(final A appendable, final CharSequence prefix, final CharSequence suffix, final CharSequence delimiter,
  161.             final FailableBiConsumer<Appendable, T, IOException> appender, final T... elements) throws IOException {
  162.         return joinArray(appendable, prefix, suffix, delimiter, appender, elements);
  163.     }

  164.     private static <A extends Appendable, T> A joinArray(final A appendable, final CharSequence prefix, final CharSequence suffix, final CharSequence delimiter,
  165.             final FailableBiConsumer<Appendable, T, IOException> appender, final T[] elements) throws IOException {
  166.         appendable.append(prefix);
  167.         if (elements != null) {
  168.             if (elements.length > 0) {
  169.                 appender.accept(appendable, elements[0]);
  170.             }
  171.             for (int i = 1; i < elements.length; i++) {
  172.                 appendable.append(delimiter);
  173.                 appender.accept(appendable, elements[i]);
  174.             }
  175.         }
  176.         appendable.append(suffix);
  177.         return appendable;
  178.     }

  179.     /** Could be public in the future, in some form. */
  180.     static <T> StringBuilder joinI(final StringBuilder stringBuilder, final CharSequence prefix, final CharSequence suffix, final CharSequence delimiter,
  181.             final FailableBiConsumer<Appendable, T, IOException> appender, final Iterable<T> elements) {
  182.         try {
  183.             return joinIterable(stringBuilder, prefix, suffix, delimiter, appender, elements);
  184.         } catch (final IOException e) {
  185.             // Cannot happen with a StringBuilder.
  186.             throw new UncheckedException(e);
  187.         }
  188.     }

  189.     private static <A extends Appendable, T> A joinIterable(final A appendable, final CharSequence prefix, final CharSequence suffix,
  190.             final CharSequence delimiter, final FailableBiConsumer<Appendable, T, IOException> appender, final Iterable<T> elements) throws IOException {
  191.         appendable.append(prefix);
  192.         if (elements != null) {
  193.             final Iterator<T> iterator = elements.iterator();
  194.             if (iterator.hasNext()) {
  195.                 appender.accept(appendable, iterator.next());
  196.             }
  197.             while (iterator.hasNext()) {
  198.                 appendable.append(delimiter);
  199.                 appender.accept(appendable, iterator.next());
  200.             }
  201.         }
  202.         appendable.append(suffix);
  203.         return appendable;
  204.     }

  205.     /** Could be public in the future, in some form. */
  206.     @SafeVarargs
  207.     static <T> StringBuilder joinSB(final StringBuilder stringBuilder, final CharSequence prefix, final CharSequence suffix, final CharSequence delimiter,
  208.             final FailableBiConsumer<Appendable, T, IOException> appender, final T... elements) {
  209.         try {
  210.             return joinArray(stringBuilder, prefix, suffix, delimiter, appender, elements);
  211.         } catch (final IOException e) {
  212.             // Cannot happen with a StringBuilder.
  213.             throw new UncheckedException(e);
  214.         }
  215.     }

  216.     private static CharSequence nonNull(final CharSequence value) {
  217.         return value != null ? value : StringUtils.EMPTY;
  218.     }

  219.     /** The sequence of characters to be used at the beginning. */
  220.     private final CharSequence prefix;

  221.     /** The sequence of characters to be used at the end. */
  222.     private final CharSequence suffix;

  223.     /** The delimiter that separates each element. */
  224.     private final CharSequence delimiter;

  225.     private final FailableBiConsumer<Appendable, T, IOException> appender;

  226.     /**
  227.      * Constructs a new instance.
  228.      */
  229.     private AppendableJoiner(final CharSequence prefix, final CharSequence suffix, final CharSequence delimiter,
  230.             final FailableBiConsumer<Appendable, T, IOException> appender) {
  231.         this.prefix = nonNull(prefix);
  232.         this.suffix = nonNull(suffix);
  233.         this.delimiter = nonNull(delimiter);
  234.         this.appender = appender != null ? appender : (a, e) -> a.append(String.valueOf(e));
  235.     }

  236.     /**
  237.      * Joins stringified objects from the given Iterable into a StringBuilder.
  238.      *
  239.      * @param stringBuilder The target.
  240.      * @param elements      The source.
  241.      * @return The given StringBuilder.
  242.      */
  243.     public StringBuilder join(final StringBuilder stringBuilder, final Iterable<T> elements) {
  244.         return joinI(stringBuilder, prefix, suffix, delimiter, appender, elements);
  245.     }

  246.     /**
  247.      * Joins stringified objects from the given array into a StringBuilder.
  248.      *
  249.      * @param stringBuilder The target.
  250.      * @param elements      The source.
  251.      * @return the given target StringBuilder.
  252.      */
  253.     public StringBuilder join(final StringBuilder stringBuilder, @SuppressWarnings("unchecked") final T... elements) {
  254.         return joinSB(stringBuilder, prefix, suffix, delimiter, appender, elements);
  255.     }

  256.     /**
  257.      * Joins stringified objects from the given Iterable into an Appendable.
  258.      *
  259.      * @param <A>        the Appendable type.
  260.      * @param appendable The target.
  261.      * @param elements   The source.
  262.      * @return The given StringBuilder.
  263.      * @throws IOException If an I/O error occurs
  264.      */
  265.     public <A extends Appendable> A joinA(final A appendable, final Iterable<T> elements) throws IOException {
  266.         return joinIterable(appendable, prefix, suffix, delimiter, appender, elements);
  267.     }

  268.     /**
  269.      * Joins stringified objects from the given array into an Appendable.
  270.      *
  271.      * @param <A>        the Appendable type.
  272.      * @param appendable The target.
  273.      * @param elements   The source.
  274.      * @return The given StringBuilder.
  275.      * @throws IOException If an I/O error occurs
  276.      */
  277.     public <A extends Appendable> A joinA(final A appendable, @SuppressWarnings("unchecked") final T... elements) throws IOException {
  278.         return joinA(appendable, prefix, suffix, delimiter, appender, elements);
  279.     }

  280. }