pythonmain參數
『壹』 python 多重繼承,繼承的幾個父類都需要傳遞參數,怎麼在子類計算出父類傳遞的參數總和呢
運行你的代碼:出錯位置: c = C()
出錯結果:TypeError: __init__() missing 1 required positional argument: ' num1 '
先來看你的程序__main()__部分:a = A(2) 和 b = B(5) 這是類A和類B的一個實例。在python中實例變數是用於每個實例的唯一數據,這就說明你這里的傳遞參數2或者是5隻能用在實例化的 a 或者是 b 下才有作用。 那麼重點看c = C( ) ,c是類對象C的實例化,c 只能用自身實例變數才有用,因此前面的實例 a 下的變數 num1=2 和 實例 b 下的變數 num1=5對實例c是無用的。所以,出錯結果很明顯了缺少傳遞的位置參數了。這為什麼提示缺少1個位置參數呢?下面為你簡單講解一下吧。你也可以用內置方法__mro__() :查看方法或者屬性的調用路徑——print(類名.__mro__)
類C是多繼承類A和類B的,多繼承(不存在super()重寫方法下),類C的實例化c是怎麼工作的——對於實例c調用方法或屬性過程是這樣的:查找當前類C中是否存在,然後在多個父類中按照從左往右順序查找(類A中先查找,類B中後查找),只要在這個過程中找到就退出了,後面的就不再查找了。
好吧,給你分析一下你程序的過程:類A和類B中都有__init__()同一個方法,方法相同那首先就查找唄——先查找類C(沒有對__init__()進行修改,那就是跳過了),然後再去類A查找,好嘛這就找到了__init__(self, num1),找到了就退出了。所以這樣一看對類C進行實例化就需要傳遞一個參數給num1就夠了。你也可以交換繼承位置class C(B, A),這樣就是類C實例化需要傳遞一個參數給num2就夠了。這樣過程就清晰了。
好第三個問題來了:你類C中有兩個參數呀num1和num2,而實例化又僅需要一個參數就夠了,這樣就肯定會產生問題了。
不信你試試給c = C(2)產生錯誤:AttributeError: 'C' object has no attribute 'num2'
解決方法1:既然沒有屬性num2就在類C中刪掉就是了,然後c = C(2)就可以運行成功了。
解決方案2:類變數用於類的所有實例共享的屬性和方法。因此,要想共用這兩個變數num1和num2,就得讓搜索的時候不要進到類A和類B中前提下,將它們變成對應的類變數就可以了。第一個前提很好實現:在類C下 定義:def __init__(self) : pass 第二個條件也比較好實現:將類A或類B的 __init__(self, num) : X.num = num X為對應的類名。(說明:self表示類實例化對象,即self.num 表示實例變數;X表示類對象,則X.num表示類變數),這樣就可以共享類A和類B的變數了。
classA:
def__init__(self,num1):
A.num1=num1
classB:
def__init__(self,num2):
B.num2=num2
classC(A,B):
def__init__(self):
pass
defnum_sum(self):
returnself.num2+self.num1
if__name__=='__main__':
a=A(2)
b=B(5)
c=C()
print(c.num_sum())
『貳』 python如何運行程序並向程序傳入參數
參數個數使用len()函數來獲取import sys print sys.argv if __name__=='__main__': print "Program name", sys.argv[0] for i in range(1, len(sys.argv)): print "arg%d"%i,sys.argv[i]
注意調用的時候百必須使用
python 腳本名.py arg1 arg2
因為度win下面的命令解釋器認為.py不是可執行文件,直接調用文件的內時候,不會傳入參數,容如下面的代碼則不會傳入參數:
腳本名.py arg1 arg2
『叄』 python的參數如果是以字元串的形式組成,該怎麼處理變數的值
代碼如下:
def main(param):
param=param.replace(',',';')
exec(param)
print a
print b
print c
param = "a=1,b=2,c=3"
main(param)
==============================
exec()函數可以將字元串轉換為代碼來執行,但是python中,兩句語句間要用『;』來隔開而不是『,』。
所以叫加上一句 param=param.replace(',',';'),將逗號都替換為分號。
希望對你有所幫助!
『肆』 Python 函數中,參數是傳值,還是傳引用
首先還是應該科普下函數參數傳遞機制,傳值和傳引用是什麼意思?
函數參數傳遞機制問題在本質上是調用函數(過程)和被調用函數(過程)在調用發生時進行通信的方法問題。基本的參數傳遞機制有兩種:值傳遞和引用傳遞。
值傳遞(passl-by-value)過程中,被調函數的形式參數作為被調函數的局部變數處理,即在堆棧數森中開辟了內存空間以存放由主調函數放進來的實參的值,從而成為了實參的一個副本。值傳遞的特點是被調函數對形式參數的任何操作都是作為局部變數進行,不會影響主調函數的實參變數的值。
引用傳遞(pass-by-reference)過程中,被調函數的形式參數雖然也作為局部變數在堆棧中開辟了內存空間,但是這時存放的是由主調函數放進來的實參變數的地址。被調函數對形參的任何操作都被處理成間接定址,即通過堆棧中存放的地址訪問主調函數中的實參變數。正因為如此,被調函數對形參做的任何操作都影響了主調函數中的實參掘掘變數。
在python中實際又是怎麼樣的呢?
先看一個簡單的例子:
from ctypes import *
import os.path
import sys
def test(c):
print "test before "
print id(c)
c+=2
print "test after +"
print id(c)
return c
def printIt(t):
for i in range(len(t)):
print t[i]
if __name__=="__main__":
a=2
print "main before invoke test"
print id(a)
n=test(a)
print "main afterf invoke test"
print a
print id(a)
運行後結果如下:
>>>
main before invoke test
39601564
test before
39601564
test after +
39601540
main afterf invoke test
2
39601564
id函數可以獲得對象的內存地址.很明顯從上面例子可以看出,將a變數作為參數傳遞給了test函數,傳遞了a的一個引用,把a的地址傳遞過去了,所以在函數內獲取的變數C的地址跟變數a的地址是一樣的,但是在函數內,對C進行賦值運算,C的值從2變成了4,實際上2和4所佔的內存空間都還是存判畢核在的,賦值運算後,C指向4所在的內存。而a仍然指向2所在的內存,所以後面列印a,其值還是2.
如果還不能理解,先看下面例子
>>> a=1
>>> b=1
>>> id(a)
40650152
>>> id(b)
40650152
>>> a=2
>>> id(a)
40650140
a和b都是int類型的值,值都是1,而且內存地址都是一樣的,這已經表明了在python中,可以有多個引用指向同一個內存(畫了一個很挫的圖,見諒),在給a賦值為2後,再次查看a的內存地址,都已經變化了
而基於最前面的例子,大概可以這樣描述:
那python函數傳參就是傳引用?然後傳參的值在被調函數內被修改也不影響主調函數的實參變數的值?再來看個例子。
from ctypes import *
import os.path
import sys
def test(list2):
print "test before "
print id(list2)
list2[1]=30
print "test after +"
print id(list2)
return list2
def printIt(t):
for i in range(len(t)):
print t[i]
if __name__=="__main__":
list1=["loleina",25,'female']
print "main before invoke test"
print id(list1)
list3=test(list1)
print "main afterf invoke test"
print list1
print id(list1)
實際值為:
>>>
main before invoke test
64129944
test before
64129944
test after +
64129944
main afterf invoke test
['loleina', 30, 'female']
64129944
發現一樣的傳值,而第二個變數居然變化,為啥呢?
實際上是因為python中的序列:列表是一個可變的對象,就基於list1=[1,2] list1[0]=[0]這樣前後的查看list1的內存地址,是一樣的。
>>> list1=[1,2]
>>> id(list1)
64185208
>>> list1[0]=[0]
>>> list1
[[0], 2]
>>> id(list1)
64185208
結論:python不允許程序員選擇採用傳值還是傳引用。Python參數傳遞採用的肯定是「傳對象引用」的方式。這種方式相當於傳值和傳引用的一種綜合。如果函數收到的是一個可變對象(比如字典或者列表)的引用,就能修改對象的原始值--相當於通過「傳引用」來傳遞對象。如果函數收到的是一個不可變對象(比如數字、字元或者元組)的引用,就不能直接修改原始對象--相當於通過「傳值'來傳遞對象。
分類: python 基礎語法
『伍』 為什麼電腦上的PYTHON運行main指令會報錯
if name == '__main__': 相當於一個新的函數,在這個函數里,你調用了main()函數,並傳入了args參數,但是在if name == '__main__': 這個函數域內,你並沒有定義args,所以報錯
『陸』 在Python中定義Main函數
目錄
許多編程語言都有一個特殊的函數,當操作系統開始運行程序時會自動執行該函數。這個函數通常被命名為main(),並且依據語言標准具有特定的返回類型和參數。另一方面,Python解釋器從文件頂部開始執行腳本,並且沒有自動執行的特殊函數。
盡管如此,為程序的執行定義一個起始點有助於理解程序是如何運行的。Python程序員提出了幾種方式對此進行實現。
本文結束時,您將了解以下內容:
Python中的基本main()函數
一些Python腳本中,包含一個函數定義和一個條件語句,如下所示:
此代碼中,包含一個main()函數,在程序執行時列印Hello World!。此外,還包含一個條件(或if)語句,用於檢查__name__的值並將其與字元串"__main__"進行比較。當if語句為True時,Python解釋器將執行main()函數。更多關於Python條件語句的信息可以由此獲得。
這種代碼模式在Python文件中非常常見,它將作為腳本執行並導入另一個模塊。為了幫助理解這段代碼的執行方式,首先需要了解Python解釋器如何根據代碼的執行方式設置__name__。
Python中的執行模式
Python解釋器執行代碼有兩種方式:
更多內容可參考如何運行Python腳本。無論採用哪種方式,Python都會定義一個名為__name__的特殊變數,該變數包含一個字元串,其值取決於代碼的使用方式。
本文將如下示例文件保存為execution_methods.py,以 探索 代碼如何根據上下文改變行為:
在此文件中,定義了三個對print()函數的調用。前兩個列印一些介紹性短語。第三個print()會先列印短語The value __name__ is,之後將使用Python內置的repr()函數列印出__name__變數。
在Python中,repr()函數將對象轉化為供解釋器讀取的形式。上述示例通過使用repr()函數來強調__name__的值為字元串。更多關於repr()的內容可參考Python文檔。
在本文中,您將隨處可見文件(file),模塊(mole)和腳本(script)這三個字眼。實際上,三者之間並無太大的差別。不過,在強調代碼目的時,還是存在細微的差異:
「如何運行Python腳本」一文也討論了三者的差別。
基於命令行執行
在這類方法中,Python腳本將通過命令行來執行。
執行腳本時,無法與Python解釋器正在執行的代碼交互。關於如何通過命令行執行代碼的詳細信息對本文而言並不重要,但您可以通過展開下框閱讀更多有關Windows,Linux和macOS之間命令行差異的內容。
命令行環境
不同的操作系統在使用命令行執行代碼時存在細微的差異。
在Linux和macOS中,通常使用如下命令:
美元符號($)之前的內容可能有所不同,具體取決於您的用戶名和計算機名稱。您鍵入的命令位於$之後。在Linux或macOS上,Python3的可執行文件名為python3,因此可以通過輸入python3 script_name.py來運行python腳本。
在Windows上,命令提示符通常如下所示:
根據您的用戶名,>之前的內容可能會有所不同,您輸入的命令位於>之後。在Windows上,Python3的可執行文件通常為python。因此可以通過輸入python script_name.py來運行python腳本。
無論哪種操作系統,本文的Python腳本的輸出結果都是相同的。因此本文以Linux和macOS為例。
使用命令行執行execution_methods.py,如下所示:
在這個示例中,__name__具有值'__main__',其中引號(')表明該值為字元串類型。
請記住,在Python中,使用單引號(')和雙引號(")定義的字元串沒有區別。更多關於字元串的內容請參考Python的基本數據類型。
如果在腳本中包含"shebang行"並直接執行它(./execution_methods.py),或者使用IPython或Jupyter Notebook的%run,將會獲取相同的結果。
您還可以通過向命令行添加-m參數的方法實現以模塊的方式執行。通常情況下,推薦如下方式pip: python3 -m pip install package_name。
添加-m參數將會運行包中__main__.py的代碼。更多關於__main__.py文件的內容可參考如何將開源Python包發布到PyPI中。
在三種情況中,__name__都具有相同的值:字元串'__main__'。
技術細節:Python文檔中具體定義了__name__何時取值為'__main__'。
當通過標准輸入,腳本或者交互提示中讀取數據時,模塊的__name__將取值為'__main__'。(來源)
__name__與__doc__,__package__和其他屬性一起存儲在模塊的全局命名空間。更多關於屬性的信息可參考Python數據模型文檔,特別是關於模塊和包的信息,請參閱Python Import文檔。
導入模塊或解釋器
接下來是Python解釋器執行代碼的第二種方式:導入。在開發模塊或腳本時,可以使用import關鍵字導入他人已經構建的模塊。
在導入過程中,Python執行指定模塊中定義的語句(但僅在第一次導入模塊時)。要演示導入execution_methods.py文件的結果,需要啟動Python解釋器,然後導入execution_methods.py文件:
在此代碼輸出中,Python解釋器執行了三次print()函數調用。前兩行由於沒有變數,在輸出方面與在命令行上作為腳本執行時完全相同。但是第三個輸出存在差異。
當Python解釋器導入代碼時,__name__的值與要導入的模塊的名稱相同。您可以通過第三行的輸出了解這一點。__name__的值為'execution_methods',是Python導入的.py文件。
注意如果您在沒有退出Python時再次導入模塊,將不會有輸出。
注意:更多關於導入在Python中如何工作的內容請參考官方文檔和Python中的絕對和相對導入。
Main函數的最佳實踐
既然您已經了解兩種執行方式上的差異,那麼掌握一些最佳實踐方案還是很有用的。它們將適用於編寫作為腳本運行的代碼或者在另一個模塊導入的代碼。
如下是四種實踐方式:
將大部分代碼放入函數或類中
請記住,Python解釋器在導入模塊時會執行模塊中的所有代碼。有時如果想要實現用戶可控的代碼,會導致一些副作用,例如:
在這種情況下,想要實現用戶控制觸發此代碼的執行,而不是讓Python解釋器在導入模塊時執行代碼。
因此,最佳方法是將大部分代碼包含在函數或類中。這是因為當Python解釋器遇到def或class關鍵字時,它只存儲這些定義供以後使用,並且在用戶通知之前不會實際執行。
將如下代碼保存在best_practices.py以證明這個想法:
在此代碼中,首先從time模塊中導入sleep()。
在這個示例中,參數以秒的形式傳入sleep()函數中,解釋器將暫停一段時間再運行。隨後,使用print()函數列印關於代碼描述的語句。
之後,定義一個process_data()函數,執行如下五項操作:
在命令行中執行
當你將此文件作為腳本用命令行執行時會發生什麼呢?
Python解釋器將執行函數定義之外的from time import sleep和print(),之後將創建函數process_data()。然後,腳本將退出而不做任何進一步的操作,因為腳本沒有任何執行process_data()的代碼。
如下是這段腳本的執行結果:
我們在這里看到的輸出是第一個print()的結果。注意,從time導入和定義process_data()函數不產生結果。具體來說,調用定義在process_data()內部的print()不會列印結果。
導入模塊或解釋器執行
在會話(或其他模塊)中導入此文件時,Python解釋器將執行相同的步驟。
Python解釋器導入文件後,您可以使用已導入模塊中定義的任何變數,類或函數。為了證明這一點,我們將使用可交互的Python解釋器。啟動解釋器,然後鍵入import best_practices:
導入best_practices.py後唯一的輸出來自process_data()函數外定義的print()。導入模塊或解釋器執行與基於命令行執行類似。
使用__name__控制代碼的執行
如何實現基於命令行而不使用Python解釋器導入文件來執行呢?
您可以使用__name__來決定執行上下文,並且當__name__等於"__main__"時才執行process_data()。在best_practices.py文件中添加如下代碼:
這段代碼添加了一個條件語句來檢驗__name__的值。當值為"__main__"時,條件為True。記住當__name__變數的特殊值為"__main__"時意味著Python解釋器會執行腳本而不是將其導入。
條件語塊內添加了四行代碼(第12,13,14和15行):
現在,在命令行中運行best_practices.py,並觀察輸出的變化:
首先,輸出顯示了process_data()函數外的print()的調用結果。
之後,data的值被列印。因為當Python解釋器將文件作為腳本執行時,變數__name__具有值"__main__",因此條件語句被計算為True。
接下來,腳本將調用process_data()並傳入data進行修改。當process_data執行時,將輸出一些狀態信息。最終,將輸出modified_data的值。
現在您可以驗證從解釋器(或其他模塊)導入best_practices.py後發生的事情了。如下示例演示了這種情況:
注意,當前結果與將條件語句添加到文件末尾之前相同。因為此時__name__變數的值為"best_practices",因此條件語句結果為False,Python將不執行process_data()。
創建名為main()的函數來包含要運行的代碼
現在,您可以編寫作為腳本由從命令行執行並導入且沒有副作用的Python代碼。接下來,您將學習如何編寫代碼並使其他程序員能輕松地理解其含義。
許多語言,如C,C++,Java以及其他的一些語言,都會定義一個叫做main()的函數,當編譯程序時,操作系統會自動調用該函數。此函數通常被稱為入口點(entry point),因為它是程序進入執行的起始位置。
相比之下,Python沒有一個特殊的函數作為腳本的入口點。實際上在Python中可以將入口點定義成任何名稱。
盡管Python不要求將函數命名為main(),但是最佳的做法是將入口點函數命名為main()。這樣方便其他程序員定位程序的起點。
此外,main()函數應該包含Python解釋器執行文件時要運行的任何代碼。這比將代碼放入條件語塊中更好,因為用戶可以在導入模塊時重復使用main()函數。
修改best_practices.py文件如下所示:
在這個示例中,定義了一個main()函數,它包含了上面的條件語句塊。之後修改條件語塊執行main()。如果您將此代碼作為腳本運行或導入,將獲得與上一節相同的輸出。
在main()中調用其他函數
另一種常見的實現方式是在main()中調用其他函數,而不是直接將代碼寫入main()。這樣做的好處在於可以實現將幾個獨立運行的子任務整合。
例如,某個腳本有如下功能:
如果在單獨的函數中各自實現這些子任務,您(或其他用戶)可以很容易地實現代碼重用。之後您可以在main()函數中創建默認的工作流。
您可以根據自己的情況選擇是否使用此方案。將任務拆分為多個函數會使重用更容易,但會增加他人理解代碼的難度。
修改best_practices.py文件如下所示:
在此示例代碼中,文件的前10行具有與之前相同的內容。第12行的第二個函數創建並返回一些示例數據,第17行的第三個函數模擬將修改後的數據寫入資料庫。
第21行定義了main()函數。在此示例中,對main()做出修改,它將調用數據讀取,數據處理以及數據寫入等功能。
首先,從read_data_from_web()中創建data。將data作為參數傳入process_data(),之後將返回modified_data。最後,將modified_data傳入write_data_to_database()。
腳本的最後兩行是條件語塊用於驗證__name__,並且如果if語句為True,則執行main()。
在命令行中運行如下所示:
根據執行結果,Python解釋器在執行main()函數時,將依次執行read_data_from_web(),process_data()以及write_data_to_database()。當然,您也可以導入best_practices.py文件並重用process_data()作為不同的數據輸入源,如下所示:
在此示例中,導入了best_practices並且將其簡寫為bp。
導入過程會導致Python解釋器執行best_practices.py的全部代碼,因此輸出顯示解釋文件用途的信息。
然後,從文件中存儲數據而不是從Web中讀取數據。之後,可以重用best_practices.py文件中的process_data()和write_data_to_database()函數。在此情況下,可以利用代碼重寫來取代在main()函數中實現全部的代碼邏輯。
實踐總結
以下是Python中main()函數的四個關鍵最佳實踐:
結論
恭喜!您現在已經了解如何創建Python main()函數了。
本文介紹了如下內容:
現在,您可以開始編寫一些非常棒的關於Python main()函數代碼啦!
『柒』 python將類作為參數傳遞
mport sys print sys.argv[1]#保存為main.py#在控制台下輸入 python main.py "hello"#就有hello列印出來了 前提是你配置好了環境變數