A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 黄敏文 黑马帝   /  2011-8-19 09:45  /  3272 人查看  /  6 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

Person类有一个字段private Date birthday;  get和set方法就省略不写了
用BeanUtils操作属性时,如下
[code=java]
Person p = new Person("zhangsan", new Date());
BeanUtils.setProperty(p, "birthday.time", "new");
System.out.println(BeanUtils.getProperty(p, "birthday.time"));
[/code]
结果显示为0

而当把设置的语句改为下面的代码时,可以正常显示出结果
[code=java]
BeanUtils.setProperty(p, "birthday.time", "111");
[/code]

想问一下这是什么原因

评分

参与人数 1技术分 +1 收起 理由
wangfayin + 1

查看全部评分

6 个回复

倒序浏览
黑马网友  发表于 2011-8-19 13:12:07
沙发
把你的person类 发出来看看
回复 使用道具 举报
黑马网友  发表于 2011-8-19 20:07:58
藤椅
1. 概述

BeanUtils主要提供了对于JavaBean进行各种操作,

BeanUtils一共分4个包:

    * org.apache.commons.beanutils
    * org.apache.commons.beanutils.converters
    * org.apache.commons.beanutils.locale
    * org.apache.commons.beanutils.locale.converters

converters 就是专门处理不同传入的 object 该如何转换, locale 呢, 就是为了国际化的处理。其中上面两个是BeanUtils的默认实现,它没有针对本地化的任何处理,这个可以提高执行效率。

2. org.apache.commons.beanutils

2.1.BeanUtil

1、这个包主要提供用于操作JavaBean的工具类,Jakarta-Common-BeanUtils的主要功能都在这个包里实现。

public class Employee {

    public Address getAddress(String type);

    public void setAddress(String type, Address address);

    public Employee getSubordinate(int index);

    public void setSubordinate(int index, Employee subordinate);

    public String getFirstName();

    public void setFirstName(String firstName);

    public String getLastName();

    public void setLastName(String lastName);

}



2、BeanUtils可以直接get和set一个属性的值。它将property分成3种类型:

    * Simple——简单类型,如Stirng、Int……
    * Indexed——索引类型,如数组、arrayList……
    * Maped——这个不用说也该知道,就是指Map啦,比如HashMap……

访问不同类型的数据可以直接调用函数getProperty和setProperty。它们都只有2个参数,第一个是JavaBean对象,第二个是要操作的属性名。

Company c = new Company();

c.setName("Simple");

对于Simple类型,参数二直接是属性名即可

//Simple

    System.out.println(BeanUtils.getProperty(c, "name"));

对于Map类型,则需要以“属性名(key值)”的形式

//Map

    System.out.println(BeanUtils.getProperty(c, "address (A2)"));

    HashMap am = new HashMap();

    am.put("1","234-222-1222211");

    am.put("2","021-086-1232323");

    BeanUtils.setProperty(c,"telephone",am);

    System.out.println(BeanUtils.getProperty(c, "telephone (2)"));

对于Indexed,则为“属性名[索引值]”,注意这里对于ArrayList和数组都可以用一样的方式进行操作。

//index

    System.out.println(BeanUtils.getProperty(c, "otherInfo[2]"));

    BeanUtils.setProperty(c, "product[1]", "NOTES SERVER");

    System.out.println(BeanUtils.getProperty(c, "product[1]"));


当然这3种类也可以组合使用啦!

//nest

    System.out.println(BeanUtils.getProperty(c, "employee[1].name"));



3、此外,还有一个很重要的方法copyProperty,可以直接进行Bean之间的clone。

Company c2 = new Company();

BeanUtils.copyProperties(c2, c);

但是这种copy都是浅拷贝,复制后的2个Bean的同一个属性可能拥有同一个对象的ref,这个在使用时要小心,特别是对于属性为自定义类的情况。

4、最后还有populate,它用于将一个map的值填充到一个bean中

2.2.LazyDynaBean

它实现一个动态的Bean,可以直接往里面加入属性,作为一个JavaBean一样使用,也可以用上面的BeanUtils或get/set方法进行操作,而不用事先定义一个标准的JavaBean类啦:)

记得在J2ee设计模式中有一种Value Object的模式,用于在MVC各层之间传递数据,避免直接传递大业务对象引起的性能问题,为了避免在项目中出现很多Bean类,在书中提供了一个动态 Value Object的实现(通过扩展Map)。这里LazyDynaBean则可以作为一种更加成熟、稳定的实现来使用。

//这里使用LazyDynaMap,它是LazyBean的一个轻量级实现

      LazyDynaMap dynaBean1 = new LazyDynaMap();

      dynaBean1.set("foo", "bar");                  // simple

      dynaBean1.set("customer", "title", "Mr");        // mapped

      dynaBean1.set("address", 0, "address1");         // indexed

      System.out.println(dynaBean1.get("address",0));

      Map myMap = dynaBean1.getMap();           // retrieve the Map

      System.out.println(myMap.toString());

上面的例子可以看到,它可以在set时自动增加bean的property(既赋值的同时增加Bean中的property),同时也支持3中类型的property,并且LazyDynaMap还可以导出为map。

