之前發表過一篇文章《如何學習一門新的語言》。之後就開始學習python了,具體原因也不是很清楚了。我先是從python簡明教程開始,寫了兩篇筆記《python初體驗1》《python初體驗2》,但是沒有太多的感受。後面因為一個小的項目,我想嘗試一下python,在這個過程中,體驗了python的簡潔和高效,有一種飛起來的感覺——也許我是一名c語言程式員。
事實也再次證明,學習一門新的語言最好的方法就是在實際開發中去使用它。我旁邊沒有人有python經驗,所以學習過程有些曲折,還好很多問題都可以通過網路解決。另外,學習的過程中發現一本書《python參考手冊》非常不錯,有時間一定要通讀一下。
學習一門語言最大的障礙不是文法,也不是庫的使用,而是之前語言的思維方式——思維定勢會讓人傾向於用新的語言來類比自己熟悉的語言,而這正是最大的障礙所在,你將無法體會到這門語言的優勢。所以要學會用新的語言的思維方式來解決問題。還是讓我們在來溫習一下這個非常有寓意的比喻:
這就像你本來只有一個榔頭,有人給了你一個螺絲刀,你說“哎,這不是一把好榔頭,但是我應該可以倒著拿螺絲刀,用螺絲刀來砸東西。”你得到了一個很爛的榔頭,但事實上它確實一把很不錯的螺絲刀。所以你應該對所有的事物保持開放和積極的心態。”
這方面賴勇浩翻譯過一篇文章《pythonic到底是什麼玩意》,應該是同一個意思。python方面賴勇浩寫的文章非常不錯。
好了,進入正題了:下面是我的總結,可能不夠pythonic,如果你有更好的方案,就指出了。
通訊端編程:
1、 函數的功能基本和c類似,唯一不同的地方在於當發生錯誤時,它不是通過傳回值來告知的,而是通過觸發異常,所以udp中的bind, recvfrom, sendto必須要進行捕捉異常。
2、 通訊端在垃圾收集的時候也會關閉。
3、 擷取網卡的IP:
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
return socket.inet_ntoa(fcntl.ioctl(s.fileno(), 0X8915, struct.pack('256s', ethname[:15]))[20:24])
字串的使用:
1、 字串中刪除一個字串。沒有直接提供這個方法,但是replace可以實現:
"abcdef".replace(" ", "")
同樣的功能還有一個方法:translate。它的原有作用是將字串中的某個字元替換為另外一個字元,注意,不是字串。它的第一個參數是一個轉換表。第二個參數是要刪除的字串。我們可以利用第二個參數del,實現這個功能。同時,第一個參數設定為None。
translate可能更高效一點。另外,它的第二個參數可以使一個字串,含有多個字元,這樣就會刪除多個。
2、 strip方法:去除字串兩側的空格,返回新的字串。這個功能非常有用。
3、 endswitch:檢查字串是否已某字串結尾。
4、 partition:它將字串按指定的字串分為三個部分,返回一個元組。第一個是指定字串前面內容,第二個是指定字串,第三個是指定字串後面的內容。用於字串解析非常好用。
5、 split:將字串按照某指定字串分割成多個子字串,返回一個分割後的列表。
6、 join:將一個字串列表中的各個字串串連起來,中間插入指定的字串。
函數的使用:
1、函數的範圍:函數中定義一個變數,如果和全域變數重名,則全域變數名稱就會被覆蓋,也就是,這裡對這個變數的更改,不會更改全域變數。但是,如果直接使用的話,是會使用全域變數的。同時,如果想要修改全域變數,需要制定是全域變數:global a
字典的使用:
1、 字典的刪除:直接使用del dict[k]可能會引發異常;首先判斷k是否存在則效率有些低;使用異常使程式結構看起來不好。一個好的方法是pop(k, default v)。這個刪除一個k項,並且返回。如果不存在返回預設的v。如果不加預設值,則會引發異常。
2、 直接使用字典下標擷取字典的值可能會引發一場。使用get方法則不會,如果不存在會返回none。另外,還可以設定不存在的預設值。
3、 通過字典格式化字串:print “value is %(key)s” % kvdict
4、 item方法返回一個列表,列表中的元素是一個元組,第一個是key,第二個是value。比較好用的方法。
列表的使用:
1、 列表的刪除:不可以在遍曆的過程中刪除鏈表,這樣會得到不可預知的後果。可以使用列表的過濾,來獲得新的列表。
2、 列表的過濾:
def filterFun(node):#這個函數做了兩個事情哎。
node.cycleCount = node.cycleCount -1
return node.cycleCount < 0
timeoutList = filter(filterFun, timerList)
對timerList中的每個節點執行函數filterFun,根據filterFun返回的結果,為真的項組成一個新的列表。
3、 map: kvlist = map(lambda x:x.strip(), kvlist)
4、
XML的使用:
1、 處理xml訊息包比較好用的模組是xml.etree.ElementTree。
2、 Element執行xml的根節點。
3、 elem.find(path):尋找根節點下面路徑為path的子節點。
4、 elem.findall(path):同樣的子節點可能有多個,這裡會返回一個列表。
5、 elem.findtext(path):擷取指定路徑子節點的內容,這個我們會經常使用。
6、 elem.get(key);擷取屬性的值。
7、 上面如果沒用,則返回none
8、 elem.append:添加自節點。
9、 elem.tag:返回tag值,也就是name。
10、 elem.text:返回內容。
11、 elem.attrib:返回屬性的字典。
12、 SubElement:產生一個節點,自動添加為父節點的子節點。
13、 tostring:轉化為xml文本字串。但是不包括xml頭。如果編碼方式為UTF-8或者GB2312,gb2312都會產生xml頭;如果是utf-8,則不會產生xml頭
14、 fromstring:從字串轉化為ElementTree對象。和XML同樣的功能。
15、 elem.set();設定屬性值
time的使用:
1、 time.sleep()函數函數具有c下sleep函數功能,單位為秒,但是可以接受浮點數。這樣可以表示毫秒。
2、 ti = datetime.datetime.now()可以顯示當前的時間,包括當前的微秒也可以顯示出來。兩個的差值可以表示時間的間隔:microsecondLong = timeLong.seconds *1000000 + timeLong.microseconds。差值的成員是seconds和microseconds
3、
OO的使用:
1、 如果不想讓成員變數或者方法被外部使用(也就是private特性),可以以__雙底線開通。
2、 Python也可以實現抽象基類,也就是介面:
3、 __str__屬性可以將對象轉換為字串,也就是調用print(object)是會列印的字串。
4、 __call__可以見對象作為函數來調用。給它一個入參就可以。
5、 對象執行個體是否可以刪除?
6、 Python參考手冊要好好看一下。
模組的使用:
1、 如果不想將模組的某些函數和變數被別的模組使用,可以以單底線開頭。這樣import *是沒有的,但是使用importmode,然後mode._fun仍然可以調用。
2、
日誌的使用:
1、 日誌的標準模組logging基本可以滿足我的工作。
2、 設定log的初始化工作:
logging.basicConfig(
filename = "test.log",
format = "[%(asctime)s-%(levelname)s] %(message)s[%(filename)s,%(lineno)d]",
level = logging.INFO,
datefmt = "%F %T")
3、 除此之外,一個比較強大的功能就是過濾功能:可以針對層級,檔案,行號等等很多的東西進行過濾。
4、
自省的使用:
1、 type()可以查看對象的類型。這就是自省。也就是可以看看自己是什麼類型。這個功能在動態語言中非常有用。
2、
設定檔讀取的使用:
1、 使用模組ConfigParser。執行個體如下:
conf = ConfigParser()
conf.read("snmp_agent.ini")
print(conf.get("main", "log_level"))
print(conf.getint("main", "ne_agent_port"))
print(conf.get("main", "ne_agent_qip"))
異常的使用:
1、 盡量少用。它會使程式難以理解,而且還會發生不可預知的情況,比如異常的發生使程式的狀態變為一個未知狀態。
2、 可以尋找替代方案。
3、 程式非常重要,不可以停止,可以在主迴圈封裝在異常處理中運行。
4、 列印出異常的資訊,供後面的定位:log.error(traceback.format_exc())