对象属性复制常用方法包括:1. 手动getter/setter,适用于简单对象;2. Spring BeanUtils,便捷高效,适合Spring项目;3. MapStruct,编译期生成,性能高;4. 序列化实现深拷贝,完整隔离引用但性能低;5. Cloneable接口,需重写支持深拷贝。选择依据场景需求与性能权衡。

在 Java 中,将一个对象的属性复制到另一个对象是开发中常见的需求,比如在 DTO 转换、缓存处理或避免对象引用污染时。实现方式有多种,各有适用场景和优缺点。以下是几种主流的对象属性复制方法。
1. 手动 Getter/Setter 赋值
最直接的方式是通过调用源对象的 getter 方法获取属性值,再通过目标对象的 setter 方法设置。
- 适用于属性少、结构简单的对象
- 代码清晰,易于调试
- 但当属性较多时,代码冗长且易出错
示例:
User user = new User();
user.setName("张三");
user.setAge(25);
UserInfo userInfo = new UserInfo();
userInfo.setName(user.getName());
userInfo.setAge(user.getAge());
2. 使用 BeanUtils 工具类(Apache 或 Spring)
借助第三方工具类可以自动完成同名属性的拷贝,大幅减少样板代码。
立即学习“Java免费学习笔记(深入)”;
- Apache Commons BeanUtils:功能强大,但性能较差,且依赖较多
- Spring BeanUtils:轻量、高效,推荐在 Spring 项目中使用
- 仅支持 public 属性和标准 getter/setter
- 不同类型属性不会自动转换(除非使用 ConvertUtils)
Spring BeanUtils 示例:
User source = new User("李四", 30);
UserCopy target = new UserCopy();
org.springframework.beans.BeanUtils.copyProperties(source, target);
注意:copyProperties 按属性名匹配,忽略 null 值(默认行为),不递归复制复杂嵌套对象。
3. 使用 MapStruct 等编译期生成映射代码
MapStruct 是一个注解处理器,能在编译时生成类型安全的映射实现类,性能高且可定制。
- 编译期生成代码,运行时无反射开销
- 支持复杂映射、字段转换、嵌套对象、集合映射
- 需要配置接口和注解,学习成本略高
- 适合大型项目或对性能要求高的场景
示例:
@Mapper
public interface UserMapper {
UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);
UserCopy toUserCopy(User user);
}
调用:UserCopy copy = UserMapper.INSTANCE.toUserCopy(user);
4. 序列化反序列化实现深拷贝
通过对象序列化再反序列化的方式,实现真正的深拷贝,适用于需要完全隔离对象引用的场景。
- 能复制对象及其内部所有引用对象
- 要求类实现 Serializable 接口
- 性能较低,不适用于高频操作
- 静态变量和 transient 字段不会被复制
示例:
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(originalObject);
ObjectInputStream ois = new ObjectInputStream(
new ByteArrayInputStream(bos.toByteArray()));
Object cloned = ois.readObject();
5. 实现 Cloneable 接口(浅拷贝 / 深拷贝)
Java 提供了 clone() 方法,但默认是浅拷贝,需手动重写实现深拷贝。
- 浅拷贝:基本类型复制,引用类型仍共享
- 深拷贝:需在 clone() 中递归克隆引用对象
- 侵入性较强,每个类都要实现 Cloneable 并重写 clone()
- 容易遗漏字段或引用对象
示例:
public class User implements Cloneable {
private String name;
private Address address; // 引用类型
@Override
public User clone() {
try {
User cloned = (User) super.clone();
cloned.address = this.address.clone(); // 深拷贝引用对象
return cloned;
} catch (CloneNotSupportedException e) {
throw new AssertionError();
}
}
}
基本上就这些常用方式。选择哪种取决于你的具体需求:简单场景用手动赋值或 Spring BeanUtils;高性能要求用 MapStruct;必须深拷贝考虑序列化或自定义 clone。每种方法都有适用边界,关键是理解其原理和限制。











