Python缓存类实例

本篇文章的内容主要包含

  1. 利用Python弱引用存储字典缓存类的实例,让参数相同的实例不用重复生成
  2. 略过复杂的通用化代码编写,利用Python自带库来缓存实例和方法对象

在Python的许多库中都有缓存实例的例子,比如logging模块的Logger类实例

import logginga = logging.getLogger("abc")
b = logging.getLogger("abc")print(a is b)   # True

弱引用(weakref)通常用于缓存或映射数据量较大的对象,当你使用python字典存放例如key为name,value为一个很大的图像对象,或者反过来,引用和修改这个图像对象时始终是“实时的”,因为它实际的存在字典中。但是利用弱引用中的WeakKeyDictionaryWeakValueDictionary,你引用这个图像时只需要使用它的名字即可,它不需要占用很大的空间来存储,实际上弱引用就是一个对象指针。

创建实例缓存的一种方式是写一个方法,每次都通过这个方法访问实例

import weakrefclass Person:def __init__(self, name):print("person initializing")self.name = name_cache_instance = weakref.WeakValueDictionary()  # 这里应该使用全局变量,因为是共用的存储对象def get_person(name):if name not in _cache_instance:instance = Person(name)_cache_instance[name] = instancereturn instanceelse:return _cache_instance[name]def test():q = get_person("q")e = get_person("q")print(q is e)r = get_person("q")print(e is r)if __name__ == '__main__':test()

输出

person initializing
True
True

从输出可以看出3次调用参数相同的Person只实例了一次

看到这里很容易联想到使用装饰器来简化代码

import weakref
from functools import wraps_cache_instance = weakref.WeakValueDictionary()  def instance_cache(cls_instance):@wraps(cls_instance)def inner(name, *args, **kwargs):if name not in _cache_instance:instance = cls_instance(name, *args, **kwargs)_cache_instance[name] = instanceelse:instance = _cache_instance[name]return instancereturn inner@instance_cache
class Person:def __init__(self, name):print("person initializing")self.name = namedef test():q = Person("abc")e = Person("abc")print(q is e)r = Person("abc")print(e is r)if __name__ == '__main__':test()

输出

person initializing
True
True

输出在期望之内,但这个装饰器要实现到更通用化还是不够,想要把存入的参数通用化作为键并且适用于不同的类实例和方法并不是一件简单的事情,幸好Python为我们提供了一个非常好用和方便的最近最久未使用的缓存方法functools.lru_cache

from functools import lru_cache@lru_cache()
class Person:def __init__(self, fist_name, last_name=None):print("person initializing")self.first_name = fist_nameself.last_name = last_namedef set_last_name(self, name):self.last_name = namedef test():# 测试相同参数的实例是否会重复生成p1 = Person("abc", "d")p2 = Person("abc", "d")print(p1 is p2)p3 = Person("abc", "d")print(p2 is p3)def test1():# 测试较复杂的参数的实例是否会重复生成import jsond1 = {"a": [1, 2], "b": "3"}d2 = {"a": [1, 2], "b": "3"}d3 = {"a": [1, 2], "b": "3"}d1 = json.dumps(d1)d2 = json.dumps(d2)d3 = json.dumps(d3)p1 = Person("abc", d1)p2 = Person("abc", d2)print(p1 is p2)p3 = Person("abc", d3)print(p2 is p3)def test2():# 测试实例发生改变后的情况p1 = Person("abc")p2 = Person("abc")print(p1 is p2)p2.set_last_name("def")print(p1 is p2)if __name__ == '__main__':# test()# test1()test2()

输出

person initializing
True
True

输出一切都是我们所期望的,而且不管是类方法还是普通的函数,都可以使用lru_cache来实现对象方法的缓存,此外lru_cache还支持缓存大小限制(maxsize)和严格的参数类型匹配校验(typed)以及提供被装饰的对象方法以缓存的清除和信息查看等方法

@lru_cache
def function():print("call function")return 123def test3():# 只会调用一次functionf = function()f2 = function()# 缓存的参数信息print(function.cache_parameters())# 缓存信息print(function.cache_info())# 清除缓存function.cache_clear()print(function.cache_info())if __name__ == '__main__':test3()

输出

call function
CacheInfo(hits=1, misses=1, maxsize=128, currsize=1)
{'maxsize': 128, 'typed': False}
CacheInfo(hits=0, misses=0, maxsize=128, currsize=0)

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

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

相关文章

【Pytorch神经网络理论篇】 06 神经元+神经网络模型+全连接网络模型

同学你好!本文章于2021年末编写,获得广泛的好评! 故在2022年末对本系列进行填充与更新,欢迎大家订阅最新的专栏,获取基于Pytorch1.10版本的理论代码(2023版)实现, Pytorch深度学习理论篇(2023版)目录地址…

面试题:N皇后问题,思路和python解题笔记

