Ok。私がまさに探していたものは存在しないようです。そこで、この問題の回避策である解決策を見つけました。
メモリをプロファイリングする代わりに、オブジェクトをプロファイリングします。このようにして、プログラム内の特定の時間にいくつのオブジェクトが存在するかを確認できます。目標を達成するために、既存のコードへの変更を最小限に抑えてメタクラスを利用しました。
次のメタクラスは、非常に単純なサブルーチン__init__
と__del__
クラスの関数を追加します。サブルーチン for__init__
は、そのクラス名を持つオブジェクトの数を 1 つ増やし、 は 1__del__
つ減らします。
class ObjectProfilerMeta(type):
#Just set metaclass of a class to ObjectProfilerMeta to profile object
def __new__(cls, name, bases, attrs):
if name.startswith('None'):
return None
if "__init__" in attrs:
attrs["__init__"]=incAndCall(name,attrs["__init__"])
else:
attrs["__init__"]=incAndCall(name,dummyFunction)
if "__del__" in attrs:
attrs["__del__"]=decAndCall(name,attrs["__del__"])
else:
attrs["__del__"]=decAndCall(name,dummyFunction)
return super(ObjectProfilerMeta, cls).__new__(cls, name, bases, attrs)
def __init__(self, name, bases, attrs):
super(ObjectProfilerMeta, self).__init__(name, bases, attrs)
def __add__(self, other):
class AutoClass(self, other):
pass
return AutoClass
incAndCall および decAndCall 関数は、それらが存在するモジュールのグローバル変数を使用します。
counter={}
def incAndCall(name,func):
if name not in counter:
counter[name]=0
def f(*args,**kwargs):
counter[name]+=1
func(*args,**kwargs)
return f
def decAndCall(name,func):
if name not in counter:
counter[name]=0
def f(*args,**kwargs):
counter[name]-=1
func(*args,**kwargs)
return f
def dummyFunction(*args,**kwargs):
pass
dummyFunction は非常に単純な回避策です。それを行うためのはるかに良い方法があると確信しています。
最後に、存在するオブジェクトの数を確認したい場合は、カウンター ディクショナリを確認するだけです。例;
>>> class A:
__metaclass__=ObjectProfilerMeta
def __init__(self):
pass
>>> class B:
__metaclass__=ObjectProfilerMeta
>>> l=[]
>>> for i in range(117):
l.append(A())
>>> for i in range(18):
l.append(B())
>>> counter
{'A': 117, 'B': 18}
>>> l.pop(15)
<__main__.A object at 0x01210CB0>
>>> counter
{'A': 116, 'B': 18}
>>> l=[]
>>> counter
{'A': 0, 'B': 0}
これがお役に立てば幸いです。私の場合はそれで十分でした。