84

複数のデータベースを開いてその内容を比較するPythonスクリプトを作成しようとしています。そのスクリプトを作成する過程で、自分が作成したオブジェクトをコンテンツとするリストを作成する際に問題が発生しました。

この投稿のために、プログラムを必要最低限​​に簡略化しました。まず、新しいクラスを作成し、その新しいインスタンスを作成し、それに属性を割り当ててから、リストに書き込みます。次に、インスタンスに新しい値を割り当て、それをリストに書き込みます...そして何度も...

問題は、それは常に同じオブジェクトなので、実際にはベースオブジェクトを変更しているだけです。リストを読むと、同じオブジェクトが何度も繰り返されます。

では、ループ内のリストにオブジェクトをどのように書き込むのでしょうか。

これが私の簡略化されたコードです

class SimpleClass(object):
    pass

x = SimpleClass
# Then create an empty list
simpleList = []
#Then loop through from 0 to 3 adding an attribute to the instance 'x' of SimpleClass
for count in range(0,4):       
    # each iteration creates a slightly different attribute value, and then prints it to
# prove that step is working
# but the problem is, I'm always updating a reference to 'x' and what I want to add to
# simplelist is a new instance of x that contains the updated attribute

x.attr1= '*Bob* '* count
print "Loop Count: %s Attribute Value %s" % (count, x.attr1)
simpleList.append(x)

print '-'*20
# And here I print out each instance of the object stored in the list 'simpleList'
# and the problem surfaces.  Every element of 'simpleList' contains the same      attribute value

y = SimpleClass
print "Reading the attributes from the objects in the list"
for count in range(0,4):
    y = simpleList[count]
    print y.attr1

では、simpleListの要素を(追加、拡張、コピーなど)どのようにして、すべてのエントリが同じオブジェクトを指すのではなく、オブジェクトの異なるインスタンスを含むようにするのでしょうか。

4

6 に答える 6

72

あなたは根本的な誤解を示しています。

SimpleClassのインスタンスを呼び出さなかったため、インスタンスを作成したことはありません。

for count in xrange(4):
    x = SimpleClass()
    x.attr = count
    simplelist.append(x)

または、クラスにパラメーターを取得させる場合は、代わりにリスト内包表記を使用できます。

simplelist = [SimpleClass(count) for count in xrange(4)]
于 2008-12-07T22:22:39.777 に答える
58

クラスの個別のインスタンスでリストを埋めるには、リストの宣言でforループを使用できます。* multiplyは、各コピーを同じインスタンスにリンクします。

instancelist = [ MyClass() for i in range(29)]

次に、リストのインデックスを介してインスタンスにアクセスします。

instancelist[5].attr1 = 'whamma'
于 2008-12-09T11:50:39.380 に答える
12

属性に基づいてデータを出力するために単純に使用している場合は、一部の人が示唆しているように、SimpleClass オブジェクトを毎回再作成する必要はありません。ただし、実際にはクラスのインスタンスを作成しているわけではありません。クラス オブジェクト自体への参照を作成しているだけです。したがって、同じクラス属性への参照を (インスタンス属性ではなく) リストに何度も追加しています。

それ以外の:

x = SimpleClass

必要なもの:

x = SimpleClass()
于 2008-12-07T23:06:58.737 に答える
5

同じインスタンスの状態を継続的に変更するのではなく、毎回新しいインスタンスを作成します。新しいインスタンスはそれぞれ正しい状態になります。

または、元のオブジェクトではなく、明示的に作成されたオブジェクトのコピー(このページのヒントを使用)を各ステップで保存します。

于 2008-12-07T22:22:32.600 に答える
3

私があなたの質問を正しく理解しているなら、あなたはオブジェクトのディープコピーを実行する方法を尋ねます。copy.deepcopyを使用するのはどうですか?

import copy

x = SimpleClass()

for count in range(0,4):
  y = copy.deepcopy(x)
  (...)
  y.attr1= '*Bob* '* count

ディープコピーは、オブジェクト全体の再帰的なコピーです。詳細については、Pythonのドキュメントを参照してください:https ://docs.python.org/2/library/copy.html

于 2008-12-07T22:25:21.067 に答える