本文適合有經(jīng)驗(yàn)的程序員盡快進(jìn)入Python世界.特別地,如果你掌握J(rèn)ava和Javascript,不用1小時(shí)你就可以用Python快速流暢地寫(xiě)有用的Python程序.
假設(shè)我們有這么一項(xiàng)任務(wù):簡(jiǎn)單測(cè)試局域網(wǎng)中的電腦是否連通.這些電腦的ip范圍從192.168.0.101到192.168.0.200.
思路:用shell編程.(Linux通常是bash而Windows是批處理腳本).例如,在Windows上用ping ip 的命令依次測(cè)試各個(gè)機(jī)器并得到控制臺(tái)輸出.由于ping通的時(shí)候控制臺(tái)文本通常是"Reply from ... " 而不通的時(shí)候文本是"time out ... " ,所以,在結(jié)果中進(jìn)行字符串查找,即可知道該機(jī)器是否連通.
實(shí)現(xiàn):Java代碼如下:
Stringcmd="cmd.exeping";
Stringipprefix="192.168.10.";
intbegin=101;
intend=200;
Processp=null;
for(inti=begin;i
p=Runtime.getRuntime().exec(cmd+i);
Stringline=null;
BufferedReaderreader=newBufferedReader(newInputStreamReader(p.getInputStream()));
while((line=reader.readLine())!=null)
{
//Handlingline,maylogsit.
}
reader.close();
p.destroy();
}
這段代碼運(yùn)行得很好,問(wèn)題是為了運(yùn)行這段代碼,你還需要做一些額外的工作.這些額外的工作包括:
編寫(xiě)一個(gè)類(lèi)文件
編寫(xiě)一個(gè)main方法
將之編譯成字節(jié)代碼
由于字節(jié)代碼不能直接運(yùn)行,你需要再寫(xiě)個(gè)小小的bat或者bash腳本來(lái)運(yùn)行.
當(dāng)然,用C/C++同樣能完成這項(xiàng)工作.但C/C++不是跨平臺(tái)語(yǔ)言.在這個(gè)足夠簡(jiǎn)單的例子中也許看不出C/C++和Java實(shí)現(xiàn)的區(qū)別,但在一些更為復(fù)雜的場(chǎng)景,比如要將連通與否的信息記錄到網(wǎng)絡(luò)數(shù)據(jù)庫(kù).由于Linux和Windows的網(wǎng)絡(luò)接口實(shí)現(xiàn)方式不同,你不得不寫(xiě)兩個(gè)函數(shù)的版本.用Java就沒(méi)有這樣的顧慮.
同樣的工作用Python實(shí)現(xiàn)如下:
importsubprocess
cmd="cmd.exe"
begin=101
end=200
whilebegin
p=subprocess.Popen(cmd,shell=True,stdout=subprocess.PIPE,
stdin=subprocess.PIPE,
stderr=subprocess.PIPE)
p.stdin.write("ping192.168.1."+str(begin)+"/n")
p.stdin.close()
p.wait()
print"executionresult:%s"%p.stdout.read()
對(duì)比Java,Python的實(shí)現(xiàn)更為簡(jiǎn)潔,你編寫(xiě)的時(shí)間更快.你不需要寫(xiě)main函數(shù),并且這個(gè)程序保存之后可以直接運(yùn)行.另外,和Java一樣,Python也是跨平臺(tái)的.
有經(jīng)驗(yàn)的C/Java程序員可能會(huì)爭(zhēng)論說(shuō)用C/Java寫(xiě)會(huì)比Python寫(xiě)得快.這個(gè)觀(guān)點(diǎn)見(jiàn)仁見(jiàn)智.我的想法是當(dāng)你同時(shí)掌握J(rèn)ava和Python之后,你會(huì)發(fā)現(xiàn)用Python寫(xiě)這類(lèi)程序的速度會(huì)比Java快上許多.例如操作本地文件時(shí)你僅需要一行代碼而不需要Java的許多流包裝類(lèi).各種語(yǔ)言有其天然的適合的應(yīng)用范圍.用Python處理一些簡(jiǎn)短程序類(lèi)似與操作系統(tǒng)的交互編程工作最省時(shí)省力.
足夠簡(jiǎn)單的任務(wù),例如一些shell編程.如果你喜歡用Python設(shè)計(jì)大型商業(yè)網(wǎng)站或者設(shè)計(jì)復(fù)雜的游戲,悉聽(tīng)尊便.
2 快速入門(mén)
安裝完P(guān)ython之后(我本機(jī)的版本是2.5.4),打開(kāi)IDLE(Python GUI) , 該程序是Python語(yǔ)言解釋器,你寫(xiě)的語(yǔ)句能夠立即運(yùn)行.我們寫(xiě)下一句著名的程序語(yǔ)句:
print"Hello,world!"
并按回車(chē).你就能看到這句被K&R引入到程序世界的名言.
在解釋器中選擇"File"--"New Window" 或快捷鍵 Ctrl+N , 打開(kāi)一個(gè)新的編輯器.寫(xiě)下如下語(yǔ)句:
print"Hello,world!"
raw_input("Pressenterkeytoclosethiswindow");
保存為a.py文件.按F5,你就可以看到程序的運(yùn)行結(jié)果了.這是Python的第二種運(yùn)行方式.
找到你保存的a.py文件,雙擊.也可以看到程序結(jié)果.Python的程序能夠直接運(yùn)行,對(duì)比Java,這是一個(gè)優(yōu)勢(shì).
我們換一種方式來(lái)問(wèn)候世界.新建一個(gè)編輯器并寫(xiě)如下代碼:
print"歡迎來(lái)到奧運(yùn)中國(guó)!"
raw_input("Pressenterkeytoclosethiswindow");
在你保存代碼的時(shí)候,Python會(huì)提示你是否改變文件的字符集,結(jié)果如下:
# -*- coding: cp936 -*-
print"歡迎來(lái)到奧運(yùn)中國(guó)!"
raw_input("Pressenterkeytoclosethiswindow");
將該字符集改為我們更熟悉的形式:
# -*- coding: GBK -*-
print"歡迎來(lái)到奧運(yùn)中國(guó)!" # 使用中文的例子
raw_input("Pressenterkeytoclosethiswindow");
程序一樣運(yùn)行良好.
用微軟附帶的計(jì)算器來(lái)計(jì)數(shù)實(shí)在太麻煩了.打開(kāi)Python解釋器,直接進(jìn)行計(jì)算:
a=100.0
b=201.1
c=2343
print(a+b+c)/c
可以如下打印出預(yù)定義輸出格式的字符串:
print"""
Usage:thingy[OPTIONS]
-hDisplaythisusagemessage
-HhostnameHostnametoconnectto
"""
字符串是怎么訪(fǎng)問(wèn)的?請(qǐng)看這個(gè)例子:
word="abcdefg"
a=word[2]
print"ais:"+a
b=word[1:3]
print"bis:"+b#index1and2elementsofword.
c=word[:2]
print"cis:"+c#index0and1elementsofword.
d=word[0:]
print"dis:"+d#Allelementsofword.
e=word[:2]+word[2:]
print"eis:"+e#Allelementsofword.
f=word[-1]
print"fis:"+f#Thelastelementsofword.
g=word[-4:-2]
print"gis:"+g#index3and4elementsofword.
h=word[-2:]
print"his:"+h#Thelasttwoelements.
i=word[:-2]
print"iis:"+i#Everythingexceptthelasttwocharacters
l=len(word)
print"Lengthofwordis:"+str(l)
請(qǐng)注意ASCII和UNICODE字符串的區(qū)別:
print"InputyourChinesename:"
s=raw_input("Pressentertobecontinued");
print"Yournameis:"+s;
l=len(s)
print"LengthofyourChinesenameinasccodesis:"+str(l);
a=unicode(s,"GBK")
l=len(a)
print"I'msorryweshoulduseunicodechar!CharactersnumberofyourChinese/
nameinunicodeis:"+str(l);
類(lèi)似Java里的List,這是一種方便易用的數(shù)據(jù)類(lèi)型:
word=['a','b','c','d','e','f','g']
a=word[2]
print"ais:"+a
b=word[1:3]
print"bis:"
printb#index1and2elementsofword.
c=word[:2]
print"cis:"
printc#index0and1elementsofword.
d=word[0:]
print"dis:"
printd#Allelementsofword.
e=word[:2]+word[2:]
print"eis:"
printe#Allelementsofword.
f=word[-1]
print"fis:"
printf#Thelastelementsofword.
g=word[-4:-2]
print"gis:"
printg#index3and4elementsofword.
h=word[-2:]
print"his:"
printh#Thelasttwoelements.
i=word[:-2]
print"iis:"
printi#Everythingexceptthelasttwocharacters
l=len(word)
print"Lengthofwordis:"+str(l)
print"Addsnewelement"
word.append('h')
printword
#Multi-waydecision
x=int(raw_input("Pleaseenteraninteger:"))
ifx<0:
x=0
print"Negativechangedtozero"
elifx==0:
print"Zero"
else:
print"More"
#LoopsList
a=['cat','window','defenestrate']
forxina:
printx,len(x)
#Defineandinvokefunction.
defsum(a,b):
returna+b
func=sum
r=func(5,6)
printr
#Definesfunctionwithdefaultargument
defadd(a,b=2):
returna+b
r=add(1)
printr
r=add(1,5)
printr
并且,介紹一個(gè)方便好用的函數(shù):
#Therange()function
a=range(5,10)
printa
a=range(-2,-7)
printa
a=range(-7,-2)
printa
a=range(-2,-11,-3)#The3rdparameterstandsforstep
printa
spath="D:/download/baa.txt"
f=open(spath,"w")#Opensfileforwriting.Createsthisfiledoesn'texist.
f.write("Firstline1./n")
f.writelines("Firstline2.")
f.close()
f=open(spath,"r")#Opensfileforreading
forlineinf:
printline
f.close()
s=raw_input("Inputyourage:")
ifs=="":
raiseException("Inputmustnobeempty.")
try:
i=int(s)
exceptValueError:
print"Couldnotconvertdatatoaninteger."
except:
print"Unknownexception!"
else:#Itisusefulforcodethatmustbeexecutedifthetryclausedoesnotraiseanexception
print"Youare%d"%i,"yearsold"
finally:#Cleanupaction
print"Goodbye!"
classBase:
def__init__(self):
self.data=[]
defadd(self,x):
self.data.append(x)
defaddtwice(self,x):
self.add(x)
self.add(x)
#ChildextendsBase
classChild(Base):
defplus(self,a,b):
returna+b
oChild=Child()
oChild.add("str1")
printoChild.data
printoChild.plus(2,3)
每一個(gè).py文件稱(chēng)為一個(gè)module,module之間可以互相導(dǎo)入.請(qǐng)參看以下例子:
#a.py
defadd_func(a,b):
returna+b
#b.py
fromaimportadd_func#Alsocanbe:importa
print"Importadd_funcfrommodulea"
print"Resultof1plus2is:"
printadd_func(1,2)#Ifusing"importa",thenhereshouldbe"a.add_func"
module可以定義在包里面.Python定義包的方式稍微有點(diǎn)古怪,假設(shè)我們有一個(gè)parent文件夾,該文件夾有一個(gè)child子文件夾.child中有一個(gè)module a.py . 如何讓Python知道這個(gè)文件層次結(jié)構(gòu)?很簡(jiǎn)單,每個(gè)目錄都放一個(gè)名為_(kāi)init_.py 的文件.該文件內(nèi)容可以為空.這個(gè)層次結(jié)構(gòu)如下所示:
parent
--__init_.py
--child
--__init_.py
--a.py
b.py
那么Python如何找到我們定義的module?在標(biāo)準(zhǔn)包sys中,path屬性記錄了Python的包路徑.你可以將之打印出來(lái):
importsys
printsys.path
通常我們可以將module的包路徑放到環(huán)境變量PYTHONPATH中,該環(huán)境變量會(huì)自動(dòng)添加到sys.path屬性.另一種方便的方法是編程中直接指定我們的module路徑到sys.path 中:
importsys
sys.path.append('D://download')
fromparent.child.aimportadd_func
printsys.path
print"Importadd_funcfrommodulea"
print"Resultof1plus2is:"
printadd_func(1,2)
總結(jié)
你會(huì)發(fā)現(xiàn)這個(gè)教程相當(dāng)?shù)暮?jiǎn)單.許多Python特性在代碼中以隱含方式提出,這些特性包括:Python不需要顯式聲明數(shù)據(jù)類(lèi)型,關(guān)鍵字說(shuō)明,字符串函數(shù)的解釋等等.我認(rèn)為一個(gè)熟練的程序員應(yīng)該對(duì)這些概念相當(dāng)了解,這樣在你擠出寶貴的一小時(shí)閱讀這篇短短的教程之后,你能夠通過(guò)已有知識(shí)的遷移類(lèi)比盡快熟悉Python,然后盡快能用它開(kāi)始編程.
當(dāng)然,1小時(shí)學(xué)會(huì)Python頗有嘩眾取寵之嫌.確切的說(shuō),編程語(yǔ)言包括語(yǔ)法和標(biāo)準(zhǔn)庫(kù).語(yǔ)法相當(dāng)于武術(shù)招式,而標(biāo)準(zhǔn)庫(kù)應(yīng)用實(shí)踐經(jīng)驗(yàn)則類(lèi)似于內(nèi)功,需要長(zhǎng)期鍛煉.Python學(xué)習(xí)了Java的長(zhǎng)處,提供了大量極方便易用的標(biāo)準(zhǔn)庫(kù)供程序員"拿來(lái)主義".(這也是Python成功的原因),在開(kāi)篇我們看到了Python如何調(diào)用Windows cmd的例子,以后我會(huì)盡量寫(xiě)上各標(biāo)準(zhǔn)庫(kù)的用法和一些應(yīng)用技巧,讓大家真正掌握Python.
但不管怎樣,至少你現(xiàn)在會(huì)用Python代替繁瑣的批處理寫(xiě)程序了.希望那些真的能在一小時(shí)內(nèi)讀完本文并開(kāi)始使用Python的程序員會(huì)喜歡這篇小文章,謝謝!
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注