本文通过优化买票的重复流程来说明享元模式,为了加深对该模式的理解,会以String和基本数据类型的包装类对该模式的设计进一步说明。
读者可以拉取完整代码到本地进行学习,实现代码均测试通过后上传到码云。
鉴于小王之前的优质表现,老王决定带小王出去旅游一下,但在火车站买票时却陷于了长长的队伍。
老王注意到,每次售票员卖票时都重新走一遍卖票的所有流程,很明显,如果始发地和目的地如果一样的成人票和儿童票是可以复用流程的,如果复用的话就可以大大提高卖票效率。
上面所说的复用流程实际上就是享元模式的设计思想,它是构造型设计模式之一,它通过共享数据使得相同对象在内存中仅创建一个实例,以降低系统创建对象实例的性能消耗。
享元模式包含三个角色:
(1)抽象享元Flyweight类:享元对象抽象基类或接口。
(2)具体享元ConcreteFlyweight类:实现抽象享元类。
(3)享元工ctory类:厂FlyweightFa享元模式的核心模块,负责管理享元对象池、创建享元对象,保证享元对象可以被系统适当地共享。
当一个客户端对象调用一个享元对象的时候,享元工厂角色会检查系统中是否已经有一个符合要求的享元对象,如果已有,享元工厂角色就提供这个已有的享元对象;如果没有就创建一个。
老王基于享元模式开发了一套卖票系统,如果起点和终点一样,成人票和儿童票就可以复用一套流程。
抽象享元类:
/**
* 抽象享元类
*/
public interface Ticket {
//显示票价,参数为列车类型
public void showPrice(String type);
}
具体享元实现类:
/**
* 享元实现类
* @author tcy
* @Date 11-08-2022
*/
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元");
}
}
}
享元工厂类:
/**
* 享元工厂
* @author tcy
* @Date 11-08-2022
*/
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;
}
}
}
客户端调用:
/**
* @author tcy
* @Date 11-08-2022
*/
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 享元工厂类。
享元模式与我们常说的缓存的概念很相似,总体来说还是一个很简单的设计模式,在我们实际使用中为了提高对象利用率,可以有意识的使用这种模式。
到这里,设计模式的第二个大家族-结构形设计模式就介绍完了,万丈高楼平地起,若想灵活的使用设计模式,大量的思考和运用是必不可少的。
推荐读者,参考软件设计七大原则 认真阅读往期的文章,认真体会。
创建型设计模式:
结构型设计模式:
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称
项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU
了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl
我在我的项目中有一个用户和一个管理员角色。我使用Devise创建了身份验证。在我的管理员角色中,我没有任何确认。在我的用户模型中,我有以下内容:devise:database_authenticatable,:confirmable,:recoverable,:rememberable,:trackable,:validatable,:timeoutable,:registerable#Setupaccessible(orprotected)attributesforyourmodelattr_accessible:email,:username,:prename,:surname,:
我经常迷上ruby的一件事是递归模式。例如,假设我有一个数组,它可能包含无限深度的数组作为元素。所以,例如:my_array=[1,[2,3,[4,5,[6,7]]]]我想创建一个方法,可以将数组展平为[1,2,3,4,5,6,7]。我知道.flatten可以完成这项工作,但这个问题是作为我经常遇到的递归问题的一个例子-因此我试图找到一个更可重用的解决方案。简而言之-我猜这种事情有一个标准模式,但我想不出任何特别优雅的东西。任何想法表示赞赏 最佳答案 递归是一种方法,它不依赖于语言。您在编写算法时要考虑两种情况:再次调用函数的情