src.main.java.com.mebigfatguy.fbcontrib.utils.ToString Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of fb-contrib Show documentation
Show all versions of fb-contrib Show documentation
An auxiliary findbugs.sourceforge.net plugin for java bug detectors that fall outside the narrow scope of detectors to be packaged with the product itself.
/*
* fb-contrib - Auxiliary detectors for Java programs
* Copyright (C) 2005-2018 Dave Brosius
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package com.mebigfatguy.fbcontrib.utils;
import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashSet;
import java.util.Set;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
/**
* an automatic toString() builder using reflection
*/
public class ToString {
/**
* holds objects that have already been converted to string to avoid infinite loops in the toString generation
*/
private static class VisitedInfo {
Set visitedHC = new HashSet<>();
int count = 0;
}
private static final ThreadLocal visited = new ThreadLocal() {
@Override
protected VisitedInfo initialValue() {
return new VisitedInfo();
}
};
private ToString() {
}
public static String build(Object o, String... ignoredFields) {
VisitedInfo vi = visited.get();
try {
vi.count++;
return generate(o, (ignoredFields == null) ? null : Arrays. asList(ignoredFields), vi.visitedHC);
} finally {
if (--vi.count == 0) {
vi.visitedHC.clear();
}
}
}
@SuppressFBWarnings(value = "RFI_SET_ACCESSIBLE", justification = "No other simple way to to show value of object")
private static String generate(Object o, Collection ignoredFields, Set visitedObjects) {
StringBuilder sb = new StringBuilder(100);
Class> cls = o.getClass();
Integer identityHC = Integer.valueOf(System.identityHashCode(o));
sb.append(cls.getSimpleName()).append('[').append(identityHC).append("]{");
if (visitedObjects.add(identityHC)) {
try {
String sep = "";
for (Field f : cls.getDeclaredFields()) {
String fieldName = f.getName();
if (!f.isSynthetic() && (fieldName.indexOf(Values.SYNTHETIC_MEMBER_CHAR) <= 0)
&& ((ignoredFields == null) || !ignoredFields.contains(fieldName))) {
sb.append(sep);
sep = ", ";
sb.append(fieldName).append('=');
try {
f.setAccessible(true);
Object value = f.get(o);
if (value == null) {
sb.append((String) null);
} else if (value.getClass().isArray()) {
sb.append(Arrays.toString((Object[]) value));
} else {
sb.append(value);
}
} catch (SecurityException e) {
sb.append("*SECURITY_EXCEPTION*");
}
}
}
} catch (IllegalAccessException | RuntimeException e) {
// if we get an exception show as much as we can get
}
}
sb.append('}');
return sb.toString();
}
}