设为首页 加入收藏

TOP

是时候丢掉BeanUtils了(一)
2023-08-06 07:49:56 】 浏览:80
Tags:时候丢 BeanUtils

前言

为了更好的进行开发和维护,我们都会对程序进行分层设计,例如常见的三层,四层,每层各司其职,相互配合。也随着分层,出现了VO,BO,PO,DTO,每层都会处理自己的数据对象,然后向上传递,这就避免不了经常要将一个对象的属性拷贝给另一个对象。

例如我有一个User对象和一个UserVO对象,要将User对象的10个属性赋值个UserVo的同名属性:
一种方式是手写,一个属性一个属性赋值,相信大家最开始学习时都是这么干的,这种方式就是太低效了。
在idea中可以安装插件帮我们快速生成set属性代码,虽然还是逐个属性赋值,但比一个个敲,效率提高了很多。

上面两种方式虽然最原始,做起来很麻烦,容易出错,但程序运行效率是最高的,现在仍有不少公司要求这么做,一是这样运行效率高,二是不需要引入其它的组件,避免出现其它问题。
但对于我们来说,这种操作要是多了,开发效率和代码可维护性都会受到影响,这种赋值属性代码很长,看起来很不舒服,所有有了下面几种方式。

bean copier

apache的BeanUtils,内部使用了反射,效率很低,在阿里java开发规范中命令禁止使用,这里就不过多讨论。

spring的BeanUtils,对apache BeanUtils做了优化,运行效率较高,可以使用。

BeanUtils.copyProperties(source, target);
BeanUtils.copyProperties(source, target, "id", "createTime"); //不拷贝指定的字段

cglib的BeanCopier,使用动态技术代替反射,在运行时生成一个子类,只有在第一次动态生成类时慢,后面基本就本接近原始的set,所以呀运行效率比上面两种要高很多。

BeanCopier beanCopier = BeanCopier.create(SourceData.class, TargetData.class, false);
beanCopier.copy(source, target, null);		

我们使用的是spring BeanUtils,至少出现过两次问题:
一次是拷贝一方的对象类型变了,从int变成long,source.id int 拷贝到 target.id long 结果是空,因为类型不匹配,BeanUtils不会拷贝。由于是使用反射,所以当时修改类型时,只修改了编译报错的地方,忘记这种方式,导致结果都是空,这也很难怪开发,这种方式太隐蔽了。同样如果属性重命名,也会得到一个空,并且只能在运行时发现。
另一次拷贝的时候会把所有属性都拷过去,漏掉忽略主键id,结果在插入的时候报了唯一索引冲突。我们的场景比较特殊,id,createTime,updateTime这三个字段是表必须有的,通常也是不能被拷贝的,如果每个地方都手写忽略,代码比较麻烦也容易忘记。

上面3种方式都非常简单,意味着功能非常有限,如果你有一些复杂场景的拷贝,它们就无法支持,例如深拷贝,拷贝一个List
另外一个最重要的点是,它们都是运行时的,这意味着你无法在编译时得到任何帮助,无法提前发现问题。
从标题可以看出我们本篇要讲的是另一个copier: mapstruct,接下来就看下它是解决我们问题的。

MapStruct

mapstruct是一个基于java注解处理器,用于生成类型安全且高性能的映射器。总结一下它有以下优点:
1.高性能。使用普通方法赋值,而非反射,mapstruct会在编译期间生成类,使用原生的set方法进行赋值,所以效率和手写set基本是一样的。
2.类型安全。mapstruct是编译时的,所以一旦有类型、名称等不匹配问题,就可以提前编译报错。
3.功能丰富。mapstruct的功能非常丰富,例如支持深拷贝,指定各种拷贝行为。
4.使用简单。你所需要做的就是定义接口和拷贝的行为,mapstruct会在编译期生成实现类。

示例
和学习其它组件一样,我们先用起来,准备两个类,SourceData,TargetData属性完全一样,其中TestData是另一个类。

public class SourceData {

	private String id;
	private String name;
	private TestData data;
        private Long createTime;

	public String getId() {
		return id;
	}
	public void setId(String id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public TestData getData() {
		return data;
	}
	public void setData(TestData data) {
		this.data = data;
	}
        public Long getCreateTime() {
		return createTime;
	}
	public void setCreateTime(Long createTime) {
		this.createTime = createTime;
	}
}

导入包

<dependency>
    <groupId>org.mapstruct</groupId>
    <artifactId>mapstruct</artifactId>
    <version>${org.mapstruct.version}</version>
</dependency>

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <version>3.8.1</version>
            <configuration>
                <source>1.8</source>
                <target>1.8</target>
                <annotationProcessorPaths>
                    <path>
                        <groupId>org.mapstruct</groupId>
                        <artifactId>mapstruct-processor</artifactId>
                        <version>${org.mapstruct.version}</version>
                    </path>
                </annotationProcessorPaths>
            </configuration>
        </plugin>
    &l
首页 上一页 1 2 3 下一页 尾页 1/3/3
】【打印繁体】【投稿】【收藏】 【推荐】【举报】【评论】 【关闭】 【返回顶部
上一篇quarkus依赖注入之二:bean的作用.. 下一篇在同事面前炫一把,用 Docker 搭..

最新文章

热门文章

Hot 文章

Python

C 语言

C++基础

大数据基础

linux编程基础

C/C++面试题目