Ответ 1
Apache commons-lang ReflectionToStringBuilder делает это для вас.
import org.apache.commons.lang3.builder.ReflectionToStringBuilder
// your code goes here
public String toString() {
return ReflectionToStringBuilder.toString(this);
}
Я писал toString() для класса в Java на днях, вручную выписав каждый элемент класса в String, и мне пришло в голову, что при использовании отражения может быть возможно создать общий метод toString() которые могут работать на ВСЕХ классах. И.Е. он будет определять имена полей и значения и отправлять их в строку.
Получение названий полей довольно просто, вот что посоветовал сотрудник:
public static List initFieldArray(String className) throws ClassNotFoundException {
Class c = Class.forName(className);
Field field[] = c.getFields();
List<String> classFields = new ArrayList(field.length);
for (int i = 0; i < field.length; i++) {
String cf = field[i].toString();
classFields.add(cf.substring(cf.lastIndexOf(".") + 1));
}
return classFields;
}
Используя factory, я мог бы снизить накладные расходы на производительность, сохраняя поля один раз, при первом вызове toString(). Однако найти значения может быть намного дороже.
Из-за производительности отражения это может быть скорее гипотетическим, чем практическим. Но меня интересует идея размышления и как я могу использовать его для улучшения моего повседневного программирования.
Apache commons-lang ReflectionToStringBuilder делает это для вас.
import org.apache.commons.lang3.builder.ReflectionToStringBuilder
// your code goes here
public String toString() {
return ReflectionToStringBuilder.toString(this);
}
W/отражение, поскольку я не знал об апачевой библиотеке:
(помните, что если вы это сделаете, вам, вероятно, придется иметь дело с подобъектами и убедиться, что они печатаются правильно - в частности, массивы не покажут вам ничего полезного)
@Override
public String toString()
{
StringBuilder b = new StringBuilder("[");
for (Field f : getClass().getFields())
{
if (!isStaticField(f))
{
try
{
b.append(f.getName() + "=" + f.get(this) + " ");
} catch (IllegalAccessException e)
{
// pass, don't print
}
}
}
b.append(']');
return b.toString();
}
private boolean isStaticField(Field f)
{
return Modifier.isStatic(f.getModifiers());
}
Если вы используете Eclipse, вы также можете посмотреть JUtils toString generator, который делает это статически (генерирование метода в вашем исходный код).
Другой вариант, если вы в порядке с JSON, это библиотека Google GSON.
public String toString() {
return new GsonBuilder().setPrettyPrinting().create().toJson(this);
}
Это сделает отражение для вас. Это создает приятный, удобный для чтения JSON файл. Легко читаемые, относительные, нетехнические люди могут найти JSON устрашающим.
Вы также можете сделать GSONBuilder переменной-членом, если вы не хотите обновлять ее каждый раз.
Если у вас есть данные, которые нельзя распечатать (например, поток) или данные, которые вы просто не хотите печатать, вы можете просто добавить теги @Expose к атрибутам, которые вы хотите распечатать, а затем использовать следующую строку.
new GsonBuilder()
.setPrettyPrinting()
.excludeFieldsWithoutExposeAnnotation()
.create()
.toJson(this);
Не отражение, но я взглянул на создание метода toString (вместе с equals/hashCode) в качестве этапа посткомпиляции с использованием манипуляции с байт-кодом. Результаты были смешаны.
Вы можете использовать уже реализованные библиотеки, как ReflectionToStringBuilder из Apache commons-lang. Как уже упоминалось.
Или напишите smt аналогично с помощью API отражения.
Вот пример:
class UniversalAnalyzer {
private ArrayList<Object> visited = new ArrayList<Object>();
/**
* Converts an object to a string representation that lists all fields.
* @param obj an object
* @return a string with the object class name and all field names and
* values
*/
public String toString(Object obj) {
if (obj == null) return "null";
if (visited.contains(obj)) return "...";
visited.add(obj);
Class cl = obj.getClass();
if (cl == String.class) return (String) obj;
if (cl.isArray()) {
String r = cl.getComponentType() + "[]{";
for (int i = 0; i < Array.getLength(obj); i++) {
if (i > 0) r += ",";
Object val = Array.get(obj, i);
if (cl.getComponentType().isPrimitive()) r += val;
else r += toString(val);
}
return r + "}";
}
String r = cl.getName();
// inspect the fields of this class and all superclasses
do {
r += "[";
Field[] fields = cl.getDeclaredFields();
AccessibleObject.setAccessible(fields, true);
// get the names and values of all fields
for (Field f : fields) {
if (!Modifier.isStatic(f.getModifiers())) {
if (!r.endsWith("[")) r += ",";
r += f.getName() + "=";
try {
Class t = f.getType();
Object val = f.get(obj);
if (t.isPrimitive()) r += val;
else r += toString(val);
} catch (Exception e) {
e.printStackTrace();
}
}
}
r += "]";
cl = cl.getSuperclass();
} while (cl != null);
return r;
}
}
Вот эквивалент Netbeans для ответа Оливье; плагин smart-codegen для Netbeans.