n皇后问题算法思路和python解法 问题描述 n皇后问题,在nn的棋盘上,解出n个皇后所有不能互相攻击的摆法, 皇后在数组中用“Q”表示,空地用“.”表示 返回的数据结构格式要求:[[“.Q…”,“…Q”,“Q…”,“…Q.”],[“…

【Pytorch神经网络理论篇】 07 激活函数+Sigmoid+tanh+ReLU+Swish+Mish+GELU

同学你好!本文章于2021年末编写,获得广泛的好评! 故在2022年末对本系列进行填充与更新,欢迎大家订阅最新的专栏,获取基于Pytorch1.10版本的理论代码(2023版)实现, Pytorch深度学习理论篇(2023版)目录地址…

截图命令

adb shell /system/bin/screencap -p /sdcard/screenshot.png adb pull /sdcard/screenshot.png E:\

【Pytorch神经网络理论篇】 08 Softmax函数(处理分类问题)

同学你好!本文章于2021年末编写,获得广泛的好评! 故在2022年末对本系列进行填充与更新,欢迎大家订阅最新的专栏,获取基于Pytorch1.10版本的理论代码(2023版)实现, Pytorch深度学习理论篇(2023版)目录地址…

Python套接字编程Socket Progaming——1

本篇文章是Network And Web Programing-Socket Programing分类中的第一篇文章,内容主要包含 Socket概念理解Socket programing介绍一个简单的TCP协议的server-client程序支持同时处理多个客户端简单server-client连接程序socket的常用选项使用 理解socket概念 一…

【Pytorch神经网络理论篇】 09 神经网络模块中的损失函数

同学你好!本文章于2021年末编写,获得广泛的好评! 故在2022年末对本系列进行填充与更新,欢迎大家订阅最新的专栏,获取基于Pytorch1.10版本的理论代码(2023版)实现, Pytorch深度学习理论篇(2023版)目录地址…

jquery hover事件中 fadeIn和fadeOut 效果不能及时停止

$(".nav ul li").hover(function () {var id $(this).attr("id");$(".nav dl").each(function (index, domEle) {if ($(domEle).attr("id") id) {$(domEle).fadeIn();}else {$(domEle).stop().fadeOut();//在这里加入.stop() 以阻止…

【Pytorch神经网络理论篇】 10 优化器模块+退化学习率

同学你好!本文章于2021年末编写,获得广泛的好评! 故在2022年末对本系列进行填充与更新,欢迎大家订阅最新的专栏,获取基于Pytorch1.10版本的理论代码(2023版)实现, Pytorch深度学习理论篇(2023版)目录地址…

Android fb0 截屏实现

问题:我们有几个项目,在项目1和项目2上实现截屏是没有问题的,但是在项目3上实现截屏是不行的 原因:分辨率差异引起的问题,分辨率长宽一定要是32的整数倍 Dear customer, Sorry for the late reply due to annual leave. I dont think this issue relates with thediff…

Python trino执行hive insert overwrite不生效的问题

使用python的trino包执行insert overwrite,但是overwrite却没有生效的问题 根据trino的官网介绍的insert overwrite的开启方式,开启hive的insert overwrite会话,使当前会话的insert into语句支持insert overwrite,也即支持插入数…

HAProxy负载均衡原理及企业级实例部署haproxy集群

HAProxy是一种高效、可靠、免费的高可用及负载均衡解决方案,非常适合于高负载站点的七层数据请求。客户端通过HAProxy代理服务器获得站点页面,而代理服务器收到客户请求后根据负载均衡的规则将请求数据转发给后端真实服务器。 同一客户端访问服务器&…

【Pytorch神经网络实战案例】07 预测泰坦尼克号上生存的乘客

1 样本处理 1.1 载入样本代码---Titanic forecast.py(第1部分) import numpy as np import torch import torch.nn as nn import torch.nn.functional as F from scipy import stats import pandas as pd import matplotlib.pyplot as plt import os o…

ubuntu下 安装 adb

1、把adb tool工具考到你要安装的目录夏目 <

基于sanic的服务使用celery完成动态修改定时任务

首先声明一下 考虑到celery目前和asyncio的不兼容性&#xff0c;协程任务需要转换为非异步的普通方法才能被当做task加入定时&#xff0c;并且celery和asyncio使用可能会带来预想不到的问题&#xff0c;在celery官方第二次承诺的6.0版本融合asyncio之前&#xff0c;需要慎重考虑…

shell 中的ifeq

libs_for_gcc -lgnunormal_libs foo: $(objects)ifeq ($(CC),gcc)$(CC) -o foo $(objects) $(libs_for_gcc)else$(CC) -o foo $(objects) $(normal_libs)endif 可见&#xff0c;在上面示例的这个规则中&#xff0c;目标“foo”可以根据变量“$(CC)”值来选取不同的函数库来编…

第一篇unity

在网上找的学习资料&#xff0c;做了点简单的效果。 半成品 http://files.cnblogs.com/files/buzhidaojiaoshenme/unity.rar 第二个游戏&#xff0c;方向键和“W”&#xff0c;”S“键移动方块&#xff0c;碰撞到最右边的方块过关。 http://files.cnblogs.com/files/buzhidaoji…

报错:OMP: Error #15: Initializing libomp.dylib, but found libiomp5.dylib already initialized.

问题描述&#xff1a; OMP: Error #15: Initializing libiomp5.dylib, but found libiomp5.dylib already initialized. OMP: Hint This means that multiple copies of the OpenMP runtime have been linked into the program. That is dangerous, since it can degrade perf…

Pyscript,使用Python编写前端脚本

介绍 Anaconda的CEO Peter Wang在前两个月的时候发布了Pyscript&#xff0c;实现了在HTML支持Python的使用&#xff0c;整个引用过程甚至不需要安装任何环境&#xff0c;只需要使用link和script标签即可引用实现Python在HTML中运行的功能&#xff0c;在HTML中也可以运行和使用…

如何把应用程序app编译进android系统

转载&#xff1a;http://ywxiao66.blog.163.com/blog/static/175482055201152710441106/------------------------------------------------------------------把常用的应用程序编译到img文件中&#xff0c;就成了系统的一部分&#xff0c;用户不必自己安装&#xff0c;当然也卸…