python 實現兩個變量值進行交換的n種操作

python 兩個變量值交換

方法一:

c = 0
c= a
a= b
b= c

方法二:

a,b = b,a 

這是python的基本方法

方法三:(隻用兩個變量實現)

a = a+b
b = a-b
a = a-b

python兩個數值互換(淺析a,b=b,a原理)

python交換兩個值得方法非常簡單,即a,b=b,a,一步操作就交換瞭兩個值,那麼這是為什麼呢?

真相:

Python的變量並不直接存儲值,而隻是引用一個內存地址,交換變量時,隻是交換瞭引用的地址。

先看下面這段程序:

import dis

def func(a,b):
    a,b=b,a
    print(a,b)
    
a=10
b=20
func(a,b)
dis.dis(func)

一般來說一個Python語句會對應若幹字節碼指令,Python的字節碼是一種類似匯編指令的中間語言,但是一個字節碼指令並不是對應一個機器指 令(二進制指令),而是對應一段C代碼,而不同的指令的性能不同,所以不能單獨通過指令數量來判斷代碼的性能,而是要通過查看調用比較頻繁的指令的代碼來 確認一段程序的性能。

一個Python的程序會有若幹代碼塊組成,例如一個Python文件會是一個代碼塊,一個類,一個函數都是一個代碼塊,一個代碼塊會對應一個運行的上下文環境以及一系列的字節碼指令。

dis的作用   

dis模塊主要是用來分析字節碼的一個內置模塊,經常會用到的方法是dis.dis([bytesource]),參數為一個代碼塊,可以得到這個代碼塊對應的字節碼指令序列。

代碼輸出結果

這裡寫圖片描述

其中隻看前面為12的結果就行瞭(在我的編譯器裡,交換的那一行代碼在第12行)

可以看出主要是ROT_TWO指令的功勞:

查閱python文檔可以知道有ROT_TWO (源碼1398行),ROT_THREE(源碼1406行), ROT_FOUR這樣的指令,可以直接

交換兩個變量、三個變量、四個變量的值

在python3.4的源碼中查閱ceval.c文件可以看到:

TARGET(ROT_TWO) {           
    PyObject *top = TOP();          
    PyObject *second = SECOND();       
    SET_TOP(second);          
    SET_SECOND(top);         
    FAST_DISPATCH();       
}        
TARGET(ROT_THREE) {            
	PyObject *top = TOP();            
	PyObject *third = THIRD();            
	SET_SECOND(third);            
	FAST_DISPATCH();        
}

附:python值的交換

變量的每一次初始化,都開辟瞭一個新的空間,將新內容的地址賦值給變量。對於下圖來說,我們重復的給str1賦值,其實在內存中的變化如圖:

這裡寫圖片描述

從上圖我們可以看出,str1在重復的初始化過程中,是因為str1中存儲的元素地址由’hello world’的地址變成瞭’new hello world’的。

對於復雜的數據類型來說,改變其內部的值對於變量的影響:

這裡寫圖片描述

這裡寫圖片描述

當對列表中的元素進行一些增刪改的操作的時候,是不會影響到lst1列表本身對於整個列表地址的,隻會改變其內部元素的地址引用。

可是當我們對於一個列表重新初始化(賦值)的時候,就給lst1這個變量重新賦予瞭一個地址,覆蓋瞭原本列表的地址,這個時候,lst1列表的內存id就發生瞭改變。上面這個道理用在所有復雜的數據類型中都是一樣的。

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: