国产一区二区精品-国产一区二区精品久-国产一区二区精品久久-国产一区二区精品久久91-免费毛片播放-免费毛片基地

千鋒教育-做有情懷、有良心、有品質的職業教育機構

手機站
千鋒教育

千鋒學習站 | 隨時隨地免費學

千鋒教育

掃一掃進入千鋒手機站

領取全套視頻
千鋒教育

關注千鋒學習站小程序
隨時隨地免費學習課程

當前位置:首頁  >  技術干貨  > Python函數傳參是傳值還是傳址

Python函數傳參是傳值還是傳址

來源:千鋒教育
發布人:syq
時間: 2022-09-07 17:19:06 1662542346

  今天在編寫python粒子群尋優時,在定義計算適應度函數中,我將適應度先初始化為每個粒子位置X(ndarray)的第一個參數,然后通過索引使一些粒子的適應度改為-1000。

  但是發現結果總是不收斂,而且跳來跳去,仔細調試了一會才發現問題出在傳參上,還是基礎掌握的不牢固。

  在fitness = X[:, 0]這一句中,python實際上并未分配新的內存空間給變量fitness,而是將fitness作為引用指向X[:, 0]所在內存地址,于是在通過索引修改fitness的值時,粒子的位置X也受到修改,因此造成結果不對。

  所以在用numpy中的深拷貝.copy()代替直接賦值=后,即fitness = (X[:, 0]).copy(),問題得到解決。

  一般來說:

  如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值。

  如果函數收到的是一個不可變對象(比如數字、字符或者元組)的引用,就不能直接修改原始對象。

  通過搜索:

在python中,不可變對象是共享的,創建可變對象永遠是分配新地址

  有以下例子供思考:

  我們將a, b都賦值為1,可以看到a, b的地址相同,這時如果我們將b+1:

  此時b與c的地址相同,即不可變對象的值是共享的。程序只會在第一次給2分配內存地址,之后再創建值為2的變量時,都會指向內存中同一個2,而不是重新分配一個內存地址寫入2。

  但是對于可變對象,即使兩個變量的值相同,其內存也會不同:

  我們可以觀察到,對于c = c + [3],程序重新分配了內存空間,符合創建可變對象永遠是分配新地址。而對列表a使用內置方法pop后,即使列表內容修改,地址也不會改變,不像第一個例子將b修改為2后b的地址也發生改變。同樣對于直接修改列表中的值也一樣,也就是我一開始碰到的問題。

  參考得:

  對于int, float, bool, str, tuple等不可變對象,變量值的改變本質上是該引用變量指向的改變,而原來那塊內存的值是不變的。

  可變對象的改變為原內存處值的改變,對于一個列表a,如果將其直接賦值給b,則a與b同時引用一個列表,一方改變,兩者同變。

  但是注意列表的這個改變應該限于內置方法以及通過索引修改列表內單元的值(?),就是因為創建可變對象永遠是分配新地址,如下面例子:

  總結:

我們沒有必要去討論函數傳參是傳值還是傳址,我們只需要注意傳給函數可變對象(比如字典或者列表)的引用,就會有破壞對象的原始值的風險,就像C++中將數組地址傳給函數一樣。其次我們需要記住:在python中,不可變對象是共享的,創建可變對象永遠是分配新地址!

所以最開始的那個問題,我們甚至可以通過fitness = X[:, 0] * 1來代替深拷貝.deep(),因為此時fitness已經指向新的地址。

tags:
聲明:本站稿件版權均屬千鋒教育所有,未經許可不得擅自轉載。
10年以上業內強師集結,手把手帶你蛻變精英
請您保持通訊暢通,專屬學習老師24小時內將與您1V1溝通
免費領取
今日已有369人領取成功
劉同學 138****2860 剛剛成功領取
王同學 131****2015 剛剛成功領取
張同學 133****4652 剛剛成功領取
李同學 135****8607 剛剛成功領取
楊同學 132****5667 剛剛成功領取
岳同學 134****6652 剛剛成功領取
梁同學 157****2950 剛剛成功領取
劉同學 189****1015 剛剛成功領取
張同學 155****4678 剛剛成功領取
鄒同學 139****2907 剛剛成功領取
董同學 138****2867 剛剛成功領取
周同學 136****3602 剛剛成功領取
相關推薦HOT