关于后端:设计模式之单例和原型

33次阅读

共计 4848 个字符,预计需要花费 13 分钟才能阅读完成。

明天这篇文章咱们来学习创立型设计模式的另外两个孪生兄弟,单例和原型,其中原型设计模式中咱们深刻到 JVM 的内存模型,最初顺便谈谈 Java 中的值传递和援用传递。

上篇文章老王买产品 咱们从最原始的根本实现办法,到简略(动态)工厂,而后应用工厂办法设计模式进行革新,最初思考产品会产生变体,咱们又扩大到了形象工厂。

设计模式所有的相干代码均已上传到码云 读者能够自行下载学习测试。

一、引出问题

明天老王又来了,还是想买咱们的产品,明天老王上老就提出来一个要求,当他购买产品的时候,每次都要从货架上给他拿雷同的一个。

如果用传统实现形式,当老王拿到产品当前,间接和上一个比对一下就行了,如果不统一老王就还回来。

但通过咱们查阅软件的七大设计准则,这很显著违反了依赖倒置准则,为了防止耦合和让代码更易于保护,老王是不能依赖具体产品的。

二、单例

咱们就须要将产品比对在创立产品的时候进行判断,老王就只管拿。

老王来之前应该还有两种状况,一种就是老王还没来,产品就筹备好了,也即饿汉式。第二种就是老王什么时候来,什么时候给他筹备产品,也即懒汉式。

咱们看具体的实现代码:

懒汉式:

/**
 * 懒汉式
 * @author tcy
 * @Date 29-07-2022
 */
public class LazySingletonProduct {

    private static volatile LazySingletonProduct instance=null;

    private LazySingletonProduct(){}

    public static synchronized LazySingletonProduct getInstance(){if (instance==null){instance=new LazySingletonProduct();

        }
        return instance;
    }

饿汉式:

/**
 * 饿汉式
 * @author tcy
 * @Date 29-07-2022
 */
public class HungrySingletonProduct {private static volatile HungrySingletonProduct instance=new HungrySingletonProduct();

    private HungrySingletonProduct(){};

    public static synchronized HungrySingletonProduct getInstance(){if (instance==null){instance=new HungrySingletonProduct();
        }
        return instance;
    }
}

老王类:

/**
 * @author tcy
 * @Date 29-07-2022
 */
public class Client {public static void main(String[] args) {HungrySingletonProduct instance1 = HungrySingletonProduct.getInstance();
        HungrySingletonProduct instance2 = HungrySingletonProduct.getInstance();

        if (instance1==instance2){System.out.println("我俩一样...");
        }else {System.out.println("我俩不一样...");
        }
    }
}

以上就是单例设计模式中的懒汉式和饿汉式,应该是设计模式中最简略的一个,了解起来难度也不大。

为了克服老王和他儿子小王一起来拿错的难堪,咱们在办法上加 synchronized 锁,对象援用上加 volatile 共享变量,但这样会带来效率问题,如果不思考多线程需要,读者可自行去掉。

三、原型

老王明天很显著是找茬,他持续说,如果我不想要一个了,我要每次买都要不同的,你看着办。

每次创立产品都要不同的,传统的形式必定就是从新 new 一个对象,但每创立一个对象都是一个简单的过程,而且这样还会带来肯定的代码冗余。

这就须要用到创立型设计模式中的原型模式中的拷贝,其中又分为浅拷贝和深拷贝。

咱们先看基本概念。

  • 浅克隆:创立一个新对象, 对象种属性和原来对象的属性完全相同, 对于非根本类型属性仍指向原有属性所指向的内存地址
  • 深克隆:创立一个新对象, 属性中援用类型也会被克隆, 不再指向原来属性所指向的内存地址

这段意思也就是,老王购买产品的时候,如果产品都是根本数据类型(byte(位)、short(短整数)、int(整数)、long(长整数)、float(单精度)、double(双精度)、char(字符)和 boolean(布尔值))和 String, 那么咱们就应用浅拷贝。

如果产品包含别的产品(对象)的援用类型就要应用深拷贝。

如果想搞明确,为什么造成深拷贝和浅拷贝这个问题,咱们就要重点说说 JVM 的内存模型。

咱们申明一个根本数据类型的变量 a =2,实际上是在栈中间接存储了一个 a =2,当拷贝的时候间接把值拷贝过来,也就是间接有了一份 a 的正本。

当咱们创立一个对象时 Student stu=new Student(),实际上对象的值存储在堆中,在栈中只寄存了 stu=” 对象地址 ”,stu 指向了堆中的地址,jvm 拷贝的时候只复制了栈中的地址,实际上他们堆中的对象还是一个。

咱们再来看 String 类型。String 存在于堆内存、常量池;这种比拟非凡, 传递是援用地址;由自身的 final 性, 每次赋值都是一个新的援用地址,原对象的援用和正本的援用互不影响。因而 String 就和根本数据类型一样, 体现出了 ” 深拷贝 ” 个性。

咱们具体看实现代码:

浅拷贝类:

/**
 * @author tcy
 * @Date 29-07-2022
 */
public class ShallowProduct implements Cloneable{

