可以将文章内容翻译成中文,广告屏蔽插件可能会导致该功能失效(如失效,请关闭广告屏蔽插件后再试):
问题:
I am making university project.
I need to get all fields from class. Even private and inherited. I tried to get all declared fields and then cast to super class and repeat.
Fragment of my code:
private void listAllFields(Object obj) {
List<Field> fieldList = new ArrayList<Field>();
while (obj != null) {
fieldList.addAll(Arrays.asList(obj.getClass().getDeclaredFields()));
obj = obj.getClass().getSuperclass().cast(obj);
}
// rest of code
But it does not work. tmpObj
after casting is still the same class (not superclass).
I will appreciate any help how to fix casting problem, or how to retrieve these fields in different way.
Problem is not to gain access to fields, but to get names of fields!
I manages it that way:
private void listAllFields(Object obj) {
List<Field> fieldList = new ArrayList<Field>();
Class tmpClass = obj.getClass();
while (tmpClass != null) {
fieldList.addAll(Arrays.asList(tmpClass .getDeclaredFields()));
tmpClass = tmpClass .getSuperclass();
}
// rest of code
回答1:
obj = obj.getClass().getSuperclass().cast(obj);
This line does not do what you expect it to do. Casting an Object
does not actually change it, it just tells the compiler to treat it as something else.
E.g. you can cast a List
to a Collection
, but it will still remain a List
.
However, looping up through the super classes to access fields works fine without casting:
Class<?> current = yourClass;
while(current.getSuperclass()!=null){ // we don't want to process Object.class
// do something with current's fields
current = current.getSuperclass();
}
BTW, if you have access to the Spring Framework, there is a handy method for looping through the fields of a class and all super classes:
ReflectionUtils.doWithFields(baseClass, FieldCallback)
(also see this previous answer of mine: Access to private inherited fields via reflection in Java)
回答2:
getDeclaredFields()
gives you all fields on that Class, including private
ones.
getFields()
gives you all public
fields on that Class AND it's superclasses.
If you want private
/ protected
methods of Super Classes, you will have to repeatedly call getSuperclass()
and then call getDeclaredFields()
on the Super Class object.
Nothing here isn't clearly explained in the javadocs
回答3:
Here is the method I use to get all the fields of an object
private <T> List<Field> getFields(T t) {
List<Field> fields = new ArrayList<>();
Class clazz = t.getClass();
while (clazz != Object.class) {
fields.addAll(Arrays.asList(clazz.getDeclaredFields()));
clazz = clazz.getSuperclass();
}
return fields;
}
回答4:
try these
How do I read a private field in Java?
Is it possible in Java to access private fields via reflection
Field.setAccessible(true) is what you have to do to make it accessible via reflection
import java.lang.reflect.Field;
class B
{
private int i = 5;
}
public class A extends B
{
public static void main(String[] args) throws Exception
{
A a = new A();
Field[] fs = a.getClass().getSuperclass().getDeclaredFields();
for (Field field : fs)
{
field.setAccessible(true);
System.out.println( field.get( a ) );
}
}
}
回答5:
This solution uses Java 8 streams, useful for those who are learning functional programming in Java. It iterates over getSuperclass and getDeclaredFields as per the other answers, but it does so in a functional way.
The following line will print the name of each field name found on SomeClass or any of its superclasses.
allFieldsFor(SomeClass.class).map(Field::getName).forEach(System.out::println);
Here is the code that steps through the superclasses to create the stream of fields.
private Stream<Field> allFieldsFor( Class c ) {
return walkInheritanceTreeFor(c).flatMap( k -> Arrays.stream(k.getDeclaredFields()) );
}
private Stream<Class> walkInheritanceTreeFor( Class c ) {
return iterate( c, k -> Optional.ofNullable(k.getSuperclass()) );
}
The following method is modelled from Streams.iterate, however Streams.iterate is designed to create infinite streams. This version has been modified to end when Optional.none() is returned from the fetchNextFunction.
private <T> Stream<T> iterate( T seed, Function<T,Optional<T>> fetchNextFunction ) {
Objects.requireNonNull(fetchNextFunction);
Iterator<T> iterator = new Iterator<T>() {
private Optional<T> t = Optional.ofNullable(seed);
public boolean hasNext() {
return t.isPresent();
}
public T next() {
T v = t.get();
t = fetchNextFunction.apply(v);
return v;
}
};
return StreamSupport.stream(
Spliterators.spliteratorUnknownSize( iterator, Spliterator.ORDERED | Spliterator.IMMUTABLE),
false
);
}
回答6:
To get superclass fields use getSuperclass()
. From there you can get fields of superclass.
回答7:
Yes, it is possible using Reflection.
In your code set Field.setAccessible(true);
Go ahead and learn the concept but DON'T do it :-)
回答8:
Previously asked.
Access to private inherited fields via reflection in Java
In your code, set field.setAccessible(true);
回答9:
ClassName obj = new ClassName();
Field field = ClassName.class.getDeclaredField ("name of the field");
field.setAccessible (true);
DataType value = (DataType) field.get (obj);
You have to use setAccessible to true before extracting the value. Sometime JVM does not allow to extract the private variable, it may give Security Exception.