Merge branch 'LANG-1167' of https://github.com/MarkDacek/commons-lang
This commit is contained in:
commit
8147cc5b3d
|
@ -301,6 +301,64 @@ public class ReflectionToStringBuilder extends ToStringBuilder {
|
||||||
return new ReflectionToStringBuilder(object, style, null, reflectUpToClass, outputTransients, outputStatics)
|
return new ReflectionToStringBuilder(object, style, null, reflectUpToClass, outputTransients, outputStatics)
|
||||||
.toString();
|
.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Builds a <code>toString</code> value through reflection.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* It uses <code>AccessibleObject.setAccessible</code> 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.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* If the <code>outputTransients</code> is <code>true</code>, transient fields will be output, otherwise they
|
||||||
|
* are ignored, as they are likely derived fields, and not part of the value of the Object.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* If the <code>outputStatics</code> is <code>true</code>, static fields will be output, otherwise they are
|
||||||
|
* ignored.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* Superclass fields will be appended up to and including the specified superclass. A null superclass is treated as
|
||||||
|
* <code>java.lang.Object</code>.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* If the style is <code>null</code>, the default <code>ToStringStyle</code> is used.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @param <T>
|
||||||
|
* the type of the object
|
||||||
|
* @param object
|
||||||
|
* the Object to be output
|
||||||
|
* @param style
|
||||||
|
* the style of the <code>toString</code> to create, may be <code>null</code>
|
||||||
|
* @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 <code>null</code>
|
||||||
|
* @return the String result
|
||||||
|
* @throws IllegalArgumentException
|
||||||
|
* if the Object is <code>null</code>
|
||||||
|
*
|
||||||
|
* @see ToStringExclude
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static <T> String toString(
|
||||||
|
final T object, final ToStringStyle style, final boolean outputTransients,
|
||||||
|
final boolean outputStatics, boolean excludeNulls, final Class<? super T> reflectUpToClass) {
|
||||||
|
return new ReflectionToStringBuilder(object, style, null, reflectUpToClass, outputTransients, outputStatics, excludeNulls)
|
||||||
|
.toString();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Builds a String for a toString method excluding the given field names.
|
* 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.
|
* Whether or not to append transient fields.
|
||||||
*/
|
*/
|
||||||
private boolean appendTransients = false;
|
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 <code>"password"</code>.
|
* Which field names to exclude from output. Intended for fields like <code>"password"</code>.
|
||||||
|
@ -482,6 +545,38 @@ public class ReflectionToStringBuilder extends ToStringBuilder {
|
||||||
this.setAppendTransients(outputTransients);
|
this.setAppendTransients(outputTransients);
|
||||||
this.setAppendStatics(outputStatics);
|
this.setAppendStatics(outputStatics);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor.
|
||||||
|
*
|
||||||
|
* @param <T>
|
||||||
|
* the type of the object
|
||||||
|
* @param object
|
||||||
|
* the Object to build a <code>toString</code> for
|
||||||
|
* @param style
|
||||||
|
* the style of the <code>toString</code> to create, may be <code>null</code>
|
||||||
|
* @param buffer
|
||||||
|
* the <code>StringBuffer</code> to populate, may be <code>null</code>
|
||||||
|
* @param reflectUpToClass
|
||||||
|
* the superclass to reflect up to (inclusive), may be <code>null</code>
|
||||||
|
* @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 <T> ReflectionToStringBuilder(
|
||||||
|
final T object, final ToStringStyle style, final StringBuffer buffer,
|
||||||
|
final Class<? super T> 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 <code>Field</code>.
|
* Returns whether or not to append the given <code>Field</code>.
|
||||||
|
@ -546,7 +641,9 @@ public class ReflectionToStringBuilder extends ToStringBuilder {
|
||||||
// Warning: Field.get(Object) creates wrappers objects
|
// Warning: Field.get(Object) creates wrappers objects
|
||||||
// for primitive types.
|
// for primitive types.
|
||||||
final Object fieldValue = this.getValue(field);
|
final Object fieldValue = this.getValue(field);
|
||||||
this.append(fieldName, fieldValue);
|
if(!excludeNullValues || fieldValue != null){
|
||||||
|
this.append(fieldName, fieldValue);
|
||||||
|
}
|
||||||
} catch (final IllegalAccessException ex) {
|
} catch (final IllegalAccessException ex) {
|
||||||
//this can't happen. Would get a Security exception
|
//this can't happen. Would get a Security exception
|
||||||
// instead
|
// instead
|
||||||
|
@ -618,6 +715,17 @@ public class ReflectionToStringBuilder extends ToStringBuilder {
|
||||||
public boolean isAppendTransients() {
|
public boolean isAppendTransients() {
|
||||||
return this.appendTransients;
|
return this.appendTransients;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Gets whether or not to append fields whose values are null.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @return Whether or not to append fields whose values are null.
|
||||||
|
*/
|
||||||
|
public boolean isExcludeNullValues() {
|
||||||
|
return this.excludeNullValues;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>
|
* <p>
|
||||||
|
@ -657,6 +765,18 @@ public class ReflectionToStringBuilder extends ToStringBuilder {
|
||||||
public void setAppendTransients(final boolean appendTransients) {
|
public void setAppendTransients(final boolean appendTransients) {
|
||||||
this.appendTransients = appendTransients;
|
this.appendTransients = appendTransients;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Sets whether or not to append fields whose values are null.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @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.
|
* Sets the field names to exclude.
|
||||||
|
|
|
@ -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));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue