java基础之拷贝、单例

一、拷贝

常见的拷贝有引用拷贝、浅拷贝、深拷贝

1.引用拷贝

就是我们常见的=赋值

实现方式 : Person p2=p1;

特点:两个对象名指向同一个内存空间,任何对象对其属性进行修改,都会直接影响到另一个变量,属于对同一个对象的操作,所以算不上真正意义上的赋值,如图:

代码例子:

java 复制代码
public class CopyTest {
    public static void main(String[] args) {
        Address addr = new Address("北京");
        User user1 = new User(20, addr);
        User user2 = user1; 

        // 修改user2,user1也会变
        user2.age = 30;
        user2.address.city = "上海";

        System.out.println(user1.age); // 30(被修改)
        System.out.println(user1.address.city); // 上海(被修改)
    }
}

2.浅拷贝

如果对象里面属性全是基本数据类型,那么浅拷贝实际上是创建了两个空间,一个对象对其属性进行修改,不会影响另一个对象,但是!如果对象属性有引用数据类型,那么两个对象的引用数据类型的变量就会指向同一个地址,没有实现完全独立

也就是说浅拷贝会直接赋值内部引用数据类型变量的引用地址,拷贝对象和原对象的属性共用一个对象(引用数据)

实现方法:类实现 Cloneable 接口;重写 clone() 调用 super.clone()

代码例子:

java 复制代码
public static void main(String[] args) throws CloneNotSupportedException {
    Address addr = new Address("北京");
    User user1 = new User(20, addr);

    // ========== 浅拷贝 ==========
    User user2 = (User) user1.clone();

    // 1. 修改基本类型:互不影响
    user2.age = 30;
    System.out.println(user1.age); // 20(独立)

    // 2. 修改引用类型:互相影响!
    user2.address.city = "上海";
    System.out.println(user1.address.city); // 上海(共享)
}

3.深拷贝

不管是引用数据类型还是基本数据类型,都会生成一份新的,对内部数据进行修改不会互相影响;

实现方法:重写 clone () + 递归拷贝引用对象;序列化 / 反序列化(最简单通用)

代实现方法一代码例子:

java 复制代码
//手动递归clone
@Override
protected Object clone() throws CloneNotSupportedException {
    // 1. 先浅拷贝当前对象
    User user = (User) super.clone();
    // 2. 手动拷贝引用类型成员 → 变成深拷贝
    user.address = (Address) this.address.clone();
    return user;
}

//测试
public static void main(String[] args) throws CloneNotSupportedException {
    Address addr = new Address("北京");
    User user1 = new User(20, addr);

    // ========== 深拷贝 ==========
    User user2 = (User) user1.clone();

    // 修改引用类型:互不影响!
    user2.address.city = "上海";
    System.out.println(user1.address.city); // 北京(完全独立)
}

二、单例

1.定义

是一种设计模式,该模式保证在整个应用中,一个类只能有唯一一个实例对象 ,全局共享这一个对象,私有化构造方法,禁止外部 new。这个类被称为单例类

2.实现思路

在一般类里面,如果想要限制只能创建一个对象,那么我们会想到用private来修饰对象,但是其他类想要调用的时候获取不到,那么我们使用get方法来获取,而get方法需要其他类来创建对象才能调用该类的方法,那么我们会想到使用static来修饰这个get方法,让它属于类,那么我就可以直接用类名来调用,与之对应的private修饰的对象也要变成static属于类;经过以上步骤就能粗略的实现只允许创建一个对象的功能

3.实现方法

(1)饿汉式

优点:实现简单,类加载时直接创建实例,多线程下安全

缺点:实例对象在类加载时就初始化,不管是否使用,都会创建,造成空间浪费

(2)懒汉式-不安全版本

优点: 在饿汉式的基础上添加了if判断,当使用对象,对象没创建的时候才创建,避免了不用也创建的缺点

缺点:在多线程下,如果多个线程同时进入判断语句,可能会创建多个实例,违反了单例原则

(3)懒汉式-安全版本

在不安全的版本中添加一道锁:synchronized

优点:加锁保证安全

缺点:每次获取都加锁,并发性能差

java 复制代码
public static synchronized Singleton getInstance(){
    if(instance == null){
        instance = new Singleton();
    }
    return instance;
}

(4)双重检查锁

优点:延迟加载 + 线程安全 + 效率高,必须加 volatile 禁止指令重排

(5)匿名内部类

优点:利用类加载机制保证线程安全,实现了懒加载、无锁、效率高。

java 复制代码
public class Singleton {
    private Singleton(){}
    // 静态内部类
    private static class Inner{
        private static final Singleton instance = new Singleton();
    }
    public static Singleton getInstance(){
        return Inner.instance;
    }
}

4.应用场景

资源共享的情况下,避免由于资源操作时导致的性能或损耗等。如上述中的日志文件,应用配置。

控制资源的情况下,方便资源之间的互相通信。如线程池等。

相关推荐
浮尘笔记2 小时前
Java Snowy框架CI/CD云效自动化部署流程
java·运维·服务器·阿里云·ci/cd·自动化
一直不明飞行9 小时前
Java的equals(),hashCode()应该在什么时候重写
java·开发语言·jvm
REDcker9 小时前
有限状态机与状态模式详解 FSM建模Java状态模式与C++表驱动模板实践
java·c++·状态模式
你的保护色9 小时前
【无标题】
java·服务器·网络
basketball61610 小时前
C++ 构造函数完全指南:从入门到进阶
java·开发语言·c++
淘矿人10 小时前
Claude辅助DevOps实践
java·大数据·运维·人工智能·算法·bug·devops
小江的记录本10 小时前
【Java基础】泛型:泛型擦除、通配符、上下界限定(附《思维导图》+《面试高频考点清单》)
java·数据结构·后端·mysql·spring·面试·职场和发展
来恩100310 小时前
请求转发与响应重定向的使用
java
@杰克成11 小时前
Java学习30
java·开发语言·学习
次元工程师!11 小时前
LangFlow开发(三)—Bundles组件架构设计(3W+字详细讲解)
java·前端·python·低代码·langflow