ToStringStyle.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.builder;

  18. import java.io.Serializable;
  19. import java.lang.reflect.Array;
  20. import java.util.Collection;
  21. import java.util.Map;
  22. import java.util.Map.Entry;
  23. import java.util.Objects;
  24. import java.util.WeakHashMap;

  25. import org.apache.commons.lang3.ClassUtils;
  26. import org.apache.commons.lang3.ObjectUtils;
  27. import org.apache.commons.lang3.StringEscapeUtils;
  28. import org.apache.commons.lang3.StringUtils;

  29. /**
  30.  * Controls {@link String} formatting for {@link ToStringBuilder}.
  31.  * The main public interface is always via {@link ToStringBuilder}.
  32.  *
  33.  * <p>These classes are intended to be used as <em>singletons</em>.
  34.  * There is no need to instantiate a new style each time. A program
  35.  * will generally use one of the predefined constants on this class.
  36.  * Alternatively, the {@link StandardToStringStyle} class can be used
  37.  * to set the individual settings. Thus most styles can be achieved
  38.  * without subclassing.</p>
  39.  *
  40.  * <p>If required, a subclass can override as many or as few of the
  41.  * methods as it requires. Each object type (from {@code boolean}
  42.  * to {@code long} to {@link Object} to {@code int[]}) has
  43.  * its own methods to output it. Most have two versions, detail and summary.
  44.  *
  45.  * <p>For example, the detail version of the array based methods will
  46.  * output the whole array, whereas the summary method will just output
  47.  * the array length.</p>
  48.  *
  49.  * <p>If you want to format the output of certain objects, such as dates, you
  50.  * must create a subclass and override a method.
  51.  * </p>
  52.  * <pre>
  53.  * public class MyStyle extends ToStringStyle {
  54.  *   protected void appendDetail(StringBuffer buffer, String fieldName, Object value) {
  55.  *     if (value instanceof Date) {
  56.  *       value = new SimpleDateFormat("yyyy-MM-dd").format(value);
  57.  *     }
  58.  *     buffer.append(value);
  59.  *   }
  60.  * }
  61.  * </pre>
  62.  *
  63.  * @since 1.0
  64.  */
  65. @SuppressWarnings("deprecation") // StringEscapeUtils
  66. public abstract class ToStringStyle implements Serializable {

  67.     /**
  68.      * Default {@link ToStringStyle}.
  69.      *
  70.      * <p>This is an inner class rather than using
  71.      * {@link StandardToStringStyle} to ensure its immutability.</p>
  72.      */
  73.     private static final class DefaultToStringStyle extends ToStringStyle {

  74.         /**
  75.          * Required for serialization support.
  76.          *
  77.          * @see java.io.Serializable
  78.          */
  79.         private static final long serialVersionUID = 1L;

  80.         /**
  81.          * Constructs a new instance.
  82.          *
  83.          * <p>Use the static constant rather than instantiating.</p>
  84.          */
  85.         DefaultToStringStyle() {
  86.         }

  87.         /**
  88.          * Ensure Singleton after serialization.
  89.          *
  90.          * @return the singleton
  91.          */
  92.         private Object readResolve() {
  93.             return DEFAULT_STYLE;
  94.         }

  95.     }

  96.     /**
  97.      * {@link ToStringStyle} that outputs with JSON format.
  98.      *
  99.      * <p>
  100.      * This is an inner class rather than using
  101.      * {@link StandardToStringStyle} to ensure its immutability.
  102.      * </p>
  103.      *
  104.      * @since 3.4
  105.      * @see <a href="https://www.json.org/">json.org</a>
  106.      */
  107.     private static final class JsonToStringStyle extends ToStringStyle {

  108.         private static final long serialVersionUID = 1L;

  109.         private static final String FIELD_NAME_QUOTE = "\"";

  110.         /**
  111.          * Constructs a new instance.
  112.          *
  113.          * <p>
  114.          * Use the static constant rather than instantiating.
  115.          * </p>
  116.          */
  117.         JsonToStringStyle() {
  118.             this.setUseClassName(false);
  119.             this.setUseIdentityHashCode(false);

  120.             this.setContentStart("{");
  121.             this.setContentEnd("}");

  122.             this.setArrayStart("[");
  123.             this.setArrayEnd("]");

  124.             this.setFieldSeparator(",");
  125.             this.setFieldNameValueSeparator(":");

  126.             this.setNullText("null");

  127.             this.setSummaryObjectStartText("\"<");
  128.             this.setSummaryObjectEndText(">\"");

  129.             this.setSizeStartText("\"<size=");
  130.             this.setSizeEndText(">\"");
  131.         }

  132.         @Override
  133.         public void append(final StringBuffer buffer, final String fieldName,
  134.                            final boolean[] array, final Boolean fullDetail) {

  135.             if (fieldName == null) {
  136.                 throw new UnsupportedOperationException(
  137.                         "Field names are mandatory when using JsonToStringStyle");
  138.             }
  139.             if (!isFullDetail(fullDetail)) {
  140.                 throw new UnsupportedOperationException(
  141.                         "FullDetail must be true when using JsonToStringStyle");
  142.             }

  143.             super.append(buffer, fieldName, array, fullDetail);
  144.         }

  145.         @Override
  146.         public void append(final StringBuffer buffer, final String fieldName, final byte[] array,
  147.                            final Boolean fullDetail) {

  148.             if (fieldName == null) {
  149.                 throw new UnsupportedOperationException(
  150.                         "Field names are mandatory when using JsonToStringStyle");
  151.             }
  152.             if (!isFullDetail(fullDetail)) {
  153.                 throw new UnsupportedOperationException(
  154.                         "FullDetail must be true when using JsonToStringStyle");
  155.             }

  156.             super.append(buffer, fieldName, array, fullDetail);
  157.         }

  158.         @Override
  159.         public void append(final StringBuffer buffer, final String fieldName, final char[] array,
  160.                            final Boolean fullDetail) {

  161.             if (fieldName == null) {
  162.                 throw new UnsupportedOperationException(
  163.                         "Field names are mandatory when using JsonToStringStyle");
  164.             }
  165.             if (!isFullDetail(fullDetail)) {
  166.                 throw new UnsupportedOperationException(
  167.                         "FullDetail must be true when using JsonToStringStyle");
  168.             }

  169.             super.append(buffer, fieldName, array, fullDetail);
  170.         }

  171.         @Override
  172.         public void append(final StringBuffer buffer, final String fieldName,
  173.                            final double[] array, final Boolean fullDetail) {

  174.             if (fieldName == null) {
  175.                 throw new UnsupportedOperationException(
  176.                         "Field names are mandatory when using JsonToStringStyle");
  177.             }
  178.             if (!isFullDetail(fullDetail)) {
  179.                 throw new UnsupportedOperationException(
  180.                         "FullDetail must be true when using JsonToStringStyle");
  181.             }

  182.             super.append(buffer, fieldName, array, fullDetail);
  183.         }

  184.         @Override
  185.         public void append(final StringBuffer buffer, final String fieldName,
  186.                            final float[] array, final Boolean fullDetail) {

  187.             if (fieldName == null) {
  188.                 throw new UnsupportedOperationException(
  189.                         "Field names are mandatory when using JsonToStringStyle");
  190.             }
  191.             if (!isFullDetail(fullDetail)) {
  192.                 throw new UnsupportedOperationException(
  193.                         "FullDetail must be true when using JsonToStringStyle");
  194.             }

  195.             super.append(buffer, fieldName, array, fullDetail);
  196.         }

  197.         @Override
  198.         public void append(final StringBuffer buffer, final String fieldName, final int[] array,
  199.                            final Boolean fullDetail) {

  200.             if (fieldName == null) {
  201.                 throw new UnsupportedOperationException(
  202.                         "Field names are mandatory when using JsonToStringStyle");
  203.             }
  204.             if (!isFullDetail(fullDetail)) {
  205.                 throw new UnsupportedOperationException(
  206.                         "FullDetail must be true when using JsonToStringStyle");
  207.             }

  208.             super.append(buffer, fieldName, array, fullDetail);
  209.         }

  210.         @Override
  211.         public void append(final StringBuffer buffer, final String fieldName, final long[] array,
  212.                            final Boolean fullDetail) {

  213.             if (fieldName == null) {
  214.                 throw new UnsupportedOperationException(
  215.                         "Field names are mandatory when using JsonToStringStyle");
  216.             }
  217.             if (!isFullDetail(fullDetail)) {
  218.                 throw new UnsupportedOperationException(
  219.                         "FullDetail must be true when using JsonToStringStyle");
  220.             }

  221.             super.append(buffer, fieldName, array, fullDetail);
  222.         }

  223.         @Override
  224.         public void append(final StringBuffer buffer, final String fieldName, final Object value,
  225.                            final Boolean fullDetail) {

  226.             if (fieldName == null) {
  227.                 throw new UnsupportedOperationException(
  228.                         "Field names are mandatory when using JsonToStringStyle");
  229.             }
  230.             if (!isFullDetail(fullDetail)) {
  231.                 throw new UnsupportedOperationException(
  232.                         "FullDetail must be true when using JsonToStringStyle");
  233.             }

  234.             super.append(buffer, fieldName, value, fullDetail);
  235.         }

  236.         @Override
  237.         public void append(final StringBuffer buffer, final String fieldName,
  238.                            final Object[] array, final Boolean fullDetail) {

  239.             if (fieldName == null) {
  240.                 throw new UnsupportedOperationException(
  241.                         "Field names are mandatory when using JsonToStringStyle");
  242.             }
  243.             if (!isFullDetail(fullDetail)) {
  244.                 throw new UnsupportedOperationException(
  245.                         "FullDetail must be true when using JsonToStringStyle");
  246.             }

  247.             super.append(buffer, fieldName, array, fullDetail);
  248.         }

  249.         @Override
  250.         public void append(final StringBuffer buffer, final String fieldName,
  251.                            final short[] array, final Boolean fullDetail) {

  252.             if (fieldName == null) {
  253.                 throw new UnsupportedOperationException(
  254.                         "Field names are mandatory when using JsonToStringStyle");
  255.             }
  256.             if (!isFullDetail(fullDetail)) {
  257.                 throw new UnsupportedOperationException(
  258.                         "FullDetail must be true when using JsonToStringStyle");
  259.             }

  260.             super.append(buffer, fieldName, array, fullDetail);
  261.         }

  262.         @Override
  263.         protected void appendDetail(final StringBuffer buffer, final String fieldName, final char value) {
  264.             appendValueAsString(buffer, String.valueOf(value));
  265.         }

  266.         @Override
  267.         protected void appendDetail(final StringBuffer buffer, final String fieldName, final Collection<?> coll) {
  268.             if (coll != null && !coll.isEmpty()) {
  269.                 buffer.append(getArrayStart());
  270.                 int i = 0;
  271.                 for (final Object item : coll) {
  272.                     appendDetail(buffer, fieldName, i++, item);
  273.                 }
  274.                 buffer.append(getArrayEnd());
  275.                 return;
  276.             }

  277.             buffer.append(coll);
  278.         }

  279.         @Override
  280.         protected void appendDetail(final StringBuffer buffer, final String fieldName, final Map<?, ?> map) {
  281.             if (map != null && !map.isEmpty()) {
  282.                 buffer.append(getContentStart());

  283.                 boolean firstItem = true;
  284.                 for (final Entry<?, ?> entry : map.entrySet()) {
  285.                     final String keyStr = Objects.toString(entry.getKey(), null);
  286.                     if (keyStr != null) {
  287.                         if (firstItem) {
  288.                             firstItem = false;
  289.                         } else {
  290.                             appendFieldEnd(buffer, keyStr);
  291.                         }
  292.                         appendFieldStart(buffer, keyStr);
  293.                         final Object value = entry.getValue();
  294.                         if (value == null) {
  295.                             appendNullText(buffer, keyStr);
  296.                         } else {
  297.                             appendInternal(buffer, keyStr, value, true);
  298.                         }
  299.                     }
  300.                 }

  301.                 buffer.append(getContentEnd());
  302.                 return;
  303.             }

  304.             buffer.append(map);
  305.         }

  306.         @Override
  307.         protected void appendDetail(final StringBuffer buffer, final String fieldName, final Object value) {

  308.             if (value == null) {
  309.                 appendNullText(buffer, fieldName);
  310.                 return;
  311.             }

  312.             if (value instanceof String || value instanceof Character) {
  313.                 appendValueAsString(buffer, value.toString());
  314.                 return;
  315.             }

  316.             if (value instanceof Number || value instanceof Boolean) {
  317.                 buffer.append(value);
  318.                 return;
  319.             }

  320.             final String valueAsString = value.toString();
  321.             if (isJsonObject(valueAsString) || isJsonArray(valueAsString)) {
  322.                 buffer.append(value);
  323.                 return;
  324.             }

  325.             appendDetail(buffer, fieldName, valueAsString);
  326.         }

  327.         @Override
  328.         protected void appendFieldStart(final StringBuffer buffer, final String fieldName) {

  329.             if (fieldName == null) {
  330.                 throw new UnsupportedOperationException(
  331.                         "Field names are mandatory when using JsonToStringStyle");
  332.             }

  333.             super.appendFieldStart(buffer, FIELD_NAME_QUOTE + StringEscapeUtils.escapeJson(fieldName)
  334.                     + FIELD_NAME_QUOTE);
  335.         }

  336.         /**
  337.          * Appends the given String enclosed in double-quotes to the given StringBuffer.
  338.          *
  339.          * @param buffer the StringBuffer to append the value to.
  340.          * @param value the value to append.
  341.          */
  342.         private void appendValueAsString(final StringBuffer buffer, final String value) {
  343.             buffer.append('"').append(StringEscapeUtils.escapeJson(value)).append('"');
  344.         }

  345.         private boolean isJsonArray(final String valueAsString) {
  346.             return valueAsString.startsWith(getArrayStart())
  347.                     && valueAsString.endsWith(getArrayEnd());
  348.         }

  349.         private boolean isJsonObject(final String valueAsString) {
  350.             return valueAsString.startsWith(getContentStart())
  351.                     && valueAsString.endsWith(getContentEnd());
  352.         }

  353.         /**
  354.          * Ensure Singleton after serialization.
  355.          *
  356.          * @return the singleton
  357.          */
  358.         private Object readResolve() {
  359.             return JSON_STYLE;
  360.         }

  361.     }

  362.     /**
  363.      * {@link ToStringStyle} that outputs on multiple lines.
  364.      *
  365.      * <p>This is an inner class rather than using
  366.      * {@link StandardToStringStyle} to ensure its immutability.</p>
  367.      */
  368.     private static final class MultiLineToStringStyle extends ToStringStyle {

  369.         private static final long serialVersionUID = 1L;

  370.         /**
  371.          * Constructs a new instance.
  372.          *
  373.          * <p>Use the static constant rather than instantiating.</p>
  374.          */
  375.         MultiLineToStringStyle() {
  376.             this.setContentStart("[");
  377.             this.setFieldSeparator(System.lineSeparator() + "  ");
  378.             this.setFieldSeparatorAtStart(true);
  379.             this.setContentEnd(System.lineSeparator() + "]");
  380.         }

  381.         /**
  382.          * Ensure Singleton after serialization.
  383.          *
  384.          * @return the singleton
  385.          */
  386.         private Object readResolve() {
  387.             return MULTI_LINE_STYLE;
  388.         }

  389.     }

  390.     /**
  391.      * {@link ToStringStyle} that does not print out the class name
  392.      * and identity hash code but prints content start and field names.
  393.      *
  394.      * <p>This is an inner class rather than using
  395.      * {@link StandardToStringStyle} to ensure its immutability.</p>
  396.      */
  397.     private static final class NoClassNameToStringStyle extends ToStringStyle {

  398.         private static final long serialVersionUID = 1L;

  399.         /**
  400.          * Constructs a new instance.
  401.          *
  402.          * <p>Use the static constant rather than instantiating.</p>
  403.          */
  404.         NoClassNameToStringStyle() {
  405.             this.setUseClassName(false);
  406.             this.setUseIdentityHashCode(false);
  407.         }

  408.         /**
  409.          * Ensure Singleton after serialization.
  410.          *
  411.          * @return the singleton
  412.          */
  413.         private Object readResolve() {
  414.             return NO_CLASS_NAME_STYLE;
  415.         }

  416.     }

  417.     /**
  418.      * {@link ToStringStyle} that does not print out
  419.      * the field names.
  420.      *
  421.      * <p>This is an inner class rather than using
  422.      * {@link StandardToStringStyle} to ensure its immutability.
  423.      */
  424.     private static final class NoFieldNameToStringStyle extends ToStringStyle {

  425.         private static final long serialVersionUID = 1L;

  426.         /**
  427.          * Constructs a new instance.
  428.          *
  429.          * <p>Use the static constant rather than instantiating.</p>
  430.          */
  431.         NoFieldNameToStringStyle() {
  432.             this.setUseFieldNames(false);
  433.         }

  434.         /**
  435.          * Ensure Singleton after serialization.
  436.          *
  437.          * @return the singleton
  438.          */
  439.         private Object readResolve() {
  440.             return NO_FIELD_NAMES_STYLE;
  441.         }

  442.     }

  443.     /**
  444.      * {@link ToStringStyle} that prints out the short
  445.      * class name and no identity hash code.
  446.      *
  447.      * <p>This is an inner class rather than using
  448.      * {@link StandardToStringStyle} to ensure its immutability.</p>
  449.      */
  450.     private static final class ShortPrefixToStringStyle extends ToStringStyle {

  451.         private static final long serialVersionUID = 1L;

  452.         /**
  453.          * Constructs a new instance.
  454.          *
  455.          * <p>Use the static constant rather than instantiating.</p>
  456.          */
  457.         ShortPrefixToStringStyle() {
  458.             this.setUseShortClassName(true);
  459.             this.setUseIdentityHashCode(false);
  460.         }

  461.         /**
  462.          * Ensure <code>Singleton</ode> after serialization.
  463.          * @return the singleton
  464.          */
  465.         private Object readResolve() {
  466.             return SHORT_PREFIX_STYLE;
  467.         }

  468.     }

  469.     /**
  470.      * {@link ToStringStyle} that does not print out the
  471.      * class name, identity hash code, content start or field name.
  472.      *
  473.      * <p>This is an inner class rather than using
  474.      * {@link StandardToStringStyle} to ensure its immutability.</p>
  475.      */
  476.     private static final class SimpleToStringStyle extends ToStringStyle {

  477.         private static final long serialVersionUID = 1L;

  478.         /**
  479.          * Constructs a new instance.
  480.          *
  481.          * <p>Use the static constant rather than instantiating.</p>
  482.          */
  483.         SimpleToStringStyle() {
  484.             this.setUseClassName(false);
  485.             this.setUseIdentityHashCode(false);
  486.             this.setUseFieldNames(false);
  487.             this.setContentStart(StringUtils.EMPTY);
  488.             this.setContentEnd(StringUtils.EMPTY);
  489.         }

  490.         /**
  491.          * Ensure <code>Singleton</ode> after serialization.
  492.          * @return the singleton
  493.          */
  494.         private Object readResolve() {
  495.             return SIMPLE_STYLE;
  496.         }

  497.     }

  498.     /**
  499.      * Serialization version ID.
  500.      */
  501.     private static final long serialVersionUID = -2587890625525655916L;

  502.     /**
  503.      * The default toString style. Using the {@code Person}
  504.      * example from {@link ToStringBuilder}, the output would look like this:
  505.      *
  506.      * <pre>
  507.      * Person@182f0db[name=John Doe,age=33,smoker=false]
  508.      * </pre>
  509.      */
  510.     public static final ToStringStyle DEFAULT_STYLE = new DefaultToStringStyle();

  511.     /**
  512.      * The multi line toString style. Using the {@code Person}
  513.      * example from {@link ToStringBuilder}, the output would look like this:
  514.      *
  515.      * <pre>
  516.      * Person@182f0db[
  517.      *   name=John Doe
  518.      *   age=33
  519.      *   smoker=false
  520.      * ]
  521.      * </pre>
  522.      */
  523.     public static final ToStringStyle MULTI_LINE_STYLE = new MultiLineToStringStyle();

  524.     /**
  525.      * The no field names toString style. Using the
  526.      * {@code Person} example from {@link ToStringBuilder}, the output
  527.      * would look like this:
  528.      *
  529.      * <pre>
  530.      * Person@182f0db[John Doe,33,false]
  531.      * </pre>
  532.      */
  533.     public static final ToStringStyle NO_FIELD_NAMES_STYLE = new NoFieldNameToStringStyle();

  534.     /**
  535.      * The short prefix toString style. Using the {@code Person} example
  536.      * from {@link ToStringBuilder}, the output would look like this:
  537.      *
  538.      * <pre>
  539.      * Person[name=John Doe,age=33,smoker=false]
  540.      * </pre>
  541.      *
  542.      * @since 2.1
  543.      */
  544.     public static final ToStringStyle SHORT_PREFIX_STYLE = new ShortPrefixToStringStyle();

  545.     /**
  546.      * The simple toString style. Using the {@code Person}
  547.      * example from {@link ToStringBuilder}, the output would look like this:
  548.      *
  549.      * <pre>
  550.      * John Doe,33,false
  551.      * </pre>
  552.      */
  553.     public static final ToStringStyle SIMPLE_STYLE = new SimpleToStringStyle();

  554.     /**
  555.      * The no class name toString style. Using the {@code Person}
  556.      * example from {@link ToStringBuilder}, the output would look like this:
  557.      *
  558.      * <pre>
  559.      * [name=John Doe,age=33,smoker=false]
  560.      * </pre>
  561.      *
  562.      * @since 3.4
  563.      */
  564.     public static final ToStringStyle NO_CLASS_NAME_STYLE = new NoClassNameToStringStyle();

  565.     /**
  566.      * The JSON toString style. Using the {@code Person} example from
  567.      * {@link ToStringBuilder}, the output would look like this:
  568.      *
  569.      * <pre>
  570.      * {"name": "John Doe", "age": 33, "smoker": true}
  571.      * </pre>
  572.      *
  573.      * <strong>Note:</strong> Since field names are mandatory in JSON, this
  574.      * ToStringStyle will throw an {@link UnsupportedOperationException} if no
  575.      * field name is passed in while appending. Furthermore This ToStringStyle
  576.      * will only generate valid JSON if referenced objects also produce JSON
  577.      * when calling {@code toString()} on them.
  578.      *
  579.      * @since 3.4
  580.      * @see <a href="https://www.json.org/">json.org</a>
  581.      */
  582.     public static final ToStringStyle JSON_STYLE = new JsonToStringStyle();

  583.     /**
  584.      * A registry of objects used by {@code reflectionToString} methods
  585.      * to detect cyclical object references and avoid infinite loops.
  586.      */
  587.     private static final ThreadLocal<WeakHashMap<Object, Object>> REGISTRY = ThreadLocal.withInitial(WeakHashMap::new);
  588.     /*
  589.      * Note that objects of this class are generally shared between threads, so
  590.      * an instance variable would not be suitable here.
  591.      *
  592.      * In normal use the registry should always be left empty, because the caller
  593.      * should call toString() which will clean up.
  594.      *
  595.      * See LANG-792
  596.      */

  597.     /**
  598.      * Returns the registry of objects being traversed by the {@code reflectionToString}
  599.      * methods in the current thread.
  600.      *
  601.      * @return Set the registry of objects being traversed
  602.      */
  603.     public static Map<Object, Object> getRegistry() {
  604.         return REGISTRY.get();
  605.     }

  606.     /**
  607.      * Returns {@code true} if the registry contains the given object.
  608.      * Used by the reflection methods to avoid infinite loops.
  609.      *
  610.      * @param value
  611.      *                  The object to lookup in the registry.
  612.      * @return boolean {@code true} if the registry contains the given
  613.      *             object.
  614.      */
  615.     static boolean isRegistered(final Object value) {
  616.         return getRegistry().containsKey(value);
  617.     }

  618.     /**
  619.      * Registers the given object. Used by the reflection methods to avoid
  620.      * infinite loops.
  621.      *
  622.      * @param value
  623.      *                  The object to register.
  624.      */
  625.     static void register(final Object value) {
  626.         if (value != null) {
  627.             getRegistry().put(value, null);
  628.         }
  629.     }

  630.     /**
  631.      * Unregisters the given object.
  632.      *
  633.      * <p>
  634.      * Used by the reflection methods to avoid infinite loops.
  635.      * </p>
  636.      *
  637.      * @param value
  638.      *                  The object to unregister.
  639.      */
  640.     static void unregister(final Object value) {
  641.         if (value != null) {
  642.             final Map<Object, Object> m = getRegistry();
  643.             m.remove(value);
  644.             if (m.isEmpty()) {
  645.                 REGISTRY.remove();
  646.             }
  647.         }
  648.     }

  649.     /**
  650.      * Whether to use the field names, the default is {@code true}.
  651.      */
  652.     private boolean useFieldNames = true;

  653.     /**
  654.      * Whether to use the class name, the default is {@code true}.
  655.      */
  656.     private boolean useClassName = true;

  657.     /**
  658.      * Whether to use short class names, the default is {@code false}.
  659.      */
  660.     private boolean useShortClassName;

  661.     /**
  662.      * Whether to use the identity hash code, the default is {@code true}.
  663.      */
  664.     private boolean useIdentityHashCode = true;

  665.     /**
  666.      * The content start {@code '['}.
  667.      */
  668.     private String contentStart = "[";

  669.     /**
  670.      * The content end {@code ']'}.
  671.      */
  672.     private String contentEnd = "]";

  673.     /**
  674.      * The field name value separator {@code '='}.
  675.      */
  676.     private String fieldNameValueSeparator = "=";

  677.     /**
  678.      * Whether the field separator should be added before any other fields.
  679.      */
  680.     private boolean fieldSeparatorAtStart;

  681.     /**
  682.      * Whether the field separator should be added after any other fields.
  683.      */
  684.     private boolean fieldSeparatorAtEnd;

  685.     /**
  686.      * The field separator {@code ','}.
  687.      */
  688.     private String fieldSeparator = ",";

  689.     /**
  690.      * The array start <code>'{'</code>.
  691.      */
  692.     private String arrayStart = "{";

  693.     /**
  694.      * The array separator {@code ','}.
  695.      */
  696.     private String arraySeparator = ",";

  697.     /**
  698.      * The detail for array content.
  699.      */
  700.     private boolean arrayContentDetail = true;

  701.     /**
  702.      * The array end {@code '}'}.
  703.      */
  704.     private String arrayEnd = "}";

  705.     /**
  706.      * The value to use when fullDetail is {@code null},
  707.      * the default value is {@code true}.
  708.      */
  709.     private boolean defaultFullDetail = true;

  710.     /**
  711.      * The {@code null} text {@code "<null>"}.
  712.      */
  713.     private String nullText = "<null>";

  714.     /**
  715.      * The summary size text start {@code "<size="}.
  716.      */
  717.     private String sizeStartText = "<size=";

  718.     /**
  719.      * The summary size text start {@code ">"}.
  720.      */
  721.     private String sizeEndText = ">";

  722.     /**
  723.      * The summary object text start {@code "<"}.
  724.      */
  725.     private String summaryObjectStartText = "<";

  726.     /**
  727.      * The summary object text start {@code ">"}.
  728.      */
  729.     private String summaryObjectEndText = ">";

  730.     /**
  731.      * Constructs a new instance.
  732.      */
  733.     protected ToStringStyle() {
  734.     }

  735.     /**
  736.      * Appends to the {@code toString} a {@code boolean}
  737.      * value.
  738.      *
  739.      * @param buffer  the {@link StringBuffer} to populate
  740.      * @param fieldName  the field name
  741.      * @param value  the value to add to the {@code toString}
  742.      */
  743.     public void append(final StringBuffer buffer, final String fieldName, final boolean value) {
  744.         appendFieldStart(buffer, fieldName);
  745.         appendDetail(buffer, fieldName, value);
  746.         appendFieldEnd(buffer, fieldName);
  747.     }

  748.     /**
  749.      * Appends to the {@code toString} a {@code boolean}
  750.      * array.
  751.      *
  752.      * @param buffer  the {@link StringBuffer} to populate
  753.      * @param fieldName  the field name
  754.      * @param array  the array to add to the toString
  755.      * @param fullDetail  {@code true} for detail, {@code false}
  756.      *  for summary info, {@code null} for style decides
  757.      */
  758.     public void append(final StringBuffer buffer, final String fieldName, final boolean[] array, final Boolean fullDetail) {
  759.         appendFieldStart(buffer, fieldName);

  760.         if (array == null) {
  761.             appendNullText(buffer, fieldName);

  762.         } else if (isFullDetail(fullDetail)) {
  763.             appendDetail(buffer, fieldName, array);

  764.         } else {
  765.             appendSummary(buffer, fieldName, array);
  766.         }

  767.         appendFieldEnd(buffer, fieldName);
  768.     }

  769.     /**
  770.      * Appends to the {@code toString} a {@code byte}
  771.      * value.
  772.      *
  773.      * @param buffer  the {@link StringBuffer} to populate
  774.      * @param fieldName  the field name
  775.      * @param value  the value to add to the {@code toString}
  776.      */
  777.     public void append(final StringBuffer buffer, final String fieldName, final byte value) {
  778.         appendFieldStart(buffer, fieldName);
  779.         appendDetail(buffer, fieldName, value);
  780.         appendFieldEnd(buffer, fieldName);
  781.     }

  782.     /**
  783.      * Appends to the {@code toString} a {@code byte}
  784.      * array.
  785.      *
  786.      * @param buffer  the {@link StringBuffer} to populate
  787.      * @param fieldName  the field name
  788.      * @param array  the array to add to the {@code toString}
  789.      * @param fullDetail  {@code true} for detail, {@code false}
  790.      *  for summary info, {@code null} for style decides
  791.      */
  792.     public void append(final StringBuffer buffer, final String fieldName, final byte[] array, final Boolean fullDetail) {
  793.         appendFieldStart(buffer, fieldName);

  794.         if (array == null) {
  795.             appendNullText(buffer, fieldName);

  796.         } else if (isFullDetail(fullDetail)) {
  797.             appendDetail(buffer, fieldName, array);

  798.         } else {
  799.             appendSummary(buffer, fieldName, array);
  800.         }

  801.         appendFieldEnd(buffer, fieldName);
  802.     }

  803.     /**
  804.      * Appends to the {@code toString} a {@code char}
  805.      * value.
  806.      *
  807.      * @param buffer  the {@link StringBuffer} to populate
  808.      * @param fieldName  the field name
  809.      * @param value  the value to add to the {@code toString}
  810.      */
  811.     public void append(final StringBuffer buffer, final String fieldName, final char value) {
  812.         appendFieldStart(buffer, fieldName);
  813.         appendDetail(buffer, fieldName, value);
  814.         appendFieldEnd(buffer, fieldName);
  815.     }

  816.     /**
  817.      * Appends to the {@code toString} a {@code char}
  818.      * array.
  819.      *
  820.      * @param buffer  the {@link StringBuffer} to populate
  821.      * @param fieldName  the field name
  822.      * @param array  the array to add to the {@code toString}
  823.      * @param fullDetail  {@code true} for detail, {@code false}
  824.      *  for summary info, {@code null} for style decides
  825.      */
  826.     public void append(final StringBuffer buffer, final String fieldName, final char[] array, final Boolean fullDetail) {
  827.         appendFieldStart(buffer, fieldName);

  828.         if (array == null) {
  829.             appendNullText(buffer, fieldName);

  830.         } else if (isFullDetail(fullDetail)) {
  831.             appendDetail(buffer, fieldName, array);

  832.         } else {
  833.             appendSummary(buffer, fieldName, array);
  834.         }

  835.         appendFieldEnd(buffer, fieldName);
  836.     }

  837.     /**
  838.      * Appends to the {@code toString} a {@code double}
  839.      * value.
  840.      *
  841.      * @param buffer  the {@link StringBuffer} to populate
  842.      * @param fieldName  the field name
  843.      * @param value  the value to add to the {@code toString}
  844.      */
  845.     public void append(final StringBuffer buffer, final String fieldName, final double value) {
  846.         appendFieldStart(buffer, fieldName);
  847.         appendDetail(buffer, fieldName, value);
  848.         appendFieldEnd(buffer, fieldName);
  849.     }

  850.     /**
  851.      * Appends to the {@code toString} a {@code double}
  852.      * array.
  853.      *
  854.      * @param buffer  the {@link StringBuffer} to populate
  855.      * @param fieldName  the field name
  856.      * @param array  the array to add to the toString
  857.      * @param fullDetail  {@code true} for detail, {@code false}
  858.      *  for summary info, {@code null} for style decides
  859.      */
  860.     public void append(final StringBuffer buffer, final String fieldName, final double[] array, final Boolean fullDetail) {
  861.         appendFieldStart(buffer, fieldName);

  862.         if (array == null) {
  863.             appendNullText(buffer, fieldName);

  864.         } else if (isFullDetail(fullDetail)) {
  865.             appendDetail(buffer, fieldName, array);

  866.         } else {
  867.             appendSummary(buffer, fieldName, array);
  868.         }

  869.         appendFieldEnd(buffer, fieldName);
  870.     }

  871.     /**
  872.      * Appends to the {@code toString} a {@code float}
  873.      * value.
  874.      *
  875.      * @param buffer  the {@link StringBuffer} to populate
  876.      * @param fieldName  the field name
  877.      * @param value  the value to add to the {@code toString}
  878.      */
  879.     public void append(final StringBuffer buffer, final String fieldName, final float value) {
  880.         appendFieldStart(buffer, fieldName);
  881.         appendDetail(buffer, fieldName, value);
  882.         appendFieldEnd(buffer, fieldName);
  883.     }

  884.     /**
  885.      * Appends to the {@code toString} a {@code float}
  886.      * array.
  887.      *
  888.      * @param buffer  the {@link StringBuffer} to populate
  889.      * @param fieldName  the field name
  890.      * @param array  the array to add to the toString
  891.      * @param fullDetail  {@code true} for detail, {@code false}
  892.      *  for summary info, {@code null} for style decides
  893.      */
  894.     public void append(final StringBuffer buffer, final String fieldName, final float[] array, final Boolean fullDetail) {
  895.         appendFieldStart(buffer, fieldName);

  896.         if (array == null) {
  897.             appendNullText(buffer, fieldName);

  898.         } else if (isFullDetail(fullDetail)) {
  899.             appendDetail(buffer, fieldName, array);

  900.         } else {
  901.             appendSummary(buffer, fieldName, array);
  902.         }

  903.         appendFieldEnd(buffer, fieldName);
  904.     }

  905.     /**
  906.      * Appends to the {@code toString} an {@code int}
  907.      * value.
  908.      *
  909.      * @param buffer  the {@link StringBuffer} to populate
  910.      * @param fieldName  the field name
  911.      * @param value  the value to add to the {@code toString}
  912.      */
  913.     public void append(final StringBuffer buffer, final String fieldName, final int value) {
  914.         appendFieldStart(buffer, fieldName);
  915.         appendDetail(buffer, fieldName, value);
  916.         appendFieldEnd(buffer, fieldName);
  917.     }

  918.     /**
  919.      * Appends to the {@code toString} an {@code int}
  920.      * array.
  921.      *
  922.      * @param buffer  the {@link StringBuffer} to populate
  923.      * @param fieldName  the field name
  924.      * @param array  the array to add to the {@code toString}
  925.      * @param fullDetail  {@code true} for detail, {@code false}
  926.      *  for summary info, {@code null} for style decides
  927.      */
  928.     public void append(final StringBuffer buffer, final String fieldName, final int[] array, final Boolean fullDetail) {
  929.         appendFieldStart(buffer, fieldName);

  930.         if (array == null) {
  931.             appendNullText(buffer, fieldName);

  932.         } else if (isFullDetail(fullDetail)) {
  933.             appendDetail(buffer, fieldName, array);

  934.         } else {
  935.             appendSummary(buffer, fieldName, array);
  936.         }

  937.         appendFieldEnd(buffer, fieldName);
  938.     }

  939.     /**
  940.      * <p>Appends to the {@code toString} a {@code long}
  941.      * value.
  942.      *
  943.      * @param buffer  the {@link StringBuffer} to populate
  944.      * @param fieldName  the field name
  945.      * @param value  the value to add to the {@code toString}
  946.      */
  947.     public void append(final StringBuffer buffer, final String fieldName, final long value) {
  948.         appendFieldStart(buffer, fieldName);
  949.         appendDetail(buffer, fieldName, value);
  950.         appendFieldEnd(buffer, fieldName);
  951.     }

  952.     /**
  953.      * Appends to the {@code toString} a {@code long}
  954.      * array.
  955.      *
  956.      * @param buffer  the {@link StringBuffer} to populate
  957.      * @param fieldName  the field name
  958.      * @param array  the array to add to the {@code toString}
  959.      * @param fullDetail  {@code true} for detail, {@code false}
  960.      *  for summary info, {@code null} for style decides
  961.      */
  962.     public void append(final StringBuffer buffer, final String fieldName, final long[] array, final Boolean fullDetail) {
  963.         appendFieldStart(buffer, fieldName);

  964.         if (array == null) {
  965.             appendNullText(buffer, fieldName);

  966.         } else if (isFullDetail(fullDetail)) {
  967.             appendDetail(buffer, fieldName, array);

  968.         } else {
  969.             appendSummary(buffer, fieldName, array);
  970.         }

  971.         appendFieldEnd(buffer, fieldName);
  972.     }

  973.     /**
  974.      * Appends to the {@code toString} an {@link Object}
  975.      * value, printing the full {@code toString} of the
  976.      * {@link Object} passed in.
  977.      *
  978.      * @param buffer  the {@link StringBuffer} to populate
  979.      * @param fieldName  the field name
  980.      * @param value  the value to add to the {@code toString}
  981.      * @param fullDetail  {@code true} for detail, {@code false}
  982.      *  for summary info, {@code null} for style decides
  983.      */
  984.     public void append(final StringBuffer buffer, final String fieldName, final Object value, final Boolean fullDetail) {
  985.         appendFieldStart(buffer, fieldName);

  986.         if (value == null) {
  987.             appendNullText(buffer, fieldName);

  988.         } else {
  989.             appendInternal(buffer, fieldName, value, isFullDetail(fullDetail));
  990.         }

  991.         appendFieldEnd(buffer, fieldName);
  992.     }

  993.     /**
  994.      * Appends to the {@code toString} an {@link Object}
  995.      * array.
  996.      *
  997.      * @param buffer  the {@link StringBuffer} to populate
  998.      * @param fieldName  the field name
  999.      * @param array  the array to add to the toString
  1000.      * @param fullDetail  {@code true} for detail, {@code false}
  1001.      *  for summary info, {@code null} for style decides
  1002.      */
  1003.     public void append(final StringBuffer buffer, final String fieldName, final Object[] array, final Boolean fullDetail) {
  1004.         appendFieldStart(buffer, fieldName);

  1005.         if (array == null) {
  1006.             appendNullText(buffer, fieldName);

  1007.         } else if (isFullDetail(fullDetail)) {
  1008.             appendDetail(buffer, fieldName, array);

  1009.         } else {
  1010.             appendSummary(buffer, fieldName, array);
  1011.         }

  1012.         appendFieldEnd(buffer, fieldName);
  1013.     }

  1014.     /**
  1015.      * Appends to the {@code toString} a {@code short}
  1016.      * value.
  1017.      *
  1018.      * @param buffer  the {@link StringBuffer} to populate
  1019.      * @param fieldName  the field name
  1020.      * @param value  the value to add to the {@code toString}
  1021.      */
  1022.     public void append(final StringBuffer buffer, final String fieldName, final short value) {
  1023.         appendFieldStart(buffer, fieldName);
  1024.         appendDetail(buffer, fieldName, value);
  1025.         appendFieldEnd(buffer, fieldName);
  1026.     }

  1027.     /**
  1028.      * Appends to the {@code toString} a {@code short}
  1029.      * array.
  1030.      *
  1031.      * @param buffer  the {@link StringBuffer} to populate
  1032.      * @param fieldName  the field name
  1033.      * @param array  the array to add to the {@code toString}
  1034.      * @param fullDetail  {@code true} for detail, {@code false}
  1035.      *  for summary info, {@code null} for style decides
  1036.      */
  1037.     public void append(final StringBuffer buffer, final String fieldName, final short[] array, final Boolean fullDetail) {
  1038.         appendFieldStart(buffer, fieldName);

  1039.         if (array == null) {
  1040.             appendNullText(buffer, fieldName);

  1041.         } else if (isFullDetail(fullDetail)) {
  1042.             appendDetail(buffer, fieldName, array);

  1043.         } else {
  1044.             appendSummary(buffer, fieldName, array);
  1045.         }

  1046.         appendFieldEnd(buffer, fieldName);
  1047.     }

  1048.     /**
  1049.      * Appends to the {@code toString} the class name.
  1050.      *
  1051.      * @param buffer  the {@link StringBuffer} to populate
  1052.      * @param object  the {@link Object} whose name to output
  1053.      */
  1054.     protected void appendClassName(final StringBuffer buffer, final Object object) {
  1055.         if (useClassName && object != null) {
  1056.             register(object);
  1057.             if (useShortClassName) {
  1058.                 buffer.append(getShortClassName(object.getClass()));
  1059.             } else {
  1060.                 buffer.append(object.getClass().getName());
  1061.             }
  1062.         }
  1063.     }

  1064.     /**
  1065.      * Appends to the {@code toString} the content end.
  1066.      *
  1067.      * @param buffer  the {@link StringBuffer} to populate
  1068.      */
  1069.     protected void appendContentEnd(final StringBuffer buffer) {
  1070.         buffer.append(contentEnd);
  1071.     }

  1072.     /**
  1073.      * Appends to the {@code toString} the content start.
  1074.      *
  1075.      * @param buffer  the {@link StringBuffer} to populate
  1076.      */
  1077.     protected void appendContentStart(final StringBuffer buffer) {
  1078.         buffer.append(contentStart);
  1079.     }

  1080.     /**
  1081.      * Appends to the {@code toString} an {@link Object}
  1082.      * value that has been detected to participate in a cycle. This
  1083.      * implementation will print the standard string value of the value.
  1084.      *
  1085.      * @param buffer  the {@link StringBuffer} to populate
  1086.      * @param fieldName  the field name, typically not used as already appended
  1087.      * @param value  the value to add to the {@code toString},
  1088.      *  not {@code null}
  1089.      *
  1090.      * @since 2.2
  1091.      */
  1092.     protected void appendCyclicObject(final StringBuffer buffer, final String fieldName, final Object value) {
  1093.        ObjectUtils.identityToString(buffer, value);
  1094.     }

  1095.     /**
  1096.      * Appends to the {@code toString} a {@code boolean}
  1097.      * value.
  1098.      *
  1099.      * @param buffer  the {@link StringBuffer} to populate
  1100.      * @param fieldName  the field name, typically not used as already appended
  1101.      * @param value  the value to add to the {@code toString}
  1102.      */
  1103.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final boolean value) {
  1104.         buffer.append(value);
  1105.     }

  1106.     /**
  1107.      * Appends to the {@code toString} the detail of a
  1108.      * {@code boolean} array.
  1109.      *
  1110.      * @param buffer  the {@link StringBuffer} to populate
  1111.      * @param fieldName  the field name, typically not used as already appended
  1112.      * @param array  the array to add to the {@code toString},
  1113.      *  not {@code null}
  1114.      */
  1115.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final boolean[] array) {
  1116.         buffer.append(arrayStart);
  1117.         for (int i = 0; i < array.length; i++) {
  1118.             if (i > 0) {
  1119.                 buffer.append(arraySeparator);
  1120.             }
  1121.             appendDetail(buffer, fieldName, array[i]);
  1122.         }
  1123.         buffer.append(arrayEnd);
  1124.     }

  1125.     /**
  1126.      * Appends to the {@code toString} a {@code byte}
  1127.      * value.
  1128.      *
  1129.      * @param buffer  the {@link StringBuffer} to populate
  1130.      * @param fieldName  the field name, typically not used as already appended
  1131.      * @param value  the value to add to the {@code toString}
  1132.      */
  1133.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final byte value) {
  1134.         buffer.append(value);
  1135.     }

  1136.     /**
  1137.      * Appends to the {@code toString} the detail of a
  1138.      * {@code byte} array.
  1139.      *
  1140.      * @param buffer  the {@link StringBuffer} to populate
  1141.      * @param fieldName  the field name, typically not used as already appended
  1142.      * @param array  the array to add to the {@code toString},
  1143.      *  not {@code null}
  1144.      */
  1145.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final byte[] array) {
  1146.         buffer.append(arrayStart);
  1147.         for (int i = 0; i < array.length; i++) {
  1148.             if (i > 0) {
  1149.                 buffer.append(arraySeparator);
  1150.             }
  1151.             appendDetail(buffer, fieldName, array[i]);
  1152.         }
  1153.         buffer.append(arrayEnd);
  1154.     }

  1155.     /**
  1156.      * Appends to the {@code toString} a {@code char}
  1157.      * value.
  1158.      *
  1159.      * @param buffer  the {@link StringBuffer} to populate
  1160.      * @param fieldName  the field name, typically not used as already appended
  1161.      * @param value  the value to add to the {@code toString}
  1162.      */
  1163.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final char value) {
  1164.         buffer.append(value);
  1165.     }

  1166.     /**
  1167.      * Appends to the {@code toString} the detail of a
  1168.      * {@code char} array.
  1169.      *
  1170.      * @param buffer  the {@link StringBuffer} to populate
  1171.      * @param fieldName  the field name, typically not used as already appended
  1172.      * @param array  the array to add to the {@code toString},
  1173.      *  not {@code null}
  1174.      */
  1175.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final char[] array) {
  1176.         buffer.append(arrayStart);
  1177.         for (int i = 0; i < array.length; i++) {
  1178.             if (i > 0) {
  1179.                 buffer.append(arraySeparator);
  1180.             }
  1181.             appendDetail(buffer, fieldName, array[i]);
  1182.         }
  1183.         buffer.append(arrayEnd);
  1184.     }

  1185.     /**
  1186.      * Appends to the {@code toString} a {@link Collection}.
  1187.      *
  1188.      * @param buffer  the {@link StringBuffer} to populate
  1189.      * @param fieldName  the field name, typically not used as already appended
  1190.      * @param coll  the {@link Collection} to add to the
  1191.      *  {@code toString}, not {@code null}
  1192.      */
  1193.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final Collection<?> coll) {
  1194.         buffer.append(coll);
  1195.     }

  1196.     /**
  1197.      * Appends to the {@code toString} a {@code double}
  1198.      * value.
  1199.      *
  1200.      * @param buffer  the {@link StringBuffer} to populate
  1201.      * @param fieldName  the field name, typically not used as already appended
  1202.      * @param value  the value to add to the {@code toString}
  1203.      */
  1204.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final double value) {
  1205.         buffer.append(value);
  1206.     }

  1207.     /**
  1208.      * Appends to the {@code toString} the detail of a
  1209.      * {@code double} array.
  1210.      *
  1211.      * @param buffer  the {@link StringBuffer} to populate
  1212.      * @param fieldName  the field name, typically not used as already appended
  1213.      * @param array  the array to add to the {@code toString},
  1214.      *  not {@code null}
  1215.      */
  1216.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final double[] array) {
  1217.         buffer.append(arrayStart);
  1218.         for (int i = 0; i < array.length; i++) {
  1219.             if (i > 0) {
  1220.                 buffer.append(arraySeparator);
  1221.             }
  1222.             appendDetail(buffer, fieldName, array[i]);
  1223.         }
  1224.         buffer.append(arrayEnd);
  1225.     }

  1226.     /**
  1227.      * Appends to the {@code toString} a {@code float}
  1228.      * value.
  1229.      *
  1230.      * @param buffer  the {@link StringBuffer} to populate
  1231.      * @param fieldName  the field name, typically not used as already appended
  1232.      * @param value  the value to add to the {@code toString}
  1233.      */
  1234.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final float value) {
  1235.         buffer.append(value);
  1236.     }

  1237.     /**
  1238.      * Appends to the {@code toString} the detail of a
  1239.      * {@code float} array.
  1240.      *
  1241.      * @param buffer  the {@link StringBuffer} to populate
  1242.      * @param fieldName  the field name, typically not used as already appended
  1243.      * @param array  the array to add to the {@code toString},
  1244.      *  not {@code null}
  1245.      */
  1246.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final float[] array) {
  1247.         buffer.append(arrayStart);
  1248.         for (int i = 0; i < array.length; i++) {
  1249.             if (i > 0) {
  1250.                 buffer.append(arraySeparator);
  1251.             }
  1252.             appendDetail(buffer, fieldName, array[i]);
  1253.         }
  1254.         buffer.append(arrayEnd);
  1255.     }

  1256.     /**
  1257.      * Appends to the {@code toString} an {@code int}
  1258.      * value.
  1259.      *
  1260.      * @param buffer  the {@link StringBuffer} to populate
  1261.      * @param fieldName  the field name, typically not used as already appended
  1262.      * @param value  the value to add to the {@code toString}
  1263.      */
  1264.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final int value) {
  1265.         buffer.append(value);
  1266.     }

  1267.     /**
  1268.      * Appends to the {@code toString} the detail of an
  1269.      * {@link Object} array item.
  1270.      *
  1271.      * @param buffer  the {@link StringBuffer} to populate
  1272.      * @param fieldName  the field name, typically not used as already appended
  1273.      * @param i the array item index to add
  1274.      * @param item the array item to add
  1275.      * @since 3.11
  1276.      */
  1277.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final int i, final Object item) {
  1278.         if (i > 0) {
  1279.             buffer.append(arraySeparator);
  1280.         }
  1281.         if (item == null) {
  1282.             appendNullText(buffer, fieldName);
  1283.         } else {
  1284.             appendInternal(buffer, fieldName, item, arrayContentDetail);
  1285.         }
  1286.     }

  1287.     /**
  1288.      * Appends to the {@code toString} the detail of an
  1289.      * {@code int} array.
  1290.      *
  1291.      * @param buffer  the {@link StringBuffer} to populate
  1292.      * @param fieldName  the field name, typically not used as already appended
  1293.      * @param array  the array to add to the {@code toString},
  1294.      *  not {@code null}
  1295.      */
  1296.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final int[] array) {
  1297.         buffer.append(arrayStart);
  1298.         for (int i = 0; i < array.length; i++) {
  1299.             if (i > 0) {
  1300.                 buffer.append(arraySeparator);
  1301.             }
  1302.             appendDetail(buffer, fieldName, array[i]);
  1303.         }
  1304.         buffer.append(arrayEnd);
  1305.     }

  1306.     /**
  1307.      * Appends to the {@code toString} a {@code long}
  1308.      * value.
  1309.      *
  1310.      * @param buffer  the {@link StringBuffer} to populate
  1311.      * @param fieldName  the field name, typically not used as already appended
  1312.      * @param value  the value to add to the {@code toString}
  1313.      */
  1314.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final long value) {
  1315.         buffer.append(value);
  1316.     }

  1317.     /**
  1318.      * Appends to the {@code toString} the detail of a
  1319.      * {@code long} array.
  1320.      *
  1321.      * @param buffer  the {@link StringBuffer} to populate
  1322.      * @param fieldName  the field name, typically not used as already appended
  1323.      * @param array  the array to add to the {@code toString},
  1324.      *  not {@code null}
  1325.      */
  1326.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final long[] array) {
  1327.         buffer.append(arrayStart);
  1328.         for (int i = 0; i < array.length; i++) {
  1329.             if (i > 0) {
  1330.                 buffer.append(arraySeparator);
  1331.             }
  1332.             appendDetail(buffer, fieldName, array[i]);
  1333.         }
  1334.         buffer.append(arrayEnd);
  1335.     }

  1336.     /**
  1337.      * Appends to the {@code toString} a {@link Map}.
  1338.      *
  1339.      * @param buffer  the {@link StringBuffer} to populate
  1340.      * @param fieldName  the field name, typically not used as already appended
  1341.      * @param map  the {@link Map} to add to the {@code toString},
  1342.      *  not {@code null}
  1343.      */
  1344.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final Map<?, ?> map) {
  1345.         buffer.append(map);
  1346.     }

  1347.     /**
  1348.      * Appends to the {@code toString} an {@link Object}
  1349.      * value, printing the full detail of the {@link Object}.
  1350.      *
  1351.      * @param buffer  the {@link StringBuffer} to populate
  1352.      * @param fieldName  the field name, typically not used as already appended
  1353.      * @param value  the value to add to the {@code toString},
  1354.      *  not {@code null}
  1355.      */
  1356.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final Object value) {
  1357.         buffer.append(value);
  1358.     }

  1359.     /**
  1360.      * Appends to the {@code toString} the detail of an
  1361.      * {@link Object} array.
  1362.      *
  1363.      * @param buffer  the {@link StringBuffer} to populate
  1364.      * @param fieldName  the field name, typically not used as already appended
  1365.      * @param array  the array to add to the {@code toString},
  1366.      *  not {@code null}
  1367.      */
  1368.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final Object[] array) {
  1369.         buffer.append(arrayStart);
  1370.         for (int i = 0; i < array.length; i++) {
  1371.             appendDetail(buffer, fieldName, i, array[i]);
  1372.         }
  1373.         buffer.append(arrayEnd);
  1374.     }

  1375.     /**
  1376.      * Appends to the {@code toString} a {@code short}
  1377.      * value.
  1378.      *
  1379.      * @param buffer  the {@link StringBuffer} to populate
  1380.      * @param fieldName  the field name, typically not used as already appended
  1381.      * @param value  the value to add to the {@code toString}
  1382.      */
  1383.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final short value) {
  1384.         buffer.append(value);
  1385.     }

  1386.     /**
  1387.      * Appends to the {@code toString} the detail of a
  1388.      * {@code short} array.
  1389.      *
  1390.      * @param buffer  the {@link StringBuffer} to populate
  1391.      * @param fieldName  the field name, typically not used as already appended
  1392.      * @param array  the array to add to the {@code toString},
  1393.      *  not {@code null}
  1394.      */
  1395.     protected void appendDetail(final StringBuffer buffer, final String fieldName, final short[] array) {
  1396.         buffer.append(arrayStart);
  1397.         for (int i = 0; i < array.length; i++) {
  1398.             if (i > 0) {
  1399.                 buffer.append(arraySeparator);
  1400.             }
  1401.             appendDetail(buffer, fieldName, array[i]);
  1402.         }
  1403.         buffer.append(arrayEnd);
  1404.     }

  1405.     /**
  1406.      * Appends to the {@code toString} the end of data indicator.
  1407.      *
  1408.      * @param buffer  the {@link StringBuffer} to populate
  1409.      * @param object  the {@link Object} to build a
  1410.      *  {@code toString} for.
  1411.      */
  1412.     public void appendEnd(final StringBuffer buffer, final Object object) {
  1413.         if (!this.fieldSeparatorAtEnd) {
  1414.             removeLastFieldSeparator(buffer);
  1415.         }
  1416.         appendContentEnd(buffer);
  1417.         unregister(object);
  1418.     }

  1419.     /**
  1420.      * Appends to the {@code toString} the field end.
  1421.      *
  1422.      * @param buffer  the {@link StringBuffer} to populate
  1423.      * @param fieldName  the field name, typically not used as already appended
  1424.      */
  1425.     protected void appendFieldEnd(final StringBuffer buffer, final String fieldName) {
  1426.         appendFieldSeparator(buffer);
  1427.     }

  1428.     /**
  1429.      * Appends to the {@code toString} the field separator.
  1430.      *
  1431.      * @param buffer  the {@link StringBuffer} to populate
  1432.      */
  1433.     protected void appendFieldSeparator(final StringBuffer buffer) {
  1434.         buffer.append(fieldSeparator);
  1435.     }

  1436.     /**
  1437.      * Appends to the {@code toString} the field start.
  1438.      *
  1439.      * @param buffer  the {@link StringBuffer} to populate
  1440.      * @param fieldName  the field name
  1441.      */
  1442.     protected void appendFieldStart(final StringBuffer buffer, final String fieldName) {
  1443.         if (useFieldNames && fieldName != null) {
  1444.             buffer.append(fieldName);
  1445.             buffer.append(fieldNameValueSeparator);
  1446.         }
  1447.     }

  1448.     /**
  1449.      * Appends the {@link System#identityHashCode(java.lang.Object)}.
  1450.      *
  1451.      * @param buffer  the {@link StringBuffer} to populate
  1452.      * @param object  the {@link Object} whose id to output
  1453.      */
  1454.     protected void appendIdentityHashCode(final StringBuffer buffer, final Object object) {
  1455.         if (this.isUseIdentityHashCode() && object != null) {
  1456.             register(object);
  1457.             buffer.append('@');
  1458.             buffer.append(ObjectUtils.identityHashCodeHex(object));
  1459.         }
  1460.     }

  1461.     /**
  1462.      * Appends to the {@code toString} an {@link Object},
  1463.      * correctly interpreting its type.
  1464.      *
  1465.      * <p>This method performs the main lookup by Class type to correctly
  1466.      * route arrays, {@link Collection}s, {@link Map}s and
  1467.      * {@link Objects} to the appropriate method.</p>
  1468.      *
  1469.      * <p>Either detail or summary views can be specified.</p>
  1470.      *
  1471.      * <p>If a cycle is detected, an object will be appended with the
  1472.      * {@code Object.toString()} format.</p>
  1473.      *
  1474.      * @param buffer  the {@link StringBuffer} to populate
  1475.      * @param fieldName  the field name, typically not used as already appended
  1476.      * @param value  the value to add to the {@code toString},
  1477.      *  not {@code null}
  1478.      * @param detail  output detail or not
  1479.      */
  1480.     protected void appendInternal(final StringBuffer buffer, final String fieldName, final Object value, final boolean detail) {
  1481.         if (isRegistered(value)
  1482.             && !(value instanceof Number || value instanceof Boolean || value instanceof Character)) {
  1483.            appendCyclicObject(buffer, fieldName, value);
  1484.            return;
  1485.         }

  1486.         register(value);

  1487.         try {
  1488.             if (value instanceof Collection<?>) {
  1489.                 if (detail) {
  1490.                     appendDetail(buffer, fieldName, (Collection<?>) value);
  1491.                 } else {
  1492.                     appendSummarySize(buffer, fieldName, ((Collection<?>) value).size());
  1493.                 }

  1494.             } else if (value instanceof Map<?, ?>) {
  1495.                 if (detail) {
  1496.                     appendDetail(buffer, fieldName, (Map<?, ?>) value);
  1497.                 } else {
  1498.                     appendSummarySize(buffer, fieldName, ((Map<?, ?>) value).size());
  1499.                 }

  1500.             } else if (value instanceof long[]) {
  1501.                 if (detail) {
  1502.                     appendDetail(buffer, fieldName, (long[]) value);
  1503.                 } else {
  1504.                     appendSummary(buffer, fieldName, (long[]) value);
  1505.                 }

  1506.             } else if (value instanceof int[]) {
  1507.                 if (detail) {
  1508.                     appendDetail(buffer, fieldName, (int[]) value);
  1509.                 } else {
  1510.                     appendSummary(buffer, fieldName, (int[]) value);
  1511.                 }

  1512.             } else if (value instanceof short[]) {
  1513.                 if (detail) {
  1514.                     appendDetail(buffer, fieldName, (short[]) value);
  1515.                 } else {
  1516.                     appendSummary(buffer, fieldName, (short[]) value);
  1517.                 }

  1518.             } else if (value instanceof byte[]) {
  1519.                 if (detail) {
  1520.                     appendDetail(buffer, fieldName, (byte[]) value);
  1521.                 } else {
  1522.                     appendSummary(buffer, fieldName, (byte[]) value);
  1523.                 }

  1524.             } else if (value instanceof char[]) {
  1525.                 if (detail) {
  1526.                     appendDetail(buffer, fieldName, (char[]) value);
  1527.                 } else {
  1528.                     appendSummary(buffer, fieldName, (char[]) value);
  1529.                 }

  1530.             } else if (value instanceof double[]) {
  1531.                 if (detail) {
  1532.                     appendDetail(buffer, fieldName, (double[]) value);
  1533.                 } else {
  1534.                     appendSummary(buffer, fieldName, (double[]) value);
  1535.                 }

  1536.             } else if (value instanceof float[]) {
  1537.                 if (detail) {
  1538.                     appendDetail(buffer, fieldName, (float[]) value);
  1539.                 } else {
  1540.                     appendSummary(buffer, fieldName, (float[]) value);
  1541.                 }

  1542.             } else if (value instanceof boolean[]) {
  1543.                 if (detail) {
  1544.                     appendDetail(buffer, fieldName, (boolean[]) value);
  1545.                 } else {
  1546.                     appendSummary(buffer, fieldName, (boolean[]) value);
  1547.                 }

  1548.             } else if (ObjectUtils.isArray(value)) {
  1549.                 if (detail) {
  1550.                     appendDetail(buffer, fieldName, (Object[]) value);
  1551.                 } else {
  1552.                     appendSummary(buffer, fieldName, (Object[]) value);
  1553.                 }

  1554.             } else if (detail) {
  1555.                 appendDetail(buffer, fieldName, value);
  1556.             } else {
  1557.                 appendSummary(buffer, fieldName, value);
  1558.             }
  1559.         } finally {
  1560.             unregister(value);
  1561.         }
  1562.     }

  1563.     /**
  1564.      * Appends to the {@code toString} an indicator for {@code null}.
  1565.      *
  1566.      * <p>The default indicator is {@code "<null>"}.</p>
  1567.      *
  1568.      * @param buffer  the {@link StringBuffer} to populate
  1569.      * @param fieldName  the field name, typically not used as already appended
  1570.      */
  1571.     protected void appendNullText(final StringBuffer buffer, final String fieldName) {
  1572.         buffer.append(nullText);
  1573.     }

  1574.     /**
  1575.      * Appends to the {@code toString} the start of data indicator.
  1576.      *
  1577.      * @param buffer  the {@link StringBuffer} to populate
  1578.      * @param object  the {@link Object} to build a {@code toString} for
  1579.      */
  1580.     public void appendStart(final StringBuffer buffer, final Object object) {
  1581.         if (object != null) {
  1582.             appendClassName(buffer, object);
  1583.             appendIdentityHashCode(buffer, object);
  1584.             appendContentStart(buffer);
  1585.             if (fieldSeparatorAtStart) {
  1586.                 appendFieldSeparator(buffer);
  1587.             }
  1588.         }
  1589.     }

  1590.     /**
  1591.      * Appends to the {@code toString} a summary of a
  1592.      * {@code boolean} array.
  1593.      *
  1594.      * @param buffer  the {@link StringBuffer} to populate
  1595.      * @param fieldName  the field name, typically not used as already appended
  1596.      * @param array  the array to add to the {@code toString},
  1597.      *  not {@code null}
  1598.      */
  1599.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final boolean[] array) {
  1600.         appendSummarySize(buffer, fieldName, array.length);
  1601.     }

  1602.     /**
  1603.      * Appends to the {@code toString} a summary of a
  1604.      * {@code byte} array.
  1605.      *
  1606.      * @param buffer  the {@link StringBuffer} to populate
  1607.      * @param fieldName  the field name, typically not used as already appended
  1608.      * @param array  the array to add to the {@code toString},
  1609.      *  not {@code null}
  1610.      */
  1611.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final byte[] array) {
  1612.         appendSummarySize(buffer, fieldName, array.length);
  1613.     }

  1614.     /**
  1615.      * Appends to the {@code toString} a summary of a
  1616.      * {@code char} array.
  1617.      *
  1618.      * @param buffer  the {@link StringBuffer} to populate
  1619.      * @param fieldName  the field name, typically not used as already appended
  1620.      * @param array  the array to add to the {@code toString},
  1621.      *  not {@code null}
  1622.      */
  1623.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final char[] array) {
  1624.         appendSummarySize(buffer, fieldName, array.length);
  1625.     }

  1626.     /**
  1627.      * Appends to the {@code toString} a summary of a
  1628.      * {@code double} array.
  1629.      *
  1630.      * @param buffer  the {@link StringBuffer} to populate
  1631.      * @param fieldName  the field name, typically not used as already appended
  1632.      * @param array  the array to add to the {@code toString},
  1633.      *  not {@code null}
  1634.      */
  1635.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final double[] array) {
  1636.         appendSummarySize(buffer, fieldName, array.length);
  1637.     }

  1638.     /**
  1639.      * Appends to the {@code toString} a summary of a
  1640.      * {@code float} array.
  1641.      *
  1642.      * @param buffer  the {@link StringBuffer} to populate
  1643.      * @param fieldName  the field name, typically not used as already appended
  1644.      * @param array  the array to add to the {@code toString},
  1645.      *  not {@code null}
  1646.      */
  1647.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final float[] array) {
  1648.         appendSummarySize(buffer, fieldName, array.length);
  1649.     }

  1650.     /**
  1651.      * Appends to the {@code toString} a summary of an
  1652.      * {@code int} array.
  1653.      *
  1654.      * @param buffer  the {@link StringBuffer} to populate
  1655.      * @param fieldName  the field name, typically not used as already appended
  1656.      * @param array  the array to add to the {@code toString},
  1657.      *  not {@code null}
  1658.      */
  1659.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final int[] array) {
  1660.         appendSummarySize(buffer, fieldName, array.length);
  1661.     }

  1662.     /**
  1663.      * Appends to the {@code toString} a summary of a
  1664.      * {@code long} array.
  1665.      *
  1666.      * @param buffer  the {@link StringBuffer} to populate
  1667.      * @param fieldName  the field name, typically not used as already appended
  1668.      * @param array  the array to add to the {@code toString},
  1669.      *  not {@code null}
  1670.      */
  1671.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final long[] array) {
  1672.         appendSummarySize(buffer, fieldName, array.length);
  1673.     }

  1674.     /**
  1675.      * Appends to the {@code toString} an {@link Object}
  1676.      * value, printing a summary of the {@link Object}.
  1677.      *
  1678.      * @param buffer  the {@link StringBuffer} to populate
  1679.      * @param fieldName  the field name, typically not used as already appended
  1680.      * @param value  the value to add to the {@code toString},
  1681.      *  not {@code null}
  1682.      */
  1683.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final Object value) {
  1684.         buffer.append(summaryObjectStartText);
  1685.         buffer.append(getShortClassName(value.getClass()));
  1686.         buffer.append(summaryObjectEndText);
  1687.     }

  1688.     /**
  1689.      * Appends to the {@code toString} a summary of an
  1690.      * {@link Object} array.
  1691.      *
  1692.      * @param buffer  the {@link StringBuffer} to populate
  1693.      * @param fieldName  the field name, typically not used as already appended
  1694.      * @param array  the array to add to the {@code toString},
  1695.      *  not {@code null}
  1696.      */
  1697.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final Object[] array) {
  1698.         appendSummarySize(buffer, fieldName, array.length);
  1699.     }

  1700.     /**
  1701.      * Appends to the {@code toString} a summary of a
  1702.      * {@code short} array.
  1703.      *
  1704.      * @param buffer  the {@link StringBuffer} to populate
  1705.      * @param fieldName  the field name, typically not used as already appended
  1706.      * @param array  the array to add to the {@code toString},
  1707.      *  not {@code null}
  1708.      */
  1709.     protected void appendSummary(final StringBuffer buffer, final String fieldName, final short[] array) {
  1710.         appendSummarySize(buffer, fieldName, array.length);
  1711.     }

  1712.     /**
  1713.      * Appends to the {@code toString} a size summary.
  1714.      *
  1715.      * <p>The size summary is used to summarize the contents of
  1716.      * {@link Collection}s, {@link Map}s and arrays.</p>
  1717.      *
  1718.      * <p>The output consists of a prefix, the passed in size
  1719.      * and a suffix.</p>
  1720.      *
  1721.      * <p>The default format is {@code "<size=n>"}.</p>
  1722.      *
  1723.      * @param buffer  the {@link StringBuffer} to populate
  1724.      * @param fieldName  the field name, typically not used as already appended
  1725.      * @param size  the size to append
  1726.      */
  1727.     protected void appendSummarySize(final StringBuffer buffer, final String fieldName, final int size) {
  1728.         buffer.append(sizeStartText);
  1729.         buffer.append(size);
  1730.         buffer.append(sizeEndText);
  1731.     }

  1732.     /**
  1733.      * Appends to the {@code toString} the superclass toString.
  1734.      * <p>NOTE: It assumes that the toString has been created from the same ToStringStyle.</p>
  1735.      *
  1736.      * <p>A {@code null} {@code superToString} is ignored.</p>
  1737.      *
  1738.      * @param buffer  the {@link StringBuffer} to populate
  1739.      * @param superToString  the {@code super.toString()}
  1740.      * @since 2.0
  1741.      */
  1742.     public void appendSuper(final StringBuffer buffer, final String superToString) {
  1743.         appendToString(buffer, superToString);
  1744.     }

  1745.     /**
  1746.      * Appends to the {@code toString} another toString.
  1747.      * <p>NOTE: It assumes that the toString has been created from the same ToStringStyle.</p>
  1748.      *
  1749.      * <p>A {@code null} {@code toString} is ignored.</p>
  1750.      *
  1751.      * @param buffer  the {@link StringBuffer} to populate
  1752.      * @param toString  the additional {@code toString}
  1753.      * @since 2.0
  1754.      */
  1755.     public void appendToString(final StringBuffer buffer, final String toString) {
  1756.         if (toString != null) {
  1757.             final int pos1 = toString.indexOf(contentStart) + contentStart.length();
  1758.             final int pos2 = toString.lastIndexOf(contentEnd);
  1759.             if (pos1 != pos2 && pos1 >= 0 && pos2 >= 0) {
  1760.                 if (fieldSeparatorAtStart) {
  1761.                     removeLastFieldSeparator(buffer);
  1762.                 }
  1763.                 buffer.append(toString, pos1, pos2);
  1764.                 appendFieldSeparator(buffer);
  1765.             }
  1766.         }
  1767.     }

  1768.     /**
  1769.      * Gets the array end text.
  1770.      *
  1771.      * @return the current array end text
  1772.      */
  1773.     protected String getArrayEnd() {
  1774.         return arrayEnd;
  1775.     }

  1776.     /**
  1777.      * Gets the array separator text.
  1778.      *
  1779.      * @return the current array separator text
  1780.      */
  1781.     protected String getArraySeparator() {
  1782.         return arraySeparator;
  1783.     }

  1784.     /**
  1785.      * Gets the array start text.
  1786.      *
  1787.      * @return the current array start text
  1788.      */
  1789.     protected String getArrayStart() {
  1790.         return arrayStart;
  1791.     }

  1792.     /**
  1793.      * Gets the content end text.
  1794.      *
  1795.      * @return the current content end text
  1796.      */
  1797.     protected String getContentEnd() {
  1798.         return contentEnd;
  1799.     }

  1800.     /**
  1801.      * Gets the content start text.
  1802.      *
  1803.      * @return the current content start text
  1804.      */
  1805.     protected String getContentStart() {
  1806.         return contentStart;
  1807.     }

  1808.     /**
  1809.      * Gets the field name value separator text.
  1810.      *
  1811.      * @return the current field name value separator text
  1812.      */
  1813.     protected String getFieldNameValueSeparator() {
  1814.         return fieldNameValueSeparator;
  1815.     }

  1816.     /**
  1817.      * Gets the field separator text.
  1818.      *
  1819.      * @return the current field separator text
  1820.      */
  1821.     protected String getFieldSeparator() {
  1822.         return fieldSeparator;
  1823.     }

  1824.     /**
  1825.      * Gets the text to output when {@code null} found.
  1826.      *
  1827.      * @return the current text to output when null found
  1828.      */
  1829.     protected String getNullText() {
  1830.         return nullText;
  1831.     }

  1832.     /**
  1833.      * Gets the short class name for a class.
  1834.      *
  1835.      * <p>The short class name is the class name excluding
  1836.      * the package name.</p>
  1837.      *
  1838.      * @param cls  the {@link Class} to get the short name of
  1839.      * @return the short name
  1840.      */
  1841.     protected String getShortClassName(final Class<?> cls) {
  1842.         return ClassUtils.getShortClassName(cls);
  1843.     }

  1844.     /**
  1845.      * Gets the end text to output when a {@link Collection},
  1846.      * {@link Map} or array size is output.
  1847.      *
  1848.      * <p>This is output after the size value.</p>
  1849.      *
  1850.      * @return the current end of size text
  1851.      */
  1852.     protected String getSizeEndText() {
  1853.         return sizeEndText;
  1854.     }

  1855.     /**
  1856.      * Gets the start text to output when a {@link Collection},
  1857.      * {@link Map} or array size is output.
  1858.      *
  1859.      * <p>This is output before the size value.</p>
  1860.      *
  1861.      * @return the current start of size text
  1862.      */
  1863.     protected String getSizeStartText() {
  1864.         return sizeStartText;
  1865.     }

  1866.     /**
  1867.      * Gets the end text to output when an {@link Object} is
  1868.      * output in summary mode.
  1869.      *
  1870.      * <p>This is output after the size value.</p>
  1871.      *
  1872.      * @return the current end of summary text
  1873.      */
  1874.     protected String getSummaryObjectEndText() {
  1875.         return summaryObjectEndText;
  1876.     }

  1877.     /**
  1878.      * Gets the start text to output when an {@link Object} is
  1879.      * output in summary mode.
  1880.      *
  1881.      * <p>This is output before the size value.</p>
  1882.      *
  1883.      * @return the current start of summary text
  1884.      */
  1885.     protected String getSummaryObjectStartText() {
  1886.         return summaryObjectStartText;
  1887.     }

  1888.     /**
  1889.      * Gets whether to output array content detail.
  1890.      *
  1891.      * @return the current array content detail setting
  1892.      */
  1893.     protected boolean isArrayContentDetail() {
  1894.         return arrayContentDetail;
  1895.     }

  1896.     /**
  1897.      * Gets whether to use full detail when the caller doesn't
  1898.      * specify.
  1899.      *
  1900.      * @return the current defaultFullDetail flag
  1901.      */
  1902.     protected boolean isDefaultFullDetail() {
  1903.         return defaultFullDetail;
  1904.     }

  1905.     /**
  1906.      * Gets whether the field separator should be added at the end
  1907.      * of each buffer.
  1908.      *
  1909.      * @return fieldSeparatorAtEnd flag
  1910.      * @since 2.0
  1911.      */
  1912.     protected boolean isFieldSeparatorAtEnd() {
  1913.         return fieldSeparatorAtEnd;
  1914.     }

  1915.     /**
  1916.      * Gets whether the field separator should be added at the start
  1917.      * of each buffer.
  1918.      *
  1919.      * @return the fieldSeparatorAtStart flag
  1920.      * @since 2.0
  1921.      */
  1922.     protected boolean isFieldSeparatorAtStart() {
  1923.         return fieldSeparatorAtStart;
  1924.     }

  1925.     /**
  1926.      * Is this field to be output in full detail.
  1927.      *
  1928.      * <p>This method converts a detail request into a detail level.
  1929.      * The calling code may request full detail ({@code true}),
  1930.      * but a subclass might ignore that and always return
  1931.      * {@code false}. The calling code may pass in
  1932.      * {@code null} indicating that it doesn't care about
  1933.      * the detail level. In this case the default detail level is
  1934.      * used.</p>
  1935.      *
  1936.      * @param fullDetailRequest  the detail level requested
  1937.      * @return whether full detail is to be shown
  1938.      */
  1939.     protected boolean isFullDetail(final Boolean fullDetailRequest) {
  1940.         if (fullDetailRequest == null) {
  1941.             return defaultFullDetail;
  1942.         }
  1943.         return fullDetailRequest.booleanValue();
  1944.     }

  1945.     // Setters and getters for the customizable parts of the style
  1946.     // These methods are not expected to be overridden, except to make public
  1947.     // (They are not public so that immutable subclasses can be written)
  1948.     /**
  1949.      * Gets whether to use the class name.
  1950.      *
  1951.      * @return the current useClassName flag
  1952.      */
  1953.     protected boolean isUseClassName() {
  1954.         return useClassName;
  1955.     }

  1956.     /**
  1957.      * Gets whether to use the field names passed in.
  1958.      *
  1959.      * @return the current useFieldNames flag
  1960.      */
  1961.     protected boolean isUseFieldNames() {
  1962.         return useFieldNames;
  1963.     }

  1964.     /**
  1965.      * Gets whether to use the identity hash code.
  1966.      *
  1967.      * @return the current useIdentityHashCode flag
  1968.      */
  1969.     protected boolean isUseIdentityHashCode() {
  1970.         return useIdentityHashCode;
  1971.     }

  1972.     /**
  1973.      * Gets whether to output short or long class names.
  1974.      *
  1975.      * @return the current useShortClassName flag
  1976.      * @since 2.0
  1977.      */
  1978.     protected boolean isUseShortClassName() {
  1979.         return useShortClassName;
  1980.     }

  1981.     /**
  1982.      * Appends to the {@code toString} the detail of an array type.
  1983.      *
  1984.      * @param buffer  the {@link StringBuffer} to populate
  1985.      * @param fieldName  the field name, typically not used as already appended
  1986.      * @param array  the array to add to the {@code toString},
  1987.      *  not {@code null}
  1988.      * @since 2.0
  1989.      */
  1990.     protected void reflectionAppendArrayDetail(final StringBuffer buffer, final String fieldName, final Object array) {
  1991.         buffer.append(arrayStart);
  1992.         final int length = Array.getLength(array);
  1993.         for (int i = 0; i < length; i++) {
  1994.             appendDetail(buffer, fieldName, i, Array.get(array, i));
  1995.         }
  1996.         buffer.append(arrayEnd);
  1997.     }

  1998.     /**
  1999.      * Remove the last field separator from the buffer.
  2000.      *
  2001.      * @param buffer  the {@link StringBuffer} to populate
  2002.      * @since 2.0
  2003.      */
  2004.     protected void removeLastFieldSeparator(final StringBuffer buffer) {
  2005.         if (StringUtils.endsWith(buffer, fieldSeparator)) {
  2006.             buffer.setLength(buffer.length() - fieldSeparator.length());
  2007.         }
  2008.     }

  2009.     /**
  2010.      * Sets whether to output array content detail.
  2011.      *
  2012.      * @param arrayContentDetail  the new arrayContentDetail flag
  2013.      */
  2014.     protected void setArrayContentDetail(final boolean arrayContentDetail) {
  2015.         this.arrayContentDetail = arrayContentDetail;
  2016.     }

  2017.     /**
  2018.      * Sets the array end text.
  2019.      *
  2020.      * <p>{@code null} is accepted, but will be converted to
  2021.      * an empty String.</p>
  2022.      *
  2023.      * @param arrayEnd  the new array end text
  2024.      */
  2025.     protected void setArrayEnd(String arrayEnd) {
  2026.         if (arrayEnd == null) {
  2027.             arrayEnd = StringUtils.EMPTY;
  2028.         }
  2029.         this.arrayEnd = arrayEnd;
  2030.     }

  2031.     /**
  2032.      * Sets the array separator text.
  2033.      *
  2034.      * <p>{@code null} is accepted, but will be converted to
  2035.      * an empty String.</p>
  2036.      *
  2037.      * @param arraySeparator  the new array separator text
  2038.      */
  2039.     protected void setArraySeparator(String arraySeparator) {
  2040.         if (arraySeparator == null) {
  2041.             arraySeparator = StringUtils.EMPTY;
  2042.         }
  2043.         this.arraySeparator = arraySeparator;
  2044.     }

  2045.     /**
  2046.      * Sets the array start text.
  2047.      *
  2048.      * <p>{@code null} is accepted, but will be converted to
  2049.      * an empty String.</p>
  2050.      *
  2051.      * @param arrayStart  the new array start text
  2052.      */
  2053.     protected void setArrayStart(String arrayStart) {
  2054.         if (arrayStart == null) {
  2055.             arrayStart = StringUtils.EMPTY;
  2056.         }
  2057.         this.arrayStart = arrayStart;
  2058.     }

  2059.     /**
  2060.      * Sets the content end text.
  2061.      *
  2062.      * <p>{@code null} is accepted, but will be converted to
  2063.      * an empty String.</p>
  2064.      *
  2065.      * @param contentEnd  the new content end text
  2066.      */
  2067.     protected void setContentEnd(String contentEnd) {
  2068.         if (contentEnd == null) {
  2069.             contentEnd = StringUtils.EMPTY;
  2070.         }
  2071.         this.contentEnd = contentEnd;
  2072.     }

  2073.     /**
  2074.      * Sets the content start text.
  2075.      *
  2076.      * <p>{@code null} is accepted, but will be converted to
  2077.      * an empty String.</p>
  2078.      *
  2079.      * @param contentStart  the new content start text
  2080.      */
  2081.     protected void setContentStart(String contentStart) {
  2082.         if (contentStart == null) {
  2083.             contentStart = StringUtils.EMPTY;
  2084.         }
  2085.         this.contentStart = contentStart;
  2086.     }

  2087.     /**
  2088.      * Sets whether to use full detail when the caller doesn't
  2089.      * specify.
  2090.      *
  2091.      * @param defaultFullDetail  the new defaultFullDetail flag
  2092.      */
  2093.     protected void setDefaultFullDetail(final boolean defaultFullDetail) {
  2094.         this.defaultFullDetail = defaultFullDetail;
  2095.     }

  2096.     /**
  2097.      * Sets the field name value separator text.
  2098.      *
  2099.      * <p>{@code null} is accepted, but will be converted to
  2100.      * an empty String.</p>
  2101.      *
  2102.      * @param fieldNameValueSeparator  the new field name value separator text
  2103.      */
  2104.     protected void setFieldNameValueSeparator(String fieldNameValueSeparator) {
  2105.         if (fieldNameValueSeparator == null) {
  2106.             fieldNameValueSeparator = StringUtils.EMPTY;
  2107.         }
  2108.         this.fieldNameValueSeparator = fieldNameValueSeparator;
  2109.     }

  2110.     /**
  2111.      * Sets the field separator text.
  2112.      *
  2113.      * <p>{@code null} is accepted, but will be converted to
  2114.      * an empty String.</p>
  2115.      *
  2116.      * @param fieldSeparator  the new field separator text
  2117.      */
  2118.     protected void setFieldSeparator(String fieldSeparator) {
  2119.         if (fieldSeparator == null) {
  2120.             fieldSeparator = StringUtils.EMPTY;
  2121.         }
  2122.         this.fieldSeparator = fieldSeparator;
  2123.     }

  2124.     /**
  2125.      * Sets whether the field separator should be added at the end
  2126.      * of each buffer.
  2127.      *
  2128.      * @param fieldSeparatorAtEnd  the fieldSeparatorAtEnd flag
  2129.      * @since 2.0
  2130.      */
  2131.     protected void setFieldSeparatorAtEnd(final boolean fieldSeparatorAtEnd) {
  2132.         this.fieldSeparatorAtEnd = fieldSeparatorAtEnd;
  2133.     }

  2134.     /**
  2135.      * Sets whether the field separator should be added at the start
  2136.      * of each buffer.
  2137.      *
  2138.      * @param fieldSeparatorAtStart  the fieldSeparatorAtStart flag
  2139.      * @since 2.0
  2140.      */
  2141.     protected void setFieldSeparatorAtStart(final boolean fieldSeparatorAtStart) {
  2142.         this.fieldSeparatorAtStart = fieldSeparatorAtStart;
  2143.     }

  2144.     /**
  2145.      * Sets the text to output when {@code null} found.
  2146.      *
  2147.      * <p>{@code null} is accepted, but will be converted to
  2148.      * an empty String.</p>
  2149.      *
  2150.      * @param nullText  the new text to output when null found
  2151.      */
  2152.     protected void setNullText(String nullText) {
  2153.         if (nullText == null) {
  2154.             nullText = StringUtils.EMPTY;
  2155.         }
  2156.         this.nullText = nullText;
  2157.     }

  2158.     /**
  2159.      * Sets the end text to output when a {@link Collection},
  2160.      * {@link Map} or array size is output.
  2161.      *
  2162.      * <p>This is output after the size value.</p>
  2163.      *
  2164.      * <p>{@code null} is accepted, but will be converted to
  2165.      * an empty String.</p>
  2166.      *
  2167.      * @param sizeEndText  the new end of size text
  2168.      */
  2169.     protected void setSizeEndText(String sizeEndText) {
  2170.         if (sizeEndText == null) {
  2171.             sizeEndText = StringUtils.EMPTY;
  2172.         }
  2173.         this.sizeEndText = sizeEndText;
  2174.     }

  2175.     /**
  2176.      * Sets the start text to output when a {@link Collection},
  2177.      * {@link Map} or array size is output.
  2178.      *
  2179.      * <p>This is output before the size value.</p>
  2180.      *
  2181.      * <p>{@code null} is accepted, but will be converted to
  2182.      * an empty String.</p>
  2183.      *
  2184.      * @param sizeStartText  the new start of size text
  2185.      */
  2186.     protected void setSizeStartText(String sizeStartText) {
  2187.         if (sizeStartText == null) {
  2188.             sizeStartText = StringUtils.EMPTY;
  2189.         }
  2190.         this.sizeStartText = sizeStartText;
  2191.     }

  2192.     /**
  2193.      * Sets the end text to output when an {@link Object} is
  2194.      * output in summary mode.
  2195.      *
  2196.      * <p>This is output after the size value.</p>
  2197.      *
  2198.      * <p>{@code null} is accepted, but will be converted to
  2199.      * an empty String.</p>
  2200.      *
  2201.      * @param summaryObjectEndText  the new end of summary text
  2202.      */
  2203.     protected void setSummaryObjectEndText(String summaryObjectEndText) {
  2204.         if (summaryObjectEndText == null) {
  2205.             summaryObjectEndText = StringUtils.EMPTY;
  2206.         }
  2207.         this.summaryObjectEndText = summaryObjectEndText;
  2208.     }

  2209.     /**
  2210.      * Sets the start text to output when an {@link Object} is
  2211.      * output in summary mode.
  2212.      *
  2213.      * <p>This is output before the size value.</p>
  2214.      *
  2215.      * <p>{@code null} is accepted, but will be converted to
  2216.      * an empty String.</p>
  2217.      *
  2218.      * @param summaryObjectStartText  the new start of summary text
  2219.      */
  2220.     protected void setSummaryObjectStartText(String summaryObjectStartText) {
  2221.         if (summaryObjectStartText == null) {
  2222.             summaryObjectStartText = StringUtils.EMPTY;
  2223.         }
  2224.         this.summaryObjectStartText = summaryObjectStartText;
  2225.     }

  2226.     /**
  2227.      * Sets whether to use the class name.
  2228.      *
  2229.      * @param useClassName  the new useClassName flag
  2230.      */
  2231.     protected void setUseClassName(final boolean useClassName) {
  2232.         this.useClassName = useClassName;
  2233.     }

  2234.     /**
  2235.      * Sets whether to use the field names passed in.
  2236.      *
  2237.      * @param useFieldNames  the new useFieldNames flag
  2238.      */
  2239.     protected void setUseFieldNames(final boolean useFieldNames) {
  2240.         this.useFieldNames = useFieldNames;
  2241.     }

  2242.     /**
  2243.      * Sets whether to use the identity hash code.
  2244.      *
  2245.      * @param useIdentityHashCode  the new useIdentityHashCode flag
  2246.      */
  2247.     protected void setUseIdentityHashCode(final boolean useIdentityHashCode) {
  2248.         this.useIdentityHashCode = useIdentityHashCode;
  2249.     }

  2250.     /**
  2251.      * Sets whether to output short or long class names.
  2252.      *
  2253.      * @param useShortClassName  the new useShortClassName flag
  2254.      * @since 2.0
  2255.      */
  2256.     protected void setUseShortClassName(final boolean useShortClassName) {
  2257.         this.useShortClassName = useShortClassName;
  2258.     }
  2259. }