设计模式—创建型模式之原型模式

设计模式---创建型模式之原型模式

原型模式(Prototype Pattern)用于创建重复的对象,同时又能保证性能。

本体给外部提供一个克隆体进行使用。

比如我们做一个SjdwzMybatis,用来操作数据库,从数据库里面查出很多记录,其中很多记录改变很少。每次查数据库,把所有数据都封装一个对象,然后返回。假设有很多线程,来查如下记录:

java 复制代码
Student student = new Student("张三","男");

如果每次都创建对象封装并返回,这样系统就会有很多student;这样就会浪费内存。

Student类如下:

java 复制代码
public class Student {
    private String name;
    private Integer age;

    public Student() {
        System.out.println("创建了Student对象");
    }
    //省略getter() 、 setter() toString()
}

SjdwzMybatis如下:

java 复制代码
public class SjdwzMybatis {

    /**
     * 通过name获取Student
     */
    public Student queryStudent(String name){
        return queryStudentFromDB(name);
    }

    /**
     * 演示从数据库查Student
     */
    private Student queryStudentFromDB(String name) {
        //简单演示,查询到了
        System.out.println("从数据库查询到了:"+name);
        Student student = new Student();
        student.setName(name);
        student.setAge(16);
        return student;
    }
}

测试类:

java 复制代码
public class ProtoTypeTest {
    public static void main(String[] args) {
        SjdwzMybatis sjdwzMybatis = new SjdwzMybatis();
        Student stu1 = sjdwzMybatis.queryStudent("zhangsan");
        Student stu2 = sjdwzMybatis.queryStudent("zhangsan");
        Student stu3 = sjdwzMybatis.queryStudent("zhangsan");
        Student stu4 = sjdwzMybatis.queryStudent("zhangsan");
    }
}

这样会有大量具有相同属性的student被外部创建,同时查库次数过多。

我们是否能设计一个缓存,来保存查过的内容,再查相同的记录时,可以很快拿到原来的原型对象呢?

那我们的SjdwzMybatis便变成了如下代码:

java 复制代码
public class SjdwzMybatis {
	//缓存
    private Map<String,Student> stuCache = new HashMap<>();

    /**
     * 通过name获取Student
     */
    public Student queryStudent(String name){
        if(stuCache.containsKey(name)){
            return stuCache.get(name);
        }else{
            return queryStudentFromDB(name);
        }
    }

    /**
     * 演示从数据库查Student
     */
    private Student queryStudentFromDB(String name) {
        //简单演示,查询到了
        System.out.println("从数据库查询到了:"+name);
        Student student = new Student();
        student.setName(name);
        student.setAge(16);
        //存入内存
        stuCache.put(name,student);
        return student;
    }
}

但是这是否会有问题呢?

如果我们把stu1的属性改了,那么stu2、stu3、stu4的属性也会被改变,这会影响到我们缓存里的数据,造成脏缓存数据;同时我们查出来的内容,并没有提交修改,不能就把原数据给修改掉。

原型模式

我们把Student修改成如下代码,这便是原型模式:

java 复制代码
//实现Cloneable接口,这只是一个标记,还需要重写clone()方法
public class Student implements Cloneable{
    private String name;
    private Integer age;

    //重写clone方法
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Student student = new Student();
        student.setName(this.name);
        student.setAge(this.age);
        return student;
    }
}

然后SjdwzMybatis修改为如下代码:

java 复制代码
public class SjdwzMybatis {
    //缓存
    private Map<String,Student> stuCache = new HashMap<>();

    /**
     * 通过name获取Student
     */
    public Student queryStudent(String name) throws CloneNotSupportedException {
        if(stuCache.containsKey(name)){
            return (Student) stuCache.get(name).clone();
        }else{
            return queryStudentFromDB(name);
        }
    }

    /**
     * 演示从数据库查Student
     */
    private Student queryStudentFromDB(String name) throws CloneNotSupportedException {
        //简单演示,查询到了
        System.out.println("从数据库查询到了:"+name);
        Student student = new Student();
        student.setName(name);
        student.setAge(16);
        //存入内存
        stuCache.put(name,(Student) student.clone());
        return student;
    }
}

从数据库查出来放入缓存的对象与从缓存取出来的都是clone出来的。

可以看到,我们对stu1修改,并不会影响其他的数据了。

相关推荐
Tadas-Gao23 分钟前
Java设计模式全景解析:从演进历程到创新实践
java·开发语言·微服务·设计模式·云原生·架构·系统架构
励志成为架构师1 小时前
跟小白一起领悟Thread——如何开启一个线程(上)
java·后端
anlogic1 小时前
Java基础 8.16
java·开发语言
可口码农2 小时前
MixOne:Electron Remote模块的现代化继任者
java·前端·electron
蚰蜒螟2 小时前
Netty 的 Select/Poll 机制核心实现主要在 NioEventLoop 的事件循环
java·开发语言
Full Stack Developme2 小时前
Java后台生成多个Excel并用Zip打包下载
java·开发语言·excel
Brookty2 小时前
【Java学习】锁、线程死锁、线程安全2
java·开发语言·学习·java-ee
百锦再3 小时前
.NET 的 WebApi 项目必要可配置项都有哪些?
java·开发语言·c#·.net·core·net
耳东哇3 小时前
spring ai-openai-vl模型应用qwen-vl\gpt-文字识别-java
java·人工智能·spring
花开富贵ii4 小时前
代码随想录算法训练营四十三天|图论part01
java·数据结构·算法·深度优先·图论