diff --git a/src/main/java/org/apache/commons/lang3/builder/ReflectionToStringBuilder.java b/src/main/java/org/apache/commons/lang3/builder/ReflectionToStringBuilder.java index 19faa9bed..e315ed20f 100644 --- a/src/main/java/org/apache/commons/lang3/builder/ReflectionToStringBuilder.java +++ b/src/main/java/org/apache/commons/lang3/builder/ReflectionToStringBuilder.java @@ -301,6 +301,64 @@ public class ReflectionToStringBuilder extends ToStringBuilder { return new ReflectionToStringBuilder(object, style, null, reflectUpToClass, outputTransients, outputStatics) .toString(); } + + /** + *

+ * Builds a toString value through reflection. + *

+ * + *

+ * It uses AccessibleObject.setAccessible to gain access to private fields. This means that it will + * throw a security exception if run under a security manager, if the permissions are not set up correctly. It is + * also not as efficient as testing explicitly. + *

+ * + *

+ * If the outputTransients is true, transient fields will be output, otherwise they + * are ignored, as they are likely derived fields, and not part of the value of the Object. + *

+ * + *

+ * If the outputStatics is true, static fields will be output, otherwise they are + * ignored. + *

+ * + *

+ * Superclass fields will be appended up to and including the specified superclass. A null superclass is treated as + * java.lang.Object. + *

+ * + *

+ * If the style is null, the default ToStringStyle is used. + *

+ * + * @param + * the type of the object + * @param object + * the Object to be output + * @param style + * the style of the toString to create, may be null + * @param outputTransients + * whether to include transient fields + * @param outputStatics + * whether to include static fields + * @param excludeNulls + * whether to exclude fields whose values are null + * @param reflectUpToClass + * the superclass to reflect up to (inclusive), may be null + * @return the String result + * @throws IllegalArgumentException + * if the Object is null + * + * @see ToStringExclude + * @since 2.1 + */ + public static String toString( + final T object, final ToStringStyle style, final boolean outputTransients, + final boolean outputStatics, boolean excludeNulls, final Class reflectUpToClass) { + return new ReflectionToStringBuilder(object, style, null, reflectUpToClass, outputTransients, outputStatics, excludeNulls) + .toString(); + } /** * Builds a String for a toString method excluding the given field names. @@ -378,6 +436,11 @@ public class ReflectionToStringBuilder extends ToStringBuilder { * Whether or not to append transient fields. */ private boolean appendTransients = false; + + /** + * Whether or not to append fields that are null. + */ + private boolean excludeNullValues; /** * Which field names to exclude from output. Intended for fields like "password". @@ -482,6 +545,38 @@ public class ReflectionToStringBuilder extends ToStringBuilder { this.setAppendTransients(outputTransients); this.setAppendStatics(outputStatics); } + + /** + * Constructor. + * + * @param + * the type of the object + * @param object + * the Object to build a toString for + * @param style + * the style of the toString to create, may be null + * @param buffer + * the StringBuffer to populate, may be null + * @param reflectUpToClass + * the superclass to reflect up to (inclusive), may be null + * @param outputTransients + * whether to include transient fields + * @param outputStatics + * whether to include static fields + * @param excludeNullValues + * whether to exclude fields who value is null + * @since 2.1 + */ + public ReflectionToStringBuilder( + final T object, final ToStringStyle style, final StringBuffer buffer, + final Class reflectUpToClass, final boolean outputTransients, final boolean outputStatics, + final boolean excludeNullValues) { + super(checkNotNull(object), style, buffer); + this.setUpToClass(reflectUpToClass); + this.setAppendTransients(outputTransients); + this.setAppendStatics(outputStatics); + this.setExcludeNullValues(excludeNullValues); + } /** * Returns whether or not to append the given Field. @@ -546,7 +641,9 @@ public class ReflectionToStringBuilder extends ToStringBuilder { // Warning: Field.get(Object) creates wrappers objects // for primitive types. final Object fieldValue = this.getValue(field); - this.append(fieldName, fieldValue); + if(!excludeNullValues || fieldValue != null){ + this.append(fieldName, fieldValue); + } } catch (final IllegalAccessException ex) { //this can't happen. Would get a Security exception // instead @@ -618,6 +715,17 @@ public class ReflectionToStringBuilder extends ToStringBuilder { public boolean isAppendTransients() { return this.appendTransients; } + + /** + *

+ * Gets whether or not to append fields whose values are null. + *

+ * + * @return Whether or not to append fields whose values are null. + */ + public boolean isExcludeNullValues() { + return this.excludeNullValues; + } /** *

@@ -657,6 +765,18 @@ public class ReflectionToStringBuilder extends ToStringBuilder { public void setAppendTransients(final boolean appendTransients) { this.appendTransients = appendTransients; } + + /** + *

+ * Sets whether or not to append fields whose values are null. + *

+ * + * @param excludeNullValues + * Whether or not to append fields whose values are null. + */ + public void setExcludeNullValues(final boolean excludeNullValues) { + this.excludeNullValues = excludeNullValues; + } /** * Sets the field names to exclude. diff --git a/src/test/java/org/apache/commons/lang3/builder/ReflectionToStringBuilderExcludeNullValuesTest.java b/src/test/java/org/apache/commons/lang3/builder/ReflectionToStringBuilderExcludeNullValuesTest.java new file mode 100644 index 000000000..0a91609c5 --- /dev/null +++ b/src/test/java/org/apache/commons/lang3/builder/ReflectionToStringBuilderExcludeNullValuesTest.java @@ -0,0 +1,161 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. + * The ASF licenses this file to You under the Apache License, Version 2.0 + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.apache.commons.lang3.builder; + +import static org.junit.Assert.*; + +import org.junit.Test; + +public class ReflectionToStringBuilderExcludeNullValuesTest { + + class TestFixture{ + private Integer testIntegerField; + private String testStringField; + + public TestFixture(Integer a, String b){ + this.testIntegerField = a; + this.testStringField = b; + } + } + + private static final String INTEGER_FIELD_NAME = "testIntegerField"; + private static final String STRING_FIELD_NAME = "testStringField"; + private final TestFixture BOTH_NON_NULL = new TestFixture(0, "str"); + private final TestFixture FIRST_NULL = new TestFixture(null, "str"); + private final TestFixture SECOND_NULL = new TestFixture(0, null); + private final TestFixture BOTH_NULL = new TestFixture(null, null); + + @Test + public void test_NonExclude(){ + //normal case= + String toString = ReflectionToStringBuilder.toString(BOTH_NON_NULL, null, false, false, false, null); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + //make one null + toString = ReflectionToStringBuilder.toString(FIRST_NULL, null, false, false, false, null); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + //other one null + toString = ReflectionToStringBuilder.toString(SECOND_NULL, null, false, false, false, null); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + //make the both null + toString = ReflectionToStringBuilder.toString(BOTH_NULL, null, false, false, false, null); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + } + + @Test + public void test_excludeNull(){ + + //test normal case + String toString = ReflectionToStringBuilder.toString(BOTH_NON_NULL, null, false, false, true, null); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + //make one null + toString = ReflectionToStringBuilder.toString(FIRST_NULL, null, false, false, true, null); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + //other one null + toString = ReflectionToStringBuilder.toString(SECOND_NULL, null, false, false, true, null); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertFalse(toString.contains(STRING_FIELD_NAME)); + + //both null + toString = ReflectionToStringBuilder.toString(BOTH_NULL, null, false, false, true, null); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + assertFalse(toString.contains(STRING_FIELD_NAME)); + } + + @Test + public void test_ConstructorOption(){ + ReflectionToStringBuilder builder = new ReflectionToStringBuilder(BOTH_NON_NULL, null, null, null, false, false, true); + assertTrue(builder.isExcludeNullValues()); + String toString = builder.toString(); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + builder = new ReflectionToStringBuilder(FIRST_NULL, null, null, null, false, false, true); + toString = builder.toString(); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + assertTrue(toString.contains(STRING_FIELD_NAME)); + + builder = new ReflectionToStringBuilder(SECOND_NULL, null, null, null, false, false, true); + toString = builder.toString(); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + assertFalse(toString.contains(STRING_FIELD_NAME)); + + builder = new ReflectionToStringBuilder(BOTH_NULL, null, null, null, false, false, true); + toString = builder.toString(); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + assertFalse(toString.contains(STRING_FIELD_NAME)); + } + + @Test + public void test_ConstructorOptionNormal(){ + ReflectionToStringBuilder builder = new ReflectionToStringBuilder(BOTH_NULL, null, null, null, false, false, false); + assertFalse(builder.isExcludeNullValues()); + String toString = builder.toString(); + assertTrue(toString.contains(STRING_FIELD_NAME)); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + + //regression test older constructors + ReflectionToStringBuilder oldBuilder = new ReflectionToStringBuilder(BOTH_NULL); + toString = oldBuilder.toString(); + assertTrue(toString.contains(STRING_FIELD_NAME)); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + + oldBuilder = new ReflectionToStringBuilder(BOTH_NULL, null, null, null, false, false); + toString = oldBuilder.toString(); + assertTrue(toString.contains(STRING_FIELD_NAME)); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + + oldBuilder = new ReflectionToStringBuilder(BOTH_NULL, null, null); + toString = oldBuilder.toString(); + assertTrue(toString.contains(STRING_FIELD_NAME)); + assertTrue(toString.contains(INTEGER_FIELD_NAME)); + } + + @Test + public void test_ConstructorOption_ExcludeNull(){ + ReflectionToStringBuilder builder = new ReflectionToStringBuilder(BOTH_NULL, null, null, null, false, false, false); + builder.setExcludeNullValues(true); + assertTrue(builder.isExcludeNullValues()); + String toString = builder.toString(); + assertFalse(toString.contains(STRING_FIELD_NAME)); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + + builder = new ReflectionToStringBuilder(BOTH_NULL, null, null, null, false, false, true); + toString = builder.toString(); + assertFalse(toString.contains(STRING_FIELD_NAME)); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + + ReflectionToStringBuilder oldBuilder = new ReflectionToStringBuilder(BOTH_NULL); + oldBuilder.setExcludeNullValues(true); + assertTrue(oldBuilder.isExcludeNullValues()); + toString = oldBuilder.toString(); + assertFalse(toString.contains(STRING_FIELD_NAME)); + assertFalse(toString.contains(INTEGER_FIELD_NAME)); + } + +}