1.了解什么是Class类
Class类:当我们定义一个Dog类 class Dog{} 那么这个类在加载时会产生一个对象,是java.lang.Class的实例对象。比如:
class Dog{ private int id; private String name; private void say(){ System.out.println("汪汪汪..."); }}Dog dog=new Dog();
我们很清楚的知道dog是Dog类的实例对象,其实Dog这个类在jvm看来本身也是一个对象,他是java.lang.Class类的实例对象。
因为"Class"这个类的构造方法是私有的,我们没有办法通过new这个关键字来构造这个类的对象(dog),但是可以通过以下几个方法来构造这个类:
Class c=Dog.class;//通过类名获得Class类的对象(任何一个类都有隐含的成员变量Class) Class c=dog.getClass(); Class c=Class.forName("com.Dog");//这里写完整的类路径
通过以上方式我们获得了Class类的对象 c 它称为类的类型(class type)。
那么我们想要通过Dog类的对象dog除了Dog dog=new Dog()我们还可以通过获得的c来创建Dog类的实例:
Dog dog=(Dog)c.newInstance();//注意强制类型转换
其中Class.forName("类的全路径"),这是动态加载类,我们学习mysql时通过Class.forName("com.myql.jdbc.Driver"); 此时即使我们没有导入驱动的包它也不会报错,当我们真正运行时才会抛出找不到驱动类的异常。由此可以看出他是在运行时动态加载的。
我们常用的变量也会有Class的类类型(class type),比如:
Class c1=int.class; Class c2=String.class; Class c3=void.class;
当我们要获得类的名称时可以通过 c1.getName();此时返回 int, c2.getName()此时返回java.lang.String.
我们可以通过Class的函数获得各种信息,比如下面几个方法;
String className=c.getName();//获得了类的名称 Method[] ms=c.getMethods();//获得了Dog类以及从父类继承的所有public方法 Method[] ms=c.getDeclaredMethods();//获得类Dog类定义的所有方法 /* * 此时我们可以对获得的方法数组进行遍历 */ for(Method method:ms){ Class returnType=method.getReturnType();//得到了返回值的类型 String methodName=method.getName();//得到了返回值的名字 }
也可以得到成员变量的信息:(具体的自己参考帮助文档)
Field[] fs=c.getFields();//获得public类型的所有成员变量信息 Field[] fs=c.getDeclaredFields();//获得Dog类的所有成员变量
我们可以通过反射来调用私有的方法:
/** * c.getMethod(name, parameterTypes)获取的是public的方法 * c.getDeclaredMethod(name, parameterTypes)自己声明的方法 */ Method m=c.getDeclaredMethod("say", new Class[]{}); m.invoke(dog, new Object[]{});
以上我们调用了Dog的say方法。其中new Class[]{}为方法需要的参数,因为say方法不需要参数所以是空数组。
因此我们可以利用上述知识来完成一个自己的orm框架,传入任何一个对象将其存到数据库中。
public staticT save(T entity){ String tableName=entity.getClass().getAnnotation(Table.class).name(); String sql="insert into "+tableName+"("; String cols=""; String values=""; for(Field field:entity.getClass().getDeclaredFields()){//注意getFields()只能获得public访问权限的值,getDeclaredFields()可以获得所有的 String colName=field.getAnnotation(Column.class).name(); cols+=colName+","; try{ field.setAccessible(true);//通过反射访问该field时不受权限控制 if(field.getType().equals(String.class)) values+="'"+field.get(entity)+"',"; else values+=field.get(entity)+","; }catch(Exception e){ } } cols=cols.substring(0, cols.length()-1); values=values.substring(0, values.length()-1); sql+=cols+")values("+values+")"; System.out.println("SQL语句为:"+sql); //此时已经完整的拿到了sql语句,接下来执行改语句即可。 return null; }
我想这就是数据持久化框架(如Hibernate)的实现原理吧。