    private String name;

    private int num;

    public void show(){System.out.println("这是浅产品..."+name+"数量:"+num);
    }

    public String getName() {return name;}

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

    public int getNum() {return num;}

    public ShallowProduct setNum(int num) {
        this.num = num;
        return this;
    }

    @Override
    public ShallowProduct clone() throws CloneNotSupportedException {return (ShallowProduct) super.clone();}
}

如果须要哪个对象浅拷贝,须要该对象实现 Cloneable 接口,并重写 clone() 办法。

public void shallowTest()throws CloneNotSupportedException{ShallowProduct product1=new ShallowProduct();
    ShallowProduct product2 = product1.clone();
    product1.setName("老王");
    product2.setName("老李");

    product1.setNum(1);
    product2.setNum(2);

    product1.show();

    product2.show();}

调用时输入的对象中的值间接就是两个不同的对象,实现了对象的浅拷贝。

如果该对象中包含援用类型呢? 那怎么实现呢。

其实原理上也是很简略的,只须要将非根本数据类型也像浅拷贝那样操做就行了,而后在以后 clone() 办法中,调用非根本数据类型的 clone() 办法

深拷贝援用类:

/**
 * @author tcy
 * @Date 29-07-2022
 */
public class Child implements Cloneable{

    private String childName;

    public String getChildName() {return childName;}

    public Child setChildName(String childName) {
        this.childName = childName;
        return this;
    }

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

深拷贝类:

/**
 * @author tcy
 * @Date 29-07-2022
 */
public class DeepProduct implements Cloneable{

    private String name;

    private Integer num;

    private Child child;
    public String getName() {return name;}

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

    public Integer getNum() {return num;}

    public DeepProduct setNum(Integer num) {
        this.num = num;
        return this;
    }

    public void show(){System.out.println("这是深产品..."+name+"数量:"+num+"包含 child:"+child.getChildName());
    }


    @Override
    public DeepProduct clone() throws CloneNotSupportedException {DeepProduct clone = (DeepProduct) super.clone();
        clone.child=child.clone();
        return clone;
    }

    public Child getChild() {return child;}

    public DeepProduct setChild(Child child) {
        this.child = child;
        return this;
    }
}

咱们测试一下对象中的值是否产生了扭转。

public void deepTest() throws CloneNotSupportedException {DeepProduct product1=new DeepProduct();
    Child child=new Child();
    child.setChildName("老王 child");

    product1.setName("老王");
    product1.setNum(1);
    product1.setChild(child);

    //--------------
    DeepProduct product2=product1.clone();
    product2.setName("老李");
    product2.setNum(2);
    product2.getChild().setChildName("老李 child");

    product1.show();
    product2.show();}

老李、老王都正确的输入了,阐明实现没有问题。

这样就合乎了老王的要求。

既然说到了 jvm 的内存模型,就有必要说一下 java 中的值传递和援用传递。

实际上 java 应该就是值传递,在调用办法的时候,如果参数是根本数据类型,那么传递的就是正本,咱们在办法中无论怎么给他赋值,他本来的值都不会有变动。

在调用办法的时候,如果参数是援用数据类型,那么传递的就是这个对象的地址,咱们在办法中批改这个对象都会影响他本来的对象。

造成这个景象的起因其实是和浅拷贝、深拷贝的原理是一样的,都是栈、堆内存的构造导致的。

老王看他的要求都满足了,最初称心如意的拿着产品走了。

正文完
 0