以下のエディターでは、Python の可変オブジェクトと不変オブジェクトについて簡単に説明します。編集者はこれがとても良いものだと思ったので、皆さんの参考として今から共有します。エディターに従って、可変/不変オブジェクトとは何かを見てみましょう
不変オブジェクト。オブジェクトが指すメモリ内の値は変更できません。 変数が変更されると、それが参照する値は変更できないため、元の値をコピーして変更することと同じになり、新しいアドレスが開かれ、変数はこの新しいアドレスを指します。
Mutable オブジェクト。オブジェクトが指すメモリ内の値は変更できます。変数 (正確には、reference) が変更された後、それが参照する値は実際には直接変更され、コピー動作はなく、新しい送信アドレスが開かれることはありません。
Python では、数値型 (int および float)、
stringsstr、およびタプルはすべて不変型です。リスト、ディクショナリ、セットは変数タイプです。 コードを見るとより直感的です。まず不変オブジェクトの例を見てみましょう
まず、 が 2 つのオブジェクトの ID が同じかどうかを判断するのに対し、 == は内容が同じかどうかを判断することを説明しましょう。
a = 2 b = 2 c = a + 0 c += 0 print(id(a), id(b), id(2)) # id都相同 print(c is b) #True文字列
astr = 'good' bstr = 'good' cstr = astr + '' print(cstr is bstr) # True print(id(astr), id(bstr), id('good')) # 三个id相同を見てみましょう。数値型の結果は同じです。以下の状況が発生した場合、その変数は変更後はもうダメです
astr = 'good' print(id(astr)) astr += 'aa' print(id(astr)) # id和上面的不一样。それは不変オブジェクトであるため、その変数に対応するメモリの値は変更できません。変数を変更する場合、実際には元の値がコピーされてから変更され、新しいアドレスが開かれ、astr はこの新しいアドレスを指します(したがって、前と前の astr の ID は異なります)。元の astr は異なります。別のオブジェクトがそれを指している場合、それはガベージ コレクションされるからです。これはint型でもfloat型でも同様です。
もう一度タプルを見てください
add = (1, 2, 3) aee = (1, 2, 3) print(id(add), id(aee), id((1, 2, 3))) # id各不相同 aee = (1, 2, 3) print(id(aee)) aee += () # 加空元组 print(id(aee)) # id变了! print(aee) #(1 ,2,3)それらはすべて (1, 2, 3) であるように見えますが、上記と一致しているはずです。これは可変オブジェクトですか?もう一度見てください
add = (1, 2, 3) aee = add print(id(aee), id(add)) # 这两个id一样 aee += (4, 5, 6) print(id(aee)) # aee的id变了! print(add) # add还是(1, 2, 3)没有变は numeric 型と str 型と一致しています。変数オブジェクト add = aee であれば、同じアドレス (同じ ID) を指していることは確実です。ただし、これは同じオブジェクトへの異なる参照ではありません。そうである場合、aee の変更により add の変更が発生しますが、タプルの場合はそうではありません。したがって、タプルは不変オブジェクトですが、str 型や数値型とは少し異なります。タプルの不変性は、通常、タプルに格納されている値が変更できないことを意味します (タプルに格納されたリストなど、一部の特殊なケースでは、リスト内の要素は変更できます。ただし、実際には、タプルは変更されていません)かわった)。
str、int、および float の場合、同じ型と同じ値を持つ限り、それらの ID は同じになります。 (なぜ同じ種類だと言えるのですか?)
a = 2.0 b = 2 print(a is b) # False, 一个int一个float,类型都不同2 と 2.0 は同じアドレスにありません。
可変オブジェクトの例
lis = [1, 2, 3] lis2 = [1, 2, 3] # 虽然它们的内容一样,但是它们指向的是不同的内存地址 print(lis is lis2) print(id(lis), id(lis2), id([1, 2, 3])) # 三个id都不同文の割り当てを見てください
alist = [1, 2, 3] # alist实际上是对对象的引用,blist = alist即引用的传递,现在两个引用都指向了同一个对象(地址) blist = alist print(id(alist), id(blist)) # id一样 # 所以其中一个变化,会影响到另外一个 blist.append(4) print(alist) # 改变blist, alist也变成了[1 ,2 ,3 4] print(id(alist), id(blist)) # id一样,和上面值没有改变时候的id也一样blist = alist。 alist は実際にはオブジェクトへの参照であり、blist = alist は参照の転送であり、両方の参照が同じオブジェクト (アドレス) を指すようになります。したがって、1 つの変更は他の変更にも影響します
上のリストの例と一致するセットを見てください
abb = {1, 2, 3} acc = abb print(id(abb), id(acc)) acc.add(4) print(abb) # {1, 2, 3, 4} print(id(abb), id(acc)) # 相等。
指定されたオブジェクトは変更できるため、コピーを作成して変更する必要はなく、その場で直接変更できるため、新しいメモリが開かれることはなく、変更の前後で ID は変わりません。
もちろん、これは不変オブジェクトの場合には当てはまりません
abc = 3 dd = abc dd = 43 print(abc) # 3,并不随dd的改变而改变しかし、コピーの場合は、コンテンツをコピーするだけであり、参照せずに渡すことはありません。これは、元のリストを変更せずにリストの値を使用したい場合に特に便利です。
関数パラメータとしての
blist = alist[:] # or alist.copy() print(alist is blist) # False blist.append(4) print(alist) # 还是[1,2 ,3]没有变化
は、関数パラメータと同じです。変数型は参照を渡し、不変型はコンテンツを渡します。
test_list = [1, 2, 3, 4] test_str = 'HAHA' def change(alist): alist.append(5) def not_change(astr): astr.lower() change(test_list) not_change(test_str) print(test_list) # 改变了原来的值 print(test_str) # 没有变
もちろん、元のリストの値を変更したくない場合は、パラメータを列変数のコピーで渡すことができます。 alsit[:]
興味深い例
a1 = [1, 2, 3] a2 = a1 print(id(a1), id(a2)) # 实际上是a2指向了新的对象,id已经改变。 # 所以现在a2、a1并不是同一对象的两个引用了,a2变化a1不会改变 a2 = a2 + [4] # 这个等式中,右边的a2还是和a1的id一样的,一旦赋值成功,a2就指向新的对象 print(id(1), id(a2)) # 不等,a2的id变化了 print(a1) # [1, 2, 3]没有变
このように書くと
a1 = [1, 2, 3] a2 = a1 print(id(a1), id(a2)) a2 += [4] # 相当于调用了a2.extend([4]),原地改变并没有新的对象产生 print(id(1), id(a2)) # 相等,a2的id没有变化 print(a1)
違いは、a2 += [4]、この文は a2.extend([4]) を呼び出すことと同等であり、配置され、新しいオブジェクトは生成されません。
以上がPythonの可変オブジェクトと不変オブジェクトの説明の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。