百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Java中深拷贝、浅拷贝、赋值的区别?如何自己写个深拷贝工具类?

zhezhongyun 2025-07-24 23:18 5 浏览

#头条创作挑战赛#

为什么要了解JavaBean深拷贝的机制

目前大家都在常用已经封装好的工具类,很少去深入了解多种实现方式;

此处提出一种简单的实现原理,也是巩固Java中关于反射的用法和知识;

目的:可以拷贝所有子类以及List中的范型(反射实现)


Java中深拷贝、浅拷贝、赋值的区别

在日常开发中,拷贝一直分为深拷贝和浅拷贝,随着我们开发能力的提升,我们应该逐步关注一些常见但是没有真正理解的东西!

本篇的目的就是简单说明并且留有余地,让我们可以了解的同时也可以有时间充分思考。

我们的问题:

1、深拷贝和浅拷贝,他们究竟是什么意思?

2、为什么要这样区分?

3、赋值和浅拷贝是一样的吗?


概念:

浅拷贝 :只复制指向某个对象的指针,而不复制对象本身,相当于是新建了一个对象,该对象复制了原对象的指针,新旧对象还是共用一个内存块。
深拷贝:是新建一个一模一样的对象,该对象与原对象不共享内存,修改新对象也不会影响原对象。
赋值:当我们把一个对象赋值给一个变量的时候,赋值的其实是该对象的栈内存地址而不是堆内存数据

概念是不是有一些抽象?看下面表格快速了解。

来用一张表格,快速区分


是否指向同一“堆内存”

拷贝后修改数据是否影响-基本数据类型

拷贝后修改数据是否影响-引用数据类型

深拷贝

影响

影响

浅拷贝

影响

影响

赋值

影响

影响

简单说明一下引用数据类型

引用数据类型:

引用数据类型建立在基本数据类型的基础上,包括数组、类和接口。引用数据类型是由用户自定义,用来限制其他数据的类型。


OK,到这里基本已经了解好了,下面该动手了!

如何自己通过反射实现一个JavaBean的深拷贝工具类

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * @author N_Xiang
 * @describe JavaBean工具类
 * @time 2021/11/10 11:21 下午
 */
public class KtBeanUtils {
    private static Map<String,Method> cachMap = new ConcurrentHashMap<>();
    /**
     * 仅仅拷贝本身
     * @param source
     * @param target
     */
    public static void copy(Object source ,Object target){
        toJavaBean(source,target,false);
    }
    /**
     * 仅仅拷贝本身
     * @param source
     * @param target
     * @param type 类型 true拷贝父类 false正常拷贝
     */
    public static void copy(Object source ,Object target,Boolean type){
        toJavaBean(source,target,type);
    }
    /**
     * Bean拷贝
     * @param source 元数据
     * @param target 目标数据
     * @param type 类型 true拷贝父类 false正常拷贝
     */
    private static void toJavaBean(Object source ,Object target,Boolean type){
				//Map<String, String> map = KtAliasPars.ParsAlias(source);
        Class<?> cTarget = target.getClass();
        Class<?> cSource = source.getClass();
				//获得某个类的所有声明的字段
        Field[] fields = cTarget.getDeclaredFields();
        for (Field field : fields) {
            String fieldName = field.getName();
						// 获得属性的首字母并转换为大写
            String firstLetter = fieldName.substring(0, 1).toUpperCase().concat(fieldName.substring(1));
            String setMethodName = "set" + firstLetter;
            String getMethodName = "get" + firstLetter;
          
            try {
								//获取源对象的值
                Method cmethod = cachMap.getOrDefault(cSource.getName()+"@"+getMethodName,cSource.getMethod(getMethodName));
                Object cinvoke = cmethod.invoke(source);
                cachMap.put(cSource.getName()+"@"+getMethodName,cmethod);
                if (!isWrapClass(field.getType())){
                    if (field.getType().equals(List.class)){
                        List<Object> o = new ArrayList<>();
                        copyListProperties((List<Object>)cinvoke, o,getT(field));
                        cinvoke = o;
                    }else {
                        Object o = field.getType().newInstance();
                        toJavaBean(cinvoke,o,false);
                        cinvoke = o;
                    }
                }
								//调用方法写入对象
                Method setMethod = cachMap.getOrDefault(cTarget.getName()+"@"+setMethodName,cTarget.getMethod(setMethodName,
                        new Class[] { field.getType() }));
                cachMap.put(cTarget.getName()+"@"+setMethodName,setMethod);
                setMethod.invoke(target, new Object[] { cinvoke });
            } catch (Exception e) {
								// e.printStackTrace();
            }
        }
    }
    private static boolean isWrapClass(Class clz) {
        if (clz.equals(String.class)){
            return true;
        }
        try {
            return ((Class) clz.getField("TYPE").get(null)).isPrimitive();
        } catch (Exception e) {
            return false;
        }
    }
    /**
     * 带回调函数的集合数据的拷贝(可自定义字段拷贝规则)
     * @param sources : 数据源类
     * @param target : 目标类::new(eg: UserVO::new)
     * @return
     */
    public static void copyListProperties(List<Object> sources, List<Object> target,Class<?> targetType) {
        for (Object source : sources) {
            Object o = null;
            try {
                o = targetType.newInstance();
            } catch (InstantiationException e) {
                e.printStackTrace();
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
            toJavaBean(source, o,false);
            target.add(o);
        }
    }
    /**
     * 获取list的范性
     * @param f list
     * @return
     */
    private static Class<?> getT(Field f){
        Type genericType = f.getGenericType();
				// 如果是泛型参数的类型
        if(genericType instanceof ParameterizedType){
            ParameterizedType pt = (ParameterizedType) genericType;
						//得到泛型里的class类型对象
            return (Class<?>)pt.getActualTypeArguments()[0];
        }
        return null;
    }
}

Lintge-每天一个小技巧,了解一些好玩的知识

相关推荐

Trump Pushes for 15%-20% Minimum Tariffs on EU, Sticks to Auto Duties: Report

TMTPOST--U.S.PresidentDonaldTrumpistakingatougherstanceduringtradenegotiationswiththe...

Note-15.使用A4988控制步进电机(dvp15mc11t控制步进电机)

如果需要控制一堆步进电机,那么只使用一个Arduino来控制,就会占用大量的处理时间,而无法处理其他事情,除非使用一个独立的专用步进电机驱动器:A4988。A4988只需两个引脚就可以控制双极步进电机...

Negotiated settlement, not sanctions, the right way to end the Ukraine crisis

Negotiatedsettlement,notsanctions,therightwaytoendtheUkrainecrisis:ChinaDailyeditorial...

U.S. Services May be Added to EU&#39;s Retaliatory Target List as More Members Seek Powerful Trade Tool If Talks Fail

TMTPOST--AmericanservicescouldbeaddedtotheEuropeanUnion’sretaliatorytargetlist,highlig...

S7-1200伺服指令运动指令(s71200伺服位置控制实例)

1.MC_Halt指令名称:停止轴运行指令功能:停止所有运动并以组态的减速度停止轴。使用技巧:常用MC_Halt指令来停止通过MC_MoveVelocity指令触发的轴的运行。『注意』部分输入/输出管...

Deepseek太强了!等了10年的Excel模糊下拉,竟然5分钟就搞定了

今天跟大家分享下我们如何通过Deepseek来编写VBA代码,制作模糊搜索的下拉菜单,这个等来十来年的功能,用Deepseek竟然几分钟就搞定了,不得不感叹AI工具的强大,我们以后能干的过AI吗,这真...

7.Unity物理关节(unity物理骨骼)

7.物理关节Unity的物理关节组件将刚体连接到另一个刚体或空间中的固定点。施加使刚体移动的力,关节限制可以限制移动。关节赋予刚体一定的自由度,从而使这些刚体具有不同的运动。Unity提供的物理关...

西门子111报文对应FB284引脚(西门子111报文详解)

西门子FB284是基于111报文的功能块,使用FB284比较方便,直接调用就可以控制V90伺服的定位控制。下面是整理的FB284对应的111报文。便于理解FB284功能块,更方面应用在实际项目中。11...

数据质量动态探查及相关前端实现(数据质量改进实践指南)

需求背景数据探查上线之前,数据验证都是通过写SQL方式进行查询的,从编写SQL,到解析运行出结果,不仅时间长,还会反复消耗计算资源,探查上线后,只需要一次探查,就可以得到整张表的探查报告,但后续...

阿里面试官:你连个排序算法都讲不明白?出门右拐吧

排序算法一表总览其他注意事项:计数排序中,kkk是整数的范围稳定性是指,序列中相同的数是否有可能交换顺序,例如序列中有两个8,顺序为888和8′8^{'}8′,如果在排序完之后,顺序有...

直流电机速度、位置双环控制简明教程

速度、位置的双环控制是我们在电机的控制系统中常用的方法,很实用。下面让我们来看一下内部实现的原理。1.速度闭环控制我们一般在速度闭环控制系统里面,使用增量式PI控制。而在我们的微处理器里面,因为控制器...

纳米机器人的精准定位与导航(纳米机器人怎么控制位置)

纳米机器人的精准定位与导航涉及多学科交叉技术,其“源码”(控制逻辑与算法)需结合硬件特性、环境感知和执行器设计。以下从控制原理、算法逻辑、关键技术实现等方面提供概念性思路(非实际可运行代码),供技术探...

C语言进阶教程:多级指针的应用(c语言一级指针和二级指针)

在C语言中,指针可以指向变量的地址。多级指针(PointerstoPointers或MultilevelPointers)则是指向另一个指针地址的指针。这种概念可以扩展到任意级别(二级指针、...

被 Trac 的文件整理能力圈粉了!这 “香” 气挡不住

前阵子整理电脑文件夹时,我遇到了个麻烦:上千个文档和照片,要是手动一个个重命名、分类,不仅手得点到发麻,估计还得耗上一上午。于是我就琢磨着,能不能找个办法实现一键批量重命名。我先在网上搜了些批量改名软...

Enhancer-轻量化的字节码增强组件包

一、问题描述当我们的业务发展到一定阶段的时候,系统的复杂度往往会非常高,不再是一个简单的单体应用所能够承载的,随之而来的是系统架构的不断升级与演变。一般对于大型的ToC的互联网企业来说,整个系统都是...