- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
好的,这是现实世界的场景:我正在编写一个应用程序,并且我有一个代表某种类型文件的类(在我的例子中,这是照片,但该细节与问题无关)。 Photo 类的每个实例对于照片的文件名都应该是唯一的。
问题是,当用户告诉我的应用程序加载文件时,我需要能够识别文件何时已加载,并为该文件名使用现有实例,而不是在同一文件名上创建重复实例。
对我来说,这似乎是一个使用内存的好情况,并且有很多这样的例子,但在这种情况下,我不仅仅是内存一个普通的函数,我需要内存 __init__( )
。这带来了一个问题,因为当 __init__()
被调用时已经为时已晚,因为已经创建了一个新实例。
在我的研究中,我发现了 Python 的 __new__()
方法,我实际上能够编写一个可行的简单示例,但是当我尝试在我的真实世界对象上使用它时它就崩溃了,我不知道为什么(我唯一能想到的是我的真实世界对象是我无法真正控制的其他对象的子类,因此这种方法存在一些不兼容)。这就是我所拥有的:
class Flub(object):
instances = {}
def __new__(cls, flubid):
try:
self = Flub.instances[flubid]
except KeyError:
self = Flub.instances[flubid] = super(Flub, cls).__new__(cls)
print 'making a new one!'
self.flubid = flubid
print id(self)
return self
@staticmethod
def destroy_all():
for flub in Flub.instances.values():
print 'killing', flub
a = Flub('foo')
b = Flub('foo')
c = Flub('bar')
print a
print b
print c
print a is b, b is c
Flub.destroy_all()
哪个输出这个:
making a new one!
139958663753808
139958663753808
making a new one!
139958663753872
<__main__.Flub object at 0x7f4aaa6fb050>
<__main__.Flub object at 0x7f4aaa6fb050>
<__main__.Flub object at 0x7f4aaa6fb090>
True False
killing <__main__.Flub object at 0x7f4aaa6fb050>
killing <__main__.Flub object at 0x7f4aaa6fb090>
太完美了!对于给定的两个唯一 id,只创建了两个实例,而 Flub.instances 显然只列出了两个。
但是当我尝试对我正在使用的对象采用这种方法时,我得到了各种关于 __init__()
如何只接受 0 个参数而不是 2 个参数的荒谬错误。所以我会改变一些事情,然后它会告诉我 __init__()
需要一个参数。太奇怪了。
在和它打了一阵子之后,我基本上放弃了,把所有的__new__()
黑魔法都移到了一个叫做get
的静态方法中,这样我就可以调用Photograph.get(filename)
如果文件名不在 Photograph.instances
中,它只会调用 Photograph(filename)
。
有人知道我哪里出错了吗?有没有更好的方法来做到这一点?
另一种思考方式是,它类似于单例,只是它不是全局单例,只是每个文件名的单例。
Here's my real-world code using the staticmethod get如果你想一起看。
最佳答案
让我们看看关于您的问题的两点。
你可以使用 memoization,但你应该装饰 class,而不是 __init__
方法。假设我们有这个 memoizator:
def get_id_tuple(f, args, kwargs, mark=object()):
"""
Some quick'n'dirty way to generate a unique key for an specific call.
"""
l = [id(f)]
for arg in args:
l.append(id(arg))
l.append(id(mark))
for k, v in kwargs:
l.append(k)
l.append(id(v))
return tuple(l)
_memoized = {}
def memoize(f):
"""
Some basic memoizer
"""
def memoized(*args, **kwargs):
key = get_id_tuple(f, args, kwargs)
if key not in _memoized:
_memoized[key] = f(*args, **kwargs)
return _memoized[key]
return memoized
现在你只需要装饰类:
@memoize
class Test(object):
def __init__(self, somevalue):
self.somevalue = somevalue
让我们看看测试?
tests = [Test(1), Test(2), Test(3), Test(2), Test(4)]
for test in tests:
print test.somevalue, id(test)
输出如下。请注意,相同的参数会产生相同的返回对象的 id:
1 3072319660
2 3072319692
3 3072319724
2 3072319692
4 3072319756
无论如何,我更愿意创建一个函数来生成对象并对其进行内存。对我来说似乎更干净,但这可能是一些无关紧要的小问题:
class Test(object):
def __init__(self, somevalue):
self.somevalue = somevalue
@memoize
def get_test_from_value(somevalue):
return Test(somevalue)
__new__
:当然,您也可以覆盖 __new__
。前几天我发了an answer about the ins, outs and best practices of overriding __new__
这可能会有所帮助。基本上,它说总是将 *args, **kwargs
传递给您的 __new__
方法。
一方面,我更喜欢内存一个创建对象的函数,或者甚至编写一个特定的函数来处理永远不会为相同参数重新创建对象的函数。当然,不过,这主要是我的意见,而不是规则。
关于python - 如何在 Python 中内存类实例化?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10879137/
我正在处理一组标记为 160 个组的 173k 点。我想通过合并最接近的(到 9 或 10 个组)来减少组/集群的数量。我搜索过 sklearn 或类似的库,但没有成功。 我猜它只是通过 knn 聚类
我有一个扁平数字列表,这些数字逻辑上以 3 为一组,其中每个三元组是 (number, __ignored, flag[0 or 1]),例如: [7,56,1, 8,0,0, 2,0,0, 6,1,
我正在使用 pipenv 来管理我的包。我想编写一个 python 脚本来调用另一个使用不同虚拟环境(VE)的 python 脚本。 如何运行使用 VE1 的 python 脚本 1 并调用另一个 p
假设我有一个文件 script.py 位于 path = "foo/bar/script.py"。我正在寻找一种在 Python 中通过函数 execute_script() 从我的主要 Python
这听起来像是谜语或笑话,但实际上我还没有找到这个问题的答案。 问题到底是什么? 我想运行 2 个脚本。在第一个脚本中,我调用另一个脚本,但我希望它们继续并行,而不是在两个单独的线程中。主要是我不希望第
我有一个带有 python 2.5.5 的软件。我想发送一个命令,该命令将在 python 2.7.5 中启动一个脚本,然后继续执行该脚本。 我试过用 #!python2.7.5 和http://re
我在 python 命令行(使用 python 2.7)中,并尝试运行 Python 脚本。我的操作系统是 Windows 7。我已将我的目录设置为包含我所有脚本的文件夹,使用: os.chdir("
剧透:部分解决(见最后)。 以下是使用 Python 嵌入的代码示例: #include int main(int argc, char** argv) { Py_SetPythonHome
假设我有以下列表,对应于及时的股票价格: prices = [1, 3, 7, 10, 9, 8, 5, 3, 6, 8, 12, 9, 6, 10, 13, 8, 4, 11] 我想确定以下总体上最
所以我试图在选择某个单选按钮时更改此框架的背景。 我的框架位于一个类中,并且单选按钮的功能位于该类之外。 (这样我就可以在所有其他框架上调用它们。) 问题是每当我选择单选按钮时都会出现以下错误: co
我正在尝试将字符串与 python 中的正则表达式进行比较,如下所示, #!/usr/bin/env python3 import re str1 = "Expecting property name
考虑以下原型(prototype) Boost.Python 模块,该模块从单独的 C++ 头文件中引入类“D”。 /* file: a/b.cpp */ BOOST_PYTHON_MODULE(c)
如何编写一个程序来“识别函数调用的行号?” python 检查模块提供了定位行号的选项,但是, def di(): return inspect.currentframe().f_back.f_l
我已经使用 macports 安装了 Python 2.7,并且由于我的 $PATH 变量,这就是我输入 $ python 时得到的变量。然而,virtualenv 默认使用 Python 2.6,除
我只想问如何加快 python 上的 re.search 速度。 我有一个很长的字符串行,长度为 176861(即带有一些符号的字母数字字符),我使用此函数测试了该行以进行研究: def getExe
list1= [u'%app%%General%%Council%', u'%people%', u'%people%%Regional%%Council%%Mandate%', u'%ppp%%Ge
这个问题在这里已经有了答案: Is it Pythonic to use list comprehensions for just side effects? (7 个答案) 关闭 4 个月前。 告
我想用 Python 将两个列表组合成一个列表,方法如下: a = [1,1,1,2,2,2,3,3,3,3] b= ["Sun", "is", "bright", "June","and" ,"Ju
我正在运行带有最新 Boost 发行版 (1.55.0) 的 Mac OS X 10.8.4 (Darwin 12.4.0)。我正在按照说明 here构建包含在我的发行版中的教程 Boost-Pyth
学习 Python,我正在尝试制作一个没有任何第 3 方库的网络抓取工具,这样过程对我来说并没有简化,而且我知道我在做什么。我浏览了一些在线资源,但所有这些都让我对某些事情感到困惑。 html 看起来
我是一名优秀的程序员,十分优秀!