对于这个类还有两个重要的Field要注意:

returnnull——指定在get方法使用了一个没有定义过的property时,DynaBean的行为。

//取的字段的信息

      dynaBean1.setReturnNull(true);//设为ture。若Bean中没有此字段,返回null

     //默认为false。若Bean中没有此字段,自动增加一个:)

      System.out.println(dynaBean1.get("aaa"));//此时返回null

     Restricted——指定是否允许改变这个bean的property。

      //MutableDynaClass.setRestricted设为true后,字段不可再增删和修改.

     //默认为false,允许增删和修改

      dynaBean1.setRestricted(true);

     dynaBean1.set("test","error");//这里会出错!

通过设置这两个属性,可以防止意外修改DynaBean的property。在设计架构时,你可以在后台从数据表或xml文件自动产生 DynaBean,在传到控制层和表示层之前设置上述属性使其Bean结构不允许修改,如此就不可能无意中修改Bean包含的属性……这样既可以享用它的便利,有可以防止由此引入的错误可能,设计者实在深得偷懒的精髓啊!!!!!
3. 其他

3.1.BeanUtils和PropertyUtils

这两个类几乎有一摸一样的功能,唯一的区别是:BeanUtils在对Bean赋值是会进行类型转化。举例来说也就是在copyProperty时只要属性名相同,就算类型不同,BeanUtils也可以进行copy;而PropertyBean则可能会报错!!

针对上面的例子,新建一个Company2的类,其中代码与Company一样,只是将otherinfo从String[]改为String。

      Company c = init();

      Company2 c2 = new Company2();

      BeanUtils.copyProperties(c2,c);

//      PropertyUtils.copyProperties(c2,c); 这句会报错!!

      System.out.println(c2.getOtherInfo());

当然2个Bean之间的同名属性的类型必须是可以转化的,否则用BeanUtils一样会报错。

若实现了org.apache.commons.beanutils.Converter接口则可以自定义类型之间的转化。

由于不做类型转化,用PropertyUtils在速度上会有很大提高!

此外,不作类型转化还有一个好处,如下面的代码:

      //test data type convert

//BeanUtils返回的是String

//      ArrayList a1 = BeanUtils.getProperty(c,"product");

//取出后直接被转为String

      System.out.println("--" + BeanUtils.getProperty(c,"product"));     

//PropertyUtils返回的是Object

      ArrayList a = (ArrayList)PropertyUtils.getProperty(c,"product");

      System.out.println("--" + a.get(1));

用BeanUtils无法返回一个对象(除非自己写一个Converter),它会自动进行类型转化,然后返回String。对于想返回java类或自定义类的话,还是不要老它大驾转化了。

3.2.Utils类

所有的XXXUtils类都提供的是静态方法,可以直接调用,其主要实现都在相应的XXXUtilsBean中:

BeanUtils ——> BeanUtilsBean

ConvertUtils ——> ConvertUtilsBean

PropertyUtils ——> PropertyUtilsBean

其意思看类名也应该知道的差不多了,我就不再废话啦!当然你也可以直接调用那些XXXUtilsBean,功能都一样!


古之欲明明德於天下者,先治其國;欲治其國者,先齊其家;欲齊其家者,先修其身,欲修其身者,先正其心;欲正其心者,先誠其意;欲誠其意者,先致其知;致知在格物。物格而后知至,知至而后意誠,意誠而后心正,心正而而后身修,身修而后家齊,家齊而后國治,國治而后天下平。
回复 使用道具 举报
黑马网友  发表于 2011-8-20 08:26:57
板凳
BeanUtils.setProperty(p, "birthday.time", "[color=lime]new[/color]");
BeanUtils.setProperty(p, "birthday.time", "111");

可能是因为new是关键字
回复 使用道具 举报
黑马网友  发表于 2011-8-20 08:31:10
报纸
不好意思,不是这个 "new“ 的问题,刚测试过, 唉,又发了垃圾贴,抱歉
回复 使用道具 举报
黑马网友  发表于 2011-8-20 08:47:03
地板

回复 沙发 的帖子

[code]import java.util.Date;
public class Person {
        private String name;
        private Date birthday;               
       
        public Person() {
                super();
        }

        public Person(String name, Date birthday) {
                super();
                this.name = name;
                this.birthday = birthday;
        }

        public String getName() {
                return name;
        }

        public void setName(String name) {
                this.name = name;
        }

        public Date getBirthday() {
                return birthday;
        }

        public void setBirthday(Date birthday) {
                this.birthday = birthday;
        }
}[/code]
回复 使用道具 举报
黑马网友  发表于 2011-8-20 08:56:14
7#

回复 藤椅 的帖子

[quote]
访问不同类型的数据可以直接调用函数getProperty和setProperty。它们都只有2个参数,第一个是JavaBean对象,第二个是要操作的属性名。
[/quote]

我认为[quote]它们都只有2个参数[/quote]这句话说的有误,setProperty应该是有3个参数,第3个参数应该是为要操作JavaBean对象的属性赋的值
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马