利用cglib的BeanCopier用原型模式以及享元模式完成对象的拷贝

2023-11-17

实际上对象拷贝的工具有很多种,比如apache BeanUtils、apache PropertyUtils、spring BeanUtils。在一些业务代码中现在经常看到的都是spring BeanUtils来进行对象拷贝。大部分情况下来说已经足够了,但如果居于性能考虑,以上几种工具都是利用反射的原理来完成的,性能相比cglib beanCopier利用动态代理实现稍差一筹,这里不去对比几种工具的性能,只展示BeanCopier如何进行使用。

一、非cglib下的对象拷贝方式

1、原型模式就是从一个已存在的对象创建出另一个对象

在实际业务开发中我们经常需要完成对象从DO到DTO或者VO的拷贝,大部分情况下,DTO/VO的属性都会与DO中的属性保持一致,如果每次都使用set方法去设置值就太麻烦了。需要通过一个通用的方法完成对象拷贝,如可以覆盖Object类的clone方法完成克隆操作

package com.zcj.javabase.designpattern.prototype;

/**
 * 学生DO
 * @author zcj
 * @date 2021/3/24 06:16
 */
public class StudentDO {

    private Long id;

    private String name;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

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

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

2、覆盖父类clone的局限性

使用覆盖父类的clone返回的是一个Object对象,使用时可强转一个新的StudentDO对象,但我们常常需要做的是从StudentDO拷贝一个StudentVO出来。此时就需要我们自己实现一个新的对象拷贝的方法,可传入一个类型参数,返回一个对应类型的对象出去。

public <T> T clone(Class<T> clazz) {
    T clazzInstance = null;
    try {
        clazzInstance = clazz.newInstance();
        Method setId = clazz.getMethod("setId", Long.class);
        setId.invoke(clazzInstance, this.id);

        Method setName = clazz.getMethod("setName", String.class);
        setName.invoke(clazzInstance, this.name);
    } catch (Exception e) {
        e.printStackTrace();
    } 

    return clazzInstance;
}

这样的做法是不通用,并且属性非常多的时候,就会特别的麻烦,可以使用BeanUtils.copyProperties替换掉原来的一大堆getMethod和invoke,如:

public <T> T clone(Class<T> clazz) {
    T clazzInstance = null;
    try {
        clazzInstance = clazz.newInstance();
        BeanUtils.copyProperties(this, clazzInstance);
    } catch (Exception e) {
        e.printStackTrace();
    }

    return clazzInstance;
}

二、cglib的BeanCopier的的方式实现对象拷贝

1、引入对象的依赖(正常项目中spring会有cglib的包,可直接使用spring的BeanCopier类,而不用单独的引入以下依赖)

<dependency>
    <groupId>asm</groupId>
    <artifactId>asm-all</artifactId>
    <version>3.3.1</version>
</dependency>

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib-nodep</artifactId>
    <version>3.3.0</version>
</dependency>

2、增加工具类(工具类中实现了享元模式)

享元模式解决的是当我们需要多次使用某个对象时,只需要创建一次,然后缓存起来,后续再使用的时候直接获取。而不用每次使用都进行创建,可通过Map进行缓存起来。这里的工具类就是使用Map把生成的BeanCopier缓存起来

package com.zcj.javabase.designpattern.flyweight;

import org.springframework.cglib.beans.BeanCopier;
import java.util.HashMap;
import java.util.Map;

/**
 * 对象拷贝工具类
 * @author zcj
 * @since 2021/2/24
 */
public class BeanCopierUtils {

    /**
     * 缓存beanCopier对象
     */
    private static final Map<String, BeanCopier> BEAN_COPIER_MAP = new HashMap<>();

    /**
     * 属性拷贝
     * @param source 源对象
     * @param target 目标对象
     */
    public static void copyProperties(Object source, Object target) {
        String beanKey = generateKey(source.getClass(), target.getClass());
        BeanCopier copier = BEAN_COPIER_MAP.get(beanKey);
        if (copier == null) {
            synchronized (BeanCopierUtils.class) {
                if (!BEAN_COPIER_MAP.containsKey(beanKey)) {
                    copier = BeanCopier.create(source.getClass(), target.getClass(), false);
                    BEAN_COPIER_MAP.put(beanKey, copier);
                }  else {
                    copier = BEAN_COPIER_MAP.get(beanKey);
                }
            }
        }
        copier.copy(source, target, null);
    }

    private static String generateKey(Class<?> sourceClass, Class<?> targetClass) {
        return sourceClass.toString() + targetClass.toString();
    }
}  

3、修改原来的clone方法

public <T> T clone(Class<T> clazz) {
    T clazzInstance = null;
    try {
        clazzInstance = clazz.newInstance();
        BeanCopierUtils.copyProperties(this, clazzInstance);
    } catch (Exception e) {
        e.printStackTrace();
    }

    return clazzInstance;
}

4、测试

package com.zcj.javabase.designpattern.prototype;

/**
 * 学生VO
 * @author zcj
 * @date 2021/3/24 06:16
 */
public class StudentVO {

    private Long id;

    private String name;

    private String address;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

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

    public String getAddress() {
        return address;
    }
    public void setAddress(String address) {
        this.address = address;
    }
}
    public static void main(String[] args) {

        StudentDO studentDO = new StudentDO();
        studentDO.setId(100L);
        studentDO.setName("zcj");

        StudentVO clone = studentDO.clone(StudentVO.class);
        System.out.println(JSONObject.toJSONString(clone));
    }

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

利用cglib的BeanCopier用原型模式以及享元模式完成对象的拷贝 的相关文章

随机推荐

  • SQL语句的MINUS,INTERSECT和UNION ALL

    SQL语句中的三个关键字 MINUS 减去 INTERSECT 交集 和UNION ALL 并集 关于集合的概念 中学都应该学过 就不多说了 这三个关键字主要是对数据库的查询结果进行操作 正如其中文含义一样 两个查询 MINUS是从第一个查
  • vue实现下载文件和图片功能

    vue实现图片或文件下载功能 今天一个需求就是实现图片下载功能 刚开始以为很简单没有什么逻辑可写 就以为调用后端接口就可以了 调用之后发现有问 题 看来还是没有想象的那么简单 1 要自己创建一个a标签 以下就是下载功能的实现 这里是调用接口
  • 2.4 HTTP请求方法

    在客户端向服务器端发送请求时 需要确定使用的请求方法 请求方法表明对URL指定资源的操作方式 服务器会根据不同的请求方法进行不同的响应 在HTTP 1 1中 共定义了8种请求方法 具体如下 GET 请求指定的内容并返回 POST 向指定资源
  • 分布式缓存的切片模式-hash一致性切片

    文章目录 一 为什么使用缓存 二 为什么使用分布式 三 使用什么模式 四 常规切片模式的弊端 五 更加犀利的切片模式 hash一致性切片 六 不完美的数据倾斜以及解决方案 6 1 数据倾斜 6 2 解决办法 一 为什么使用缓存 当前 我们通
  • 基于stm32的keil调试

    目录 基于stm32的keil调试 前言 实验目的 问题 debug 定位问题 解决 总结 基于stm32的keil调试 本文目标 基于stm32的keil调试 按照本文的描述 应该可以跑通实验并举一反三 先决条件 装有编译和集成的开发环境
  • 06_个人注释版本(01版本)GTK播放器__基于Linux系统下的mplayer播放器

    include
  • 计算机视觉结合深度学习项目-智能停车场空车位实时识别

    欢迎来到本博客 本次博客内容将继续讲解关于OpenCV的相关知识 作者简介 目前计算机研究生在读 主要研究方向是人工智能和群智能算法方向 目前熟悉python网页爬虫 机器学习 计算机视觉 OpenCV 群智能算法 然后正在学习深度学习的相
  • Android框架源码解析之(五)Retrofit

