`
java-mans
  • 浏览: 11450556 次
文章分类
社区版块
存档分类
最新评论

“关于使用反射机制得到泛型的字段名的问题”的解答

 
阅读更多

一、问题

在之前的“关于使用反射机制得到泛型的字段名的问题”博客当中提到了问题,终于在看了《Effective Java》之后有所理解。

问题回顾

1、实体类

public class LSLSEntry {
	private String storeguid;
	private String checkguid;
	private String isstoprunning;
	public String _5801;
	public String _5802;
	public String _5803;
	public String _6001;
	public String _6002;
	public String _6003;
	public String _6101;
	public String _6201;
	public String _6301;
	public String _6401;
	public String _6501;
	public String _6502;
	public String _6601;
	public String _6602;
	public String _6701;
	public String _6702;
	public String _6703;
	public String _6704;
	public String _6801;
	public String _6802;
	public String _6803;
	public String _6804;
	public String _6805;
	public String _7201;
	public String _7401;
	public String _7402;
	public String _7403;
	public String _7404;
	public String _7601;
	public String _7701;
	public String _7702;
	public String _7703;
	public String _7704;
	public String _7705;
	public String _7706;
	public String _7707;
	public String _7708;
	public String _7709;
	public String _7801;
	public String _7802;
	public String _8001;
	public String _8101;
	public String _8102;
	public String _8103;
	public String _8104;
	public String _8105;
	public String _8106;
	public String _8107;
	public String _8108;
	public String _8109;
	public String _8110;
	public String _8111;
	public String _8112;
	public String _8201;
	public String _8301;
	public String _8401;
	public String _8402;
	public String _8403;
	public String _8404;
	public String getStoreguid() {
		return storeguid;
	}
	public void setStoreguid(String storeguid) {
		this.storeguid = storeguid;
	}
	public String getCheckguid() {
		return checkguid;
	}
	public void setCheckguid(String checkguid) {
		this.checkguid = checkguid;
	}
	public String getIsstoprunning() {
		return isstoprunning;
	}
	public void setIsstoprunning(String isstoprunning) {
		this.isstoprunning = isstoprunning;
	}
	
}
2、反射代码

import java.lang.reflect.Field;

import org.junit.Test;

public class TestReflect {
	@Test
	public void test() {
		//可以得到字段名
		LSLSEntry entry = new LSLSEntry();
		printValue(entry);
	}

	@Test
	public void test1() {
		//得不到字段名,为什么???
		printValue(LSLSEntry.class);
	}

	@Test
	public void test2() {
		//得不到字段名,为什么??
		PrintInfo2<LSLSEntry> info = new PrintInfo2<LSLSEntry>(LSLSEntry.class);
		info.printValue();
	}

	@Test
	public void test3() {
		//可以得到字段名
		LSLSEntry entry = new LSLSEntry();
		PrintInfo<LSLSEntry> info = new PrintInfo<LSLSEntry>(entry);
		info.printValue();
	}

	@Test
	public void test4() {
		//可以得到字段名
		Field[] fields = LSLSEntry.class.getFields();
		for (Field f : fields) {
			System.out.println(f.getName());
		}
	}

	private <T> void printValue(T entry) {
		Field[] fields = entry.getClass().getFields();
		for (Field f : fields) {
			System.out.println(f.getName());
		}
	}

	private class PrintInfo<T> {
		T entry;
		public PrintInfo(T entry) {
			this.entry = entry;
		}

		private void printValue() {
			Field[] fields = entry.getClass().getFields();
			for (Field f : fields) {
				System.out.println(f.getName());
			}
		}
	}
	
	private class PrintInfo2<T> {
		Class<?> entry;
		public PrintInfo2(Class<?> entry) {
			this.entry = entry;
		}

		private void printValue() {
			Field[] fields = entry.getClass().getFields();
			for (Field f : fields) {
				System.out.println(f.getName());
			}
		}
	}

}

二、问题解答

1、源代码当中参数传递的错误。


public PrintInfo(T entry) {
			this.entry = entry;
		}
这种编写方式中T entry这样的参数表示的是引用的概念,同时这表示一个特定的类型

类型推导

而这个类型的推导过程是由entry的类型所推导出来的。编译器通过检查方法参数的类型来计算类型参数的值。对于如
public static <E> Set<E> union(Set<E> s1, Set<E> s2);
而言,编译器发现union的两个参数都是Set<String>类型,因此知道类型参数E必须为String。这个过程称作类型推导。

泛型特性

泛型在运行时会擦除掉的,因此T在运行时会被擦除掉,例如List<Integer>和List<String>在运行时都会解释为List。在向printValue传递*.class时传递的是类型信息而非对具体实例的引用,而这时的类型信息被擦除掉了,entry成了Object。因此调用 entry.getClass().getFields();时得不到东西。
这样的错误同时在只传递*.class的方法或类的调用当中出现,因此出现了以上错误。

2、为什么class<?>可以

private void printValue12(Class<?> clazz){
		System.out.println(clazz.getName());
		Field[] fields = clazz.getFields();
		for (Field f : fields) {
			System.out.println(f.getName());
		}
	}
	private <T> void printValue13(Class<T> clazz){
		System.out.println(clazz.getName());
		Field[] fields = clazz.getFields();
		for (Field f : fields) {
			System.out.println(f.getName());
		}
	}
两段代码可以呢?
原因还是在以上的解答,在这里,Class<?>存储的是类型信息,而传递来的.class正是类型信息。不管是Class<?>还是Class<T>都是类型信息,而T entry 这种方式传递的是引用。

三、总结

Class<?>是类型信息的概念。T entry 这种是引用的概念,在这种方式下有类型推导的过程。
正是因为对两者概念的不清楚导致了错误的产生。


分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics