python 享元模式_python 设计模式之享元(Flyweight)模式

#写在前面

这个设计模式理解起来很容易。百度百科上说的有点绕口。

#享元模式的定义

运用共享技术来有効地支持大量细粒度对象的复用。

它通过共享已经存在的对橡大幅度减少需要创建的对象数量、避免大量相似类的开销,从而提高系统资源的利用率。

#优点

相同对象只要保存一份,这降低了系统中对象的数量,从而降低了系统中细粒度对象给内存带来的压力。

#缺点

1.为了共享对象,需要将不能共享的状态外部化,会增加程序的复杂性

2.对享元模式的外部状态会增长运行时间

#享元模式中存在的两种状态

1.内部状态,不会随着环境的改变而改变,可共享的部分

2.外部状态,会随着环境的改变而改变,是不可共享的部分.

享元模式就是要区分这两种状态,并将外部状态外部化。

#应用实例

1.java中的String,如果有则返回,如果没有就创建一个字符串保存在字符串缓冲池里面。

2.数据库的数据池

#应用场景

1.系统中存在大量的相似对象

2.细粒度的对象都具备较接近的外部状态,而且内部状态与环境无关,也就是说对象没有特定身份

3.需要缓冲池的场景

#享元模式的主要角色

抽象享元角色(Flyweight):是所有的具体享元类的基类,为具体享元规范需要实现的公共接口,非享元的外部状态以参数的形式通过方法传入。

具体享元(Concrete Flyweight)角色:实现抽象享元角色中所规定的接口。

非享元(Unsharable Flyweight)角色:是不可以共享的外部状态,它以参数的形式注入具体享元的相关方法中。

享元工厂(Flyweight Factory)角色:负责创建和管理享元角色。当客户对象请求一个享元对象时,享元工厂检査系统中是否存在符合要求的享元对象,如果存在则提供给客户;如果不存在的话,则创建一个新的享元对象。

#举例说明什么是具体享元角色和非具体享元角色

举下围棋的例子

棋子:是抽象享元角色

白子:具体享元角色

黑子:具体享元角色

围棋工厂:是享元工厂角色

下棋的位置:非具体享元角色

#享元模式的结构图

#举个例子1

class Flyweight(object):

def __init__(self, str):

self.str = str

def display(self):

print("show the string: " + self.str)

class FlyweightFactory(object):

def __init__(self):

self.flyweights = {}

def getFlyweight(self, obj):

flyweight = self.flyweights.get(obj)

if flyweight == None:

flyweight = Flyweight(str(obj))

self.flyweights[obj] = flyweight

return flyweight

def showFlyweights(self):

for i in self.flyweights:

self.flyweights[i].display()

print(len(self.flyweights))

if __name__ == "__main__":

flyweightfactory = FlyweightFactory()

fly1 = flyweightfactory.getFlyweight("hello1")

fly2 = flyweightfactory.getFlyweight("hello1")

fly3 = flyweightfactory.getFlyweight("hello2")

fly4 = flyweightfactory.getFlyweight("hello2")

fly5 = flyweightfactory.getFlyweight("hello3")

flyweightfactory.showFlyweights()

out:

show the string: hello2

show the string: hello1

show the string: hello3

3

#举个例子2

import random

from enum import Enum

TreeType=Enum("TreeType",("apple","cherry","peach"))

class Tree:

pool={} #数目池

def __new__(cls,tree_type):

obj=cls.pool.get(tree_type,None) #获取树苗类型

if not obj: #如果之前没有创建过

obj=object.__new__(cls) #开辟一块内存空间

cls.pool[tree_type]=obj #在池子里添加这个树苗类型和他的内存空间

obj.tree_type=tree_type #内存空间里添加树苗类型

return obj #返回树苗类型空间

def render(self,age,x,y):

print("创建了一个新的种类{}的树苗,他的年龄是{},地点位于{},{}".format(

self.tree_type,age,x,y

))

def main():

rnd=random.Random()

age_min,age_max=1,30 #树苗年龄在1-30之间随机

min_point,max_point=0,100 #随机地点

tree_counter=0

print(rnd)

for _ in range(10): #10个苹果树

t1=Tree(TreeType['apple'])

t1.render(rnd.randint(age_min,age_max),

rnd.randint(min_point,max_point),

rnd.randint(min_point, max_point))

tree_counter+=1

for _ in range(3): #3个cherry树

t2=Tree(TreeType['cherry'])

t2.render(rnd.randint(age_min,age_max),

rnd.randint(min_point,max_point),

rnd.randint(min_point, max_point))

tree_counter+=1

for _ in range(5): #5个peach树

t3=Tree(TreeType['peach'])

t3.render(rnd.randint(age_min,age_max),

rnd.randint(min_point,max_point),

rnd.randint(min_point, max_point))

tree_counter+=1

print("树苗创建了{}个".format(tree_counter))

print("树苗完成创建:{}种类".format(len(Tree.pool)))

t4=Tree(TreeType['cherry'])

t5 = Tree(TreeType['cherry'])

t6=Tree(TreeType['apple'])

print("{}----{}是同一颗树吗? {}".format(id(t4),id(t5),id(t4)==id(t5)))

print("{}----{}是同一颗树吗? {}".format(id(t5), id(t6), id(t6) == id(t5)))

if __name__ == '__main__':

main()

参考

http://c.biancheng.net/view/1371.html(下围棋)

https://www.runoob.com/design-pattern/flyweight-pattern.html

https://www.cnblogs.com/hujingnb/p/10171607.html

http://dongweiming.github.io/python-flyweight.html

https://blog.csdn.net/weixin_42557907/article/details/84204040(例子2出处)

https://blog.csdn.net/u013346751/article/details/78426104(例子1出处)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/410273.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[html] 怎样在<pre>标签内不转义<和>符号(原样输出html标签)?

[html] 怎样在标签内不转义<和>符号&#xff08;原样输出html标签&#xff09;&#xff1f; 将HTML代码嵌入到<script typetext/html styledisplay:block>中<script type"text/html" style"display: block;">哈哈哈dfdfd</script>…

单列模式

最近在学设计模式&#xff0c;学到创建型模式的时候&#xff0c;碰到单例模式&#xff08;或叫单件模式&#xff09;&#xff0c;现在整理一下笔记。 在《Design Patterns&#xff1a;Elements of Resuable Object-Oriented Software》中的定义是&#xff1a;Ensure a class on…

算法之排序算法-直接插入排序

package com.ebiz.sort;import java.text.SimpleDateFormat; import java.util.Date;/*** author YHj* create 2019-07-29 8:56* 插入排序*/ public class Insert {public static void main(String[] args) {// int [] arr{101,34,119,1};int[] arr new int[80000];for (int i…

佳能g3800故障灯说明书_热水器维修电话|史密斯燃气热水器出现16故障码

热水器出现故障代码其实是一件很常见的事情&#xff0c;大多是住户的热水器都出现过这样的问题&#xff0c;为了防止这样的事情也发生在我身上&#xff0c;我熟读热水器说明书&#xff0c;终于参透其中的道理&#xff0c;那么接下来我为大家介绍史密斯燃气热水器出现16故障码。…

DrawImage内存不足问题

出现这种问题&#xff0c;分析如下&#xff1a; 1.图片太大&#xff0c;绘制完没有及时释放。所谓图片太大&#xff0c;一种是原始图片本身很大&#xff0c;一种是把小图片拉伸到很大的矩形区域显示。 2.图片格式不对或者参数不对。 3.图片不完整。比如图片只下载了一半&#x…

算法之排序算法-shell排序(交换法)

可以先看注释掉的分析,最后在看三层for循环 package com.ebiz.sort;import java.util.Arrays;/*** author YHj* create 2019-07-30 8:53* shell排序-交换法*/ public class Shell{public static void main(String[] args) {int [] arr{8,9,1,7,2,3,5,4,6};getResult(arr);// Sy…

git语言包安装_Git分布式版本管理系统快速入门指南

为什么要使用版本管理系统无论有没有使用过专业化工具&#xff0c;每个人都或多或少地有版本管理的需求。我们在做论文、写报告或者设计方案时&#xff0c;因为难以避免的不断改动&#xff0c;总会形成很多个不同的版本&#xff0c;我们可能会用“某某设计方案_20180910”这样加…

[html] 如何放大点击的区域?

[html] 如何放大点击的区域&#xff1f; <div class"button">点我</div><style>.button{position:relative;}.button::after{content:"";position:absolute;top:-10px;left:-10px;right:-10px;bottom:-10px;}</style>个人简介 我…

(十)、java内部类与内部类的闭包和回调

一、成员内部类 1.可以把一个内部类看做是一个成员。成员内部类可以无条件访问外部类的所有成员属性和成员方法。 class OutterClass {//外部类private int in 0;static int inn4;public OutterClass(int in) {this.in in;}class InnerClass { //内部类public void outpu…

vb.net中滚动条一直显示没有数据时也显示_Android Studio 中 System Trace 的新增功能...

Android Studio 中 System Trace 的新增功能在 Android Studio 4.0 中&#xff0c;我们已经对 CPU Profiler 的 UI 做了大量调整来提供更加直观的工作流记录&#xff0c;而在 Android Studio 4.1 中&#xff0c;我们基于开发者们的反馈对此功能进行了持续改进&#xff0c;并且新…

[html] DOM节点的根节点是不是body?

[html] DOM节点的根节点是不是body&#xff1f; 文档对象模型 (DOM) 是HTML和XML文档的编程接口。它提供了对文档的结构化的表述&#xff0c;并定义了一种方式可以使从程序中对该结构进行访问&#xff0c;从而改变文档的结构&#xff0c;样式和内容。DOM 将文档解析为一个由节…

第一节 Memcached分布式缓存入门

关于Memcached的博文太多了&#xff0c;以下是个人学习的收集整理。 本节讨论问题: 简单介绍与应用下载安装注意事项简单测试Memcached分布式原理 一、介绍与应用 在常规的WEB开发下&#xff0c;基本都会利用到缓存用以降低对数据库的压力&#xff0c;提高访问速度。有时候缓存…

tomcat 拦截指定url_一口气说出 过滤器 和 拦截器 6个区别,别再傻傻分不清了

点击“ 程序员内点事 ”关注&#xff0c;选择“ 设置星标 ”坚持学习&#xff0c;好文每日送达&#xff01;周末有个小伙伴加我微信&#xff0c;向我请教了一个问题&#xff1a;老哥&#xff0c;「过滤器 (Filter) 和 拦截器 (Interceptor) 有啥区别啊&#xff1f;」 听到题目我…

[html] 如何使用纯html制作一个进度条?

[html] 如何使用纯html制作一个进度条&#xff1f; HTML中的progress () 元素用来显示一项任务的完成进度.虽然规范中没有规定该元素具体如何显示,浏览器开发商可以自己决定,但通常情况下,该元素都显示为一个进度条形式.个人简介 我是歌谣&#xff0c;欢迎和大家一起交流前后…

Java 多线程练习---创建两个子线程,每个线程交替输出“你好--来自线程***”...

|--需求说明 |--实现思路 1、创建一个类&#xff0c;实现Runnable&#xff0c;在这个类里面重写run()方法&#xff0c;在run()方法里面写一个20的for循环 2、创建一个类&#xff0c;实例化上面的类&#xff0c;用这个类的对象创建线程 |--代码内容 1 package cn.thread;2 3 /**…

关于Android的应用程序的发布的学习(一)

上一次写了一篇关于android应用程序打包成apk文件的签名&#xff0c;地址&#xff1a;http://blog.csdn.net/zqiang_55/article/details/6939170 最近再看sdk的时候发现其实在sdk中又了详细的少说明&#xff0c;现在将主要的翻译如下&#xff1a; 在Dev Gudie标签页中的Publish…

手动封装js的call和apply和bind和typeof和new方法

我是歌谣 放弃很容易 但是坚持一定很酷 微信公众号关注小歌谣一起学习前后端知识 闲来无事做 不如敲代码今天讲一下js里面的callapply和bind和typeof方法的手动封装由于最近比较忙但是有时间就会记录一下平时学习工作的一些代码用来分享这边就不直接多说开始我们的直接代码书写…

js保留两位小数的函数_使用率低但功能强大的6个Excel函数公式应用技巧解读!...

在Excel函数公式中&#xff0c;有部分函数的使用率是比较低的&#xff0c;但是其功能也是非常强大的。一、Median函数。功能&#xff1a;返回一组数的中值。中值就是一组数的中间数值&#xff0c;如果参数包含的数值是偶数&#xff0c;Median函数将返回位于中间两个值的平均值。…

IP包的生成和发送接口(1)

http://blog.sina.com.cn/s/indexlist_1657348185_2.html IP包的生成和发送接口 (1) Linux内核中有3种基本的IP包生成器, 它们分别为ip_build_xmit(), ip_queue_xmit(), ip_build_and_send_pkt(). ip_build_and_send_pkt()是一简单的IP包头封装接口, 它接照输入包的路由添加一…

.net复习之七

表A&#xff1a; 表B&#xff1a; 1&#xff0e; SELECT * FROM A JOIN B ON A.Id B.Id 將顯示 9 條數據。 Inner join(等值连接)只返回两个表中联结字段相等的行 2&#xff0e; SELECT * FROM A LEFT JOIN B ON A.Id B.Id 將顯示 12 條數據…