一、bean简介
在java的反射中,所有的类被抽象出一个类,即Class类,这样我们就可以在程序运行的过程中通过配置文件,动态的加载类。但是在用反射的时候有些前提,就是:当我们调用有参的constructor的时候必须先知道构造函数传入的参数是什么类型;调用Method的时候必须先知道成员函数传入的参数是什么类型;调用field的时候必须先知道成员变量的类型。但是在某些情况下,一个类中的成员属性的名字对外是不可见的,这时候我们只可以得出他的成员类型的数组field[],在不知道确切变量名的情况下不可以定位到确切的变量上。而他提供了对这个属性公开的读(get)、写(set)方法, 虽然方法名和方法操作的对象有时候有很大的关系,但是我们不能保证这个关系一定成立。而且这种只提供get、set方法,不提供变量名的情况还很常见,因此java就将这种抽象为javabean类,对javabean类的操作,通过get、set函数后面的名字就“拟”得出变量的名字。通过这个名字,我们可以确切的求出某个变量的值。而对javabean的操作是通过内省(introspector)来完成的。
二、内省访问JavaBean属性的两种方式:
通过PropertyDescriptor类操作Bean的属性
通过Introspector类获得Bean对象的 BeanInfo,然后通过 BeanInfo 来获取属性的描述器( PropertyDescriptor ),通过这个属性描述器就可以获取某个属性对应的 getter/setter 方法,然后通过反射机制来调用这些方法。
下面是实例:
package reflect;
import java.beans.BeanInfo;
import java.beans.Introspector;
import java.beans.MethodDescriptor;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
public class TestBean {
public static void main(String[] args) throws Exception {
testProDes();
testBeanInfo();
testBeanInfo1();
}
static void testBeanInfo() throws Exception{
Dog d = (Dog) Class.forName("reflect.Dog").newInstance();
S使用beanuitls框架操作bean的属性tring myname = "name";
BeanInfo beanInfo = Introspector.getBeanInfo(Dog.class);//beaninfo中存放bean的所有信息
PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
for(PropertyDescriptor pd : propertyDescriptors){
if(pd.getName().equals(myname)){
Method m1 = pd.getWriteMethod();
m1.invoke(d,"zhangsan");
Method m2 = pd.getReadMethod();
m2.invoke(d);
System.out.println(d);
break;
}
}
}
static void testProDes() throws Exception{
Dog d = (Dog) Class.forName("reflect.Dog").newInstance();
String myname = "name" ;//这里是根据函数名确定的string的变量名,而不是内部定义的私有属性名。
PropertyDescriptor pd = new PropertyDescriptor(myname, Dog.class);//propertyDescriptor直接得到属性描述符
Method m1 = pd.getWriteMethod();
m1.invoke(d,"zhangsan");
Method m2 = pd.getReadMethod();
m2.invoke(d);
System.out.println(d);
}
static void testBeanInfo1() throws Exception{
Dog d = (Dog) Class.forName("reflect.Dog").newInstance();
String myname = "name";
BeanInfo beanInfo = Introspector.getBeanInfo(Dog.class);//beaninfo中存放bean的所有信息
MethodDescriptor[] mds = beanInfo.getMethodDescriptors();
for(MethodDescriptor md : mds){
String str = md.getName();
if (str.matches("[gs]et.*")) {
System.out.println(str);//打印set/get开头的函数
}使用beanuitls框架操作bean的属性
}
}
}
class Dog {
private String n ;
private int a ;
public Dog(){
}
public Dog(String name,int age){
this.n = name;
this.a = age;
}
public String getName() {
return n;
}
public void setName(String name) {
this.n = name;
}
public int getAge() {
return a;
}
public void setAge(int age) {
this.a = age;
}
@Override
public String toString(){
return n + " : "+a;
}
}
在这里我们故意将Dog类中的get/set方法和属性名定义的不一样。但是通过内省,我们并不需要知道确切的内部属性的名字,通过get/set的名字,就可以实现对属性的设置和读取。
第一个函数直接用propertyDescriptor得出属性描述符,然后通过read方法求出get对应的method,write方法求出set对应的method,通过method的反射invoke函数实现对属性的操作。
第二种函数和第一个函数实现一样的功能,但是路径不一样,先用introspector静态方法得出beaninfo,beaninfo中包含了所有的关于类的信息,再通过beaninfo的方法得出propertyDescriptor,在按照第一种的实现方法实现对属性的操作。
第三个函数顺便测试了methoddescripor,将这个javabean类中的所有get/set的方法打印出来。
三、beanutils工具包操作bean
Sun公司的内省API过于繁琐,所以Apache组织结合很多实际开发中的应用场景开发了一套简单、易用的API操作Bean的属性——BeanUtils
Beanutils工具包的常用类:
BeanUtils
PropertyUtils
ConvertUtils.regsiter(Converter convert, Class clazz)自定义转换器
四、使用beanuitls框架操作bean的属性
1、创建Student类:
package cn.itcast.beanutils;
import java.util.Date;
//javabean (属性)
public class Student {
<span style="white-space:pre"> </span> private String name; //只要有get和set中的任意一个方法,那么就是属性,否则为字段
<span style="white-space:pre"> </span> private int age;
<span style="white-space:pre"> </span> private Date birthday;
<span style="white-space:pre"> </span>
<span style="white-space:pre"> </span> public String getName() {
<span style="white-space:pre"> </span> return name;
<span style="white-space:pre"> </span> }
<span style="white-space:pre"> </span> public void setName(String name) {
<span style="white-space:pre"> </span> this.name = name;
<span style="white-space:pre"> </span> }
<span style="white-space:pre"> </span> public int getAge() {
<span style="white-space:pre"> </span> return age;
<span style="white-space:pre"> </span> }
<span style="white-space:pre"> </span> public void setAge(int age) {
<span style="white-space:pre"> </span> this.age = age;
<span style="white-space:pre"> </span> }
<span style="white-space:pre"> </span> public Date getBirthday() {
<span style="white-space:pre"> </span> return birthday;
<span style="white-space:pre"> </span> }
<span style="white-space:pre"> </span> public void setBirthday(Date birthday) {
<span style="white-space:pre"> </span> this.birthday = birthday;
<span style="white-space:pre"> </span> }
<span style="white-space:pre"> </span> public String getX(){ //X也是属性,只要有get或者set方法就算
<span style="white-space:pre"> </span> return "X";
<span style="white-space:pre"> </span> }
//其实之中还有一个默认的getClass,所以一共有5个字段
}
2、使用beanuitls框架操作bean的属性
Person类:
package cn.itcast.beanutils;
import java.util.Date;
public class Person {
private String name;
private String password;
private int age;
private Date birthday;
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
public String getName() {
return name;
}
public String getPassword() {
return password;
}
public int getAge() {
return age;
}
public void setName(String name) {
this.name = name;
}
public void setPassword(String password) {
this.password = password;
}
public void setAge(int age) {
this.age = age;
}
}
操作Person代码:
package cn.itcast.beanutils;
import java.lang.reflect.InvocationTargetException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Locale;
import java.util.Map;
import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.beanutils.ConversionException;
import org.apache.commons.beanutils.ConvertUtils;
import org.apache.commons.beanutils.Converter;
import org.apache.commons.beanutils.locale.converters.DateLocaleConverter;
import org.junit.Test;
//使用beanUtils操纵bean的属性 ( 第三方)
public class Demo1 {
@Test
public void test1() throws Exception{
Person p=new Person();
BeanUtils.setProperty(p, "age", 456);
System.out.println(p.getAge());//456
}
@Test
public void test2() throws Exception{
String name="aaaa";
String age="123";
String password="pw";
Person p=new Person();
//支持8种基本类型自动转换
BeanUtils.setProperty(p, "name", name);
BeanUtils.setProperty(p, "age", age);
BeanUtils.setProperty(p, "password", password);
System.out.println(p.getName());//aaaa
System.out.println(p.getAge());//123
System.out.println(p.getPassword());//pw
}
@Test
public void test3() throws Exception{
String birthday="1983-12-1";
//为了让日期赋值到bean的birthday属性上,给beanUtils注册一个日期转换器
//ConvertUtils.register(converter, clazz);
ConvertUtils.register(new Converter(){
public Object convert(Class type, Object value) {
if(value==null) return null;
if(!(value instanceof String)){
throw new ConversionException("只支持String类型的转换");
}
String str=(String)value;
if(str.trim().equals("")) return null;
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd",Locale.US);
try {
return df.parse(str);
} catch (ParseException e) {
throw new RuntimeException(e);
}
}
}, Date.class);
Person p=new Person();
BeanUtils.setProperty(p, "birthday", birthday);
System.out.println(p.getBirthday());//pw
System.out.println("___"+BeanUtils.getProperty(p, "birthday"));
}
public void test5() throws Exception {
Map map=new HashMap();
map.put("name", "aaa");
map.put("password", "123");
map.put("brithday", "1980-09-09");
ConvertUtils.register(new DateLocaleConverter(), Date.class);
Person p=new Person();
//用map集合填充bean属性,map关键字和bean属性要一致
BeanUtils.populate(p, map);
}
}