    源码地址 https github com square retrofit Retrofit源码结构 可以看出Retrofit是使用idea maven依赖编写的Java工程 并不是一个Android 工程 Retrofit的简单使用 1
  • 学习docker之路(三)

    目录 一 docker容器内操作 二 网络 一 docker容器内操作 1 将容器导出为归档 docker export 容器名称 o 归档包名称 root localhost docker export 05 o mynginx v2 t
  • leetcode_165. Compare Version Numbers 比较版本大小

    题目 Compare two version numbers version1 and version2 If version1 gt version2 return 1 if version1 lt version2 return 1 o
  • git简单使用与安装(小白01)(还看不懂我下个版本给图片)

    一 1 下载git 在官网上下载 下载成功后 闭着眼睛狂点下一步 然后就成功了 嘿嘿 二 1 随便创建一个新的文件夹 打开文件夹点这里 下图 然后输入cmd 然后在按回车 cmd中输入 git init 生成git文件 该文件是隐藏状态 2
  • redis06_ redis的订阅发布模式(redis做MQ中间件)、持久化(rdb,aof)、事务

    一 redis订阅发布模式 1 1 简介 redis 可以做消息中间件 MQ message queue 通常通过订阅发布模式来实现 消息订阅发布模式 还可以基本数据类型Lists实现 点到点模式 可以使用lpush lpop 实现消息先进
  • linux中I/O流中的全缓冲、行缓冲和无缓冲,简明实例演示

    说到缓冲 缓存之类的术语 通常都会和执行效率联系到一起 在标准I O库中提供缓冲的主要目的就是减少系统函数read和write的调用 从而能够减少系统CPU时间 标准I O库的缓冲主要分为3种 全缓冲 行缓冲和不缓冲 笔者就3种缓冲写了一些
  • 时钟电路设计概述 - 数字电路设计

    时钟电路设计概述 数字电路设计 2010 09 10 阅 转 分享 本文一般性地讲解了数字电路设计中的 时钟电路设计 包括有源 晶振
  • Unity3D教程笔记——unity初始02

    史上最全Unity3D教程 哔哩哔哩 bilibilihttps www bilibili com video BV12s411g7gU p 11 vd source f38a8a7e90133354051c463eb03a3b4f 这里是
  • 键盘输入流的缓冲效应 I/O流 缓冲I/O

    引子 有时候 你会发现 你的程序老是读不进某个值 但是反反复复看代码 又感觉自己代码真的没有问题 比如说下面这个代码 include stdio h int main int num char ch scanf d num scanf c
  • 【08】频率响应_详细数学推导G(jw)_滤波器

    本部分为B站up主DR CAN系列课程动态系统的建模与分析笔记 已获得up主授权 课程视频链接为 动态系统的建模与分析 8 频率响应 详细数学推导 G jw 滤波器 哔哩哔哩 bilibili
  • Selenium中get_attribute、get_property和text详解

    Selenium中WebElement类中get attrubute get propertry和text详解 get attribute get property和text的详解 特点如下 get property self name g
  • 2D/3D人体姿态估计 (2D/3D Human Pose Estimation)

    1 基本概念 算法改进入口 网络设计 特征流 损失函数 数据集的重要性 只要有一个好的 针对性的数据集 问题都可以解决 过集成新一代AutoML技术 可降低算法试错成本 人体姿态估计 Human Pose Estimation 是指图像或视
  • 利用cglib的BeanCopier用原型模式以及享元模式完成对象的拷贝

    实际上对象拷贝的工具有很多种 比如apache BeanUtils apache PropertyUtils spring BeanUtils 在一些业务代码中现在经常看到的都是spring BeanUtils来进行对象拷贝 大部分情况下来