.当把对象放到map中以后,修改对象内容,一样会影响对象的属性的值
[java] view plaincopy
import java.util.HashMap;
import java.util.Map;
public class TTTT {
public static void main(String[] args) {
Person p1 = new Person("zhangsan", 10);
Map<String, Person> map = new HashMap<String, Person>();
map.put("test", p1);
p1.setName("lisi");
p1.setAge(11);
System.out.println(map.get("test"));
}
}
class Person{
private String name;
private int age;
public Person(){
}
public Person(String name, int age){
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String toString(){
return name + ":" + age;
}
}
输出结果:
lisi:11
这个结果是因为map中和p1始终指向同一个对象,所以修改值以后会互相影响
为了处理这种互相影响的情况,在从map或者对象后,我们通常不直接处理对象,而是通过net.sf.cglib.beans.BeanCopier复制一个对象以后再进行操作,以免影响了map的对象原型。
[java] view plaincopy
import java.util.HashMap;
import java.util.Map;
import net.sf.cglib.beans.BeanCopier;
public class TTTT {
public static void main(String[] args) {
Person p1 = new Person("zhangsan", 10);
Map<String, Person> map = new HashMap<String, Person>();
map.put("test", p1);
p1.setName("lisi");
p1.setAge(11);
BeanCopier copier = BeanCopier.create(Person.class, Person.class, false);
Person p2 = new Person();
copier.copy(p1, p2, null);
System.out.println(p2);
p2.setName("p2's name");
p2.setAge(200);
System.out.println(p2);
System.out.println(map.get("test"));
}
}
class Person{
private String name;
private int age;
public Person(){
}
public Person(String name, int age){
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String toString(){
return name + ":" + age;
}
}
上面的代码通过拷贝,实现了修改p2而不影响,map中的对象,其实复制对象还有其他很多方法,比如自己手动set,比较麻烦、使用PropertyUtils或者使用BeanUtils。
从网上查的资料显示:
BeanCopier的性能是PropertyUtils (apache-common)的504倍。
PropertyUtils的性能是BeanUtils(apache-common)的1.71倍
可见,对于对象的拷贝,应尽量使用cglib的BeanCopier.
注意:BeanCopier在get方法和set方法不严格匹配的时候会出问题.
[java] view plaincopy
import java.util.HashMap;
import java.util.Map;
import net.sf.cglib.beans.BeanCopier;
public class TTTT {
public static void main(String[] args) {
Person p1 = new Person("zhangsan", 10);
BeanCopier copier = BeanCopier.create(Person.class, Person.class, false);
Person p2 = new Person();
copier.copy(p1, p2, null);
System.out.println("p1:" + p1);
System.out.println("p2:" + p2);
}
}
class Person{
private String name;
private int age;
public Person(){
}
public Person(String name, int age){
this.name = name;
}
public int getAge() {
return age + 1;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name + "T";
}
public void setName(String name) {
this.name = name;
}
public String toString(){
return name + ":" + age;
}
}
输出结果:
p1:zhangsan:0
p2:zhangsanT:1
可见,BeanCopier在复制的时候使用get和set方法了。如果get方法不严格,会造成复制的对象属性值不正确。