我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java设计模式之享元模式

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java设计模式之享元模式

本文通过优化买票的重复流程来说明享元模式,为了加深对该模式的理解,会以String和基本数据类型的包装类对该模式的设计进一步说明。

读者可以拉取完整代码到本地进行学习,实现代码均测试通过后上传到码云,本地源码下载。

一、引出问题

鉴于小王之前的优质表现,老王决定带小王出去旅游一下,但在火车站买票时却陷于了长长的队伍。

老王注意到,每次售票员卖票时都重新走一遍卖票的所有流程,很明显,如果始发地和目的地如果一样的成人票和儿童票是可以复用流程的,如果复用的话就可以大大提高卖票效率。

二、概念和使用

上面所说的复用流程实际上就是享元模式的设计思想,它是构造型设计模式之一,它通过共享数据使得相同对象在内存中仅创建一个实例,以降低系统创建对象实例的性能消耗。

享元模式包含三个角色:

(1)抽象享元Flyweight类:享元对象抽象基类或接口。

(2)具体享元ConcreteFlyweight类:实现抽象享元类。

(3)享元工ctory类:厂FlyweightFa享元模式的核心模块,负责管理享元对象池、创建享元对象,保证享元对象可以被系统适当地共享。

当一个客户端对象调用一个享元对象的时候,享元工厂角色会检查系统中是否已经有一个符合要求的享元对象,如果已有,享元工厂角色就提供这个已有的享元对象;如果没有就创建一个。

老王基于享元模式开发了一套卖票系统,如果起点和终点一样,成人票和儿童票就可以复用一套流程。

抽象享元类:


public interface Ticket {
    //显示票价,参数为列车类型
    public void showPrice(String type);

}

具体享元实现类:


public class ConcreteTicket implements Ticket{

    String from;
    String to;
    public ConcreteTicket(String from,String to){
        this.from = from;
        this.to = to;
    }

    @Override
    public void showPrice(String type) {
        if(type.equals("adult")){
            System.out.println("从"+from+"到"+to+"的成人票价为200元");
        }else{
            System.out.println("从"+from+"到"+to+"的儿童票价为100元");
        }

    }
}

享元工厂类:


public class TicketFactory {

    static Map<String,Ticket> map= new ConcurrentHashMap< String,Ticket >();

    public static Ticket getTicket(String from,String to){
        String key = from+to;
        if(map.containsKey(key)){
            System.out.println("使用缓存"+key);
            return map.get(key);
        }else{
            System.out.println("创建对象"+key);
            Ticket ticket = new ConcreteTicket(from,to);
            map.put(key, ticket);
            return ticket;
        }
    }

}

客户端调用:


public class Client {
    public static void main(String[] args) {
        //使用时
        TicketFactory.getTicket("南京","杭州").showPrice("adult");
        TicketFactory.getTicket("南京","杭州").showPrice("children");
    }
}

上面例子是享元模式实现的典型案例。核心其实就是享元工厂类,享元工厂类设置一个缓存池,根据条件判断是否属于一个对象,如果是一个对象就不再重新创建,直接使用缓存池中的。

三、运用

1、jdk中的String就是典型的采用的享元模式的思想。

Java中将String类定义为final(不可改变的),JVM中字符串一般保存在字符串常量池中,java会确保一个字符串在常量池中只有一个拷贝,这个字符串常量池在JDK6.0以前是位于常量池中,位于永久代,而在JDK7.0中,JVM将其从永久代拿出来放置于堆中。

创建一个字符串有两种方式,一种是直接String="hello",另外一种是String s =new String("hello"),第一种是直接在字符串常量池声明一个变量,第二种方式除了是一个堆中的普通对象以外,还会在字符串常量池保存一份。

我们经常使用的一些基本数据类型的包装类实际上也使用了享元模式。我们以Integer 举例,其他包装类类似。

当我们声明一个变量时,使用Integer i1 = 88,编译器是不会报错的,在这Java上面的一个概念就叫自动装箱,编译器会自动 使用valueOf()方法创建一个Integer对象并把值赋给该对象。

查看valueOf()方法,如下:

public static Integer valueOf(int i) {
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}

Integer 使用享元模式的核心就在于IntegerCache,它是Integer 的一个内部类。

这里的 IntegerCache 相当于享元设计模式中的享元对象工厂类,既然是享元对象工厂类就一定会有判定一个对象是否一样的条件。

private static class IntegerCache {
    static final int low = -128;
    static final int high;
    static final Integer cache[];
    static {
        // high value may be configured by property
        int h = 127;
        String integerCacheHighPropValue =
            sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
        if (integerCacheHighPropValue != null) {
            try {
                int i = parseInt(integerCacheHighPropValue);
                i = Math.max(i, 127);
                // Maximum array size is Integer.MAX_VALUE
                h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
            } catch( NumberFormatException nfe) {
                // If the property cannot be parsed into an int, ignore it.
            }
        }
        high = h;
        cache = new Integer[(high - low) + 1];
        int j = low;
        for(int k = 0; k < cache.length; k++)
            cache[k] = new Integer(j++);
        // range [-128, 127] must be interned (JLS7 5.1.7)
        assert IntegerCache.high >= 127;
    }
    private IntegerCache() {}
}

通过源码我们可以看到,IntegerCache 的判定一个对象是否是同一个的判断标准就是,一个字节的大小(-128 到 127 之间的数据)都作为一个对象。

既然说到了自动装箱,那相对应的也一定会有自动拆箱。

当把包装器类型的变量i1,赋值给基本数据类型变量 j 的时候,触发自动拆箱操作,将 i1中的数据取出,赋值给 j,这就是自动拆箱的过程。

其他包装器类型,比如 Long、Short、Byte 等,也都利用了享元模式来缓存 -128 到 127 之间的数据。比如,Long 类型对应的 LongCache 享元工厂类。

四、总结

享元模式与我们常说的缓存的概念很相似,总体来说还是一个很简单的设计模式,在我们实际使用中为了提高对象利用率,可以有意识的使用这种模式。

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对编程网的支持。如果你想了解更多相关内容请查看下面相关链接

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java设计模式之享元模式

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

好程序员分享java设计模式之享元模式

  好程序员分享java设计模式之享元模式,享元模式有点类似于单例模式,都是只生成一个对象被共享使用。享元模式主要目的就是让多个对象实现共享,减少不必要的内存消耗,将多对同一对象的访问集中起来,不必为每个访问者创建一个单独的对象,以此来降低
2023-06-02

每天一个设计模式之享元模式

作者按:《每天一个设计模式》旨在初步领会设计模式的精髓,目前采用javascript和python两种语言实现。诚然,每种设计模式都有多种实现方式,但此小册只记录最直截了当的实现方式 :)个人技术博客-godbmw.com 欢迎来玩! 每周
2023-01-31

如何理解Java设计模式的享元模式

本篇内容介绍了“如何理解Java设计模式的享元模式”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、引言大家都知道单例模式,通过一个全局变量
2023-06-25

一文搞懂设计模式—享元模式

享元模式通过共享相似对象来减少内存消耗,提高系统性能。它适用于存在大量相似对象且造成内存浪费的场景,但需要注意对内部状态和外部状态的管理。合理应用享元模式可以有效优化系统架构,提升性能。

.Net结构型设计模式之享元模式怎么实现

这篇文章主要介绍了.Net结构型设计模式之享元模式怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇.Net结构型设计模式之享元模式怎么实现文章都会有所收获,下面我们一起来看看吧。一、动机(Motivate
2023-06-30

Java设计模式之状态模式

这篇文章介绍了Java设计模式之状态模式,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2022-11-13

Java设计模式之策略模式

这篇文章介绍了Java设计模式之策略模式,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2022-11-13

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录