Python面試題整理
參考資料
問題1
到底什麼是Python?你可以在回答中與其他技術進行對比(也鼓勵這樣做)。
答案
下面是一些關鍵點:
- Python是一種解釋型語言。這就是說,與C語言和C的衍生語言不同,Python程式碼在執行之前不需要編譯。其他解釋型語言還包括PHP和Ruby。
- Python是動態型別語言,指的是你在宣告變數時,不需要說明變數的型別。你可以直接編寫類似x=111和x=”I’m a string”這樣的程式碼,程式不會報錯。
- Python非常適合面向物件的程式設計(OOP),因為它支援通過組合(composition)與繼承(inheritance)的方式定義類(class)。* Python中沒有訪問說明符(access specifier,類似C++中的public和private),這麼設計的依據是“大家都是成年人了”。
- 在Python語言中,函式是第一類物件(first-class objects)。這指的是它們可以被指定給變數,函式既能返回函式型別,也可以接受函式作為輸入。類(class)也是第一類物件。
- Python程式碼編寫快,但是執行速度比編譯語言通常要慢。好在Python允許加入基於C語言編寫的擴充套件,因此我們能夠優化程式碼,消除瓶頸,這點通常是可以實現的。numpy就是一個很好地例子,它的執行速度真的非常快,因為很多算術運算其實並不是通過Python實現的。
- Python用途非常廣泛——網路應用,自動化,科學建模,大資料應用,等等。它也常被用作“膠水語言”,幫助其他語言和元件改善執行狀況。
- Python讓困難的事情變得容易,因此程式設計師可以專注於演算法和資料結構的設計,而不用處理底層的細節。
為什麼提這個問題:
如果你應聘的是一個Python開發崗位,你就應該知道這是門什麼樣的語言,以及它為什麼這麼酷。以及它哪裡不好。
問題2:
Python是如何進行記憶體管理的?
答案
從三個方面來說,一物件的引用計數機制,二垃圾回收機制,三記憶體池機制
一、物件的引用計數機制
Python內部使用引用計數,來保持追蹤記憶體中的物件,所有物件都有引用計數。
引用計數增加的情況:
- 一個物件分配一個新名稱
- 將其放入一個容器中(如列表、元組或字典)
引用計數減少的情況:
- 使用del語句對物件別名顯示的銷燬
- 引用超出作用域或被重新賦值
-
sys.getrefcount( )函式可以獲得物件的當前引用計數
多數情況下,引用計數比你猜測得要大得多。對於不可變資料(如數字和字串),直譯器會在程式的不同部分共享記憶體,以便節約記憶體。
二、垃圾回收
- 當一個物件的引用計數歸零時,它將被垃圾收集機制處理掉。
- 當兩個物件a和b相互引用時,del語句可以減少a和b的引用計數,並銷燬用於引用底層物件的名稱。然而由於每個物件都包含一個對其他物件的應用,因此引用計數不會歸零,物件也不會銷燬。(從而導致記憶體洩露)。為解決這一問題,直譯器會定期執行一個迴圈檢測器,搜尋不可訪問物件的迴圈並刪除它們。
三、記憶體池機制
Python提供了對記憶體的垃圾收集機制,但是它將不用的記憶體放到記憶體池而不是返回給作業系統。
- Pymalloc機制。為了加速Python的執行效率,Python引入了一個記憶體池機制,用於管理對小塊記憶體的申請和釋放。
- Python中所有小於256個位元組的物件都使用pymalloc實現的分配器,而大的物件則使用系統的malloc。
- 對於Python物件,如整數,浮點數和List,都有其獨立的私有記憶體池,物件間不共享他們的記憶體池。也就是說如果你分配又釋放了大量的整數,用於快取這些整數的記憶體就不能再分配給浮點數。
問題3:
什麼是lambda函式?它有什麼好處?
答案
lambda 表示式,通常是在需要一個函式,但是又不想費神去命名一個函式的場合下使用,也就是指匿名函式
lambda函式:首要用途是指點短小的回撥函式
lambda [arguments]:expression
>>> a=lambdax,y:x+y
>>> a(3,11)
問題4:
Python裡面如何實現tuple和list的轉換?
答案
答:直接使用tuple和list函式就行了,type()可以判斷物件的型別
問題5:
- 請寫出一段Python程式碼實現刪除一個list裡面的重複元素
答:
1,使用set函式,set(list)
2,使用字典函式,
a=[1,2,4,2,4,5,6,5,7,8,9,0]
b={}
b=b.fromkeys(a)
c=list(b.keys())
c
問題6:
Python裡面如何拷貝一個物件?(賦值,淺拷貝,深拷貝的區別)
答案
答:賦值(=),就是建立了物件的一個新的引用,修改其中任意一個變數都會影響到另一個。
淺拷貝:建立一個新的物件,但它包含的是對原始物件中包含項的引用(如果用引用的方式修改其中一個物件,另外一個也會修改改變){1,完全切片方法;2,工廠函式,如list();3,copy模組的copy()函式}
深拷貝:建立一個新的物件,並且遞迴的複製它所包含的物件(修改其中一個,另外一個不會改變){copy模組的deep.deepcopy()函式}
問題7:
介紹一下except的用法和作用?
答案
答:try…except…except…[else…][finally…]
執行try下的語句,如果引發異常,則執行過程會跳到except語句。
對每個except分支順序嘗試執行,如果引發的異常與except中的異常組匹配,執行相應的語句。如果所有的except都不匹配,則異常會傳遞到下一個呼叫本程式碼的最高層try程式碼中。
try下的語句正常執行,則執行else塊程式碼。如果發生異常,就不會執行
如果存在finally語句,最後總是會執行。
問題8:
Python中pass語句的作用是什麼?
答案
答:pass語句不會執行任何操作,一般作為佔位符或者建立佔位程式,whileFalse:pass
問題9:
介紹一下Python下range()函式的用法?
答案
答:列出一組資料,經常用在for in range()迴圈中
問題10:
如何用Python來進行查詢和替換一個文字字串?
答案
答:可以使用re模組中的sub()函式或者subn()函式來進行查詢和替換,
格式:sub(replacement, string[,count=0])(replacement是被替換成的文字,string是需要被替換的文字,count是一個可選引數,指最大被替換的數量)
>>> import re
>>>p=re.compile(‘blue|white|red’)
>>>print(p.sub(‘colour’,'blue socks and red shoes’))
colour socks and colourshoes
>>>print(p.sub(‘colour’,'blue socks and red shoes’,count=1))
colour socks and redshoes
subn()方法執行的效果跟sub()一樣,不過它會返回一個二維陣列,包括替換後的新的字串和總共替換的數量
問題11:
Python裡面match()和search()的區別?
答案
re模組中match(pattern,string[,flags]),檢查string的開頭是否與pattern匹配。
re模組中research(pattern,string[,flags]),在string搜尋pattern的第一個匹配值。
>>>print(re.match(‘super’, ‘superstition’).span())
(0, 5)
>>>print(re.match(‘super’, ‘insuperable’))
None
>>>print(re.search(‘super’, ‘superstition’).span())
(0, 5)
>>>print(re.search(‘super’, ‘insuperable’).span())
(2, 7)
問題12:
用Python匹配HTML tag的時候,<.>和<.?>有什麼區別?
答案
答:術語叫貪婪匹配( <.* >)和非貪婪匹配(<.*? > )
例如:
<div><span>test</span></div>
<.*> :<div><span>test</span></div>
<.*?> :<div>
問題13:
Python裡面如何生成隨機數?
答案
答:random模組
隨機整數:random.randint(a,b):返回隨機整數x,a<=x<=b
random.randrange(start,stop,[,step]):返回一個範圍在(start,stop,step)之間的隨機整數,不包括結束值。
隨機實數:random.random( ):返回0到1之間的浮點數
random.uniform(a,b):返回指定範圍內的浮點數。
問題14:
如何在一個function裡面設定一個全域性的變數?
答案
答:解決方法是在function的開始插入一個global宣告:
def f()
global x
問題15:
單引號,雙引號,三引號的區別
答案
答:單引號和雙引號是等效的,如果要換行,需要符號(),三引號則可以直接換行,並且可以包含註釋
如果要表示Let’s Go 這個字串
單引號:s4 = ‘Let\’s go’
雙引號:s5 = “Let’s go”
s6 = ‘I realy like“python”!’
這就是單引號和雙引號都可以表示字串的原因了
問題16:
python程式中文輸出問題怎麼解決?
答案
方法一:
用encode和decode
如:
import os.path
import xlrd,sys
Filename=’/home/tom/Desktop/1234.xls’
if not os.path.isfile(Filename):
raise NameError,”%s is not a valid filename”%Filename
bk=xlrd.open_workbook(Filename)
shxrange=range(bk.nsheets)
print shxrange
for x in shxrange:
p=bk.sheets()[x].name.encode(‘utf-8′)
print p.decode(‘utf-8′)
方法二:
在檔案開頭加上
reload(sys)
sys.setdefaultencoding(‘utf8′)
這2行,再試著執行一下
字串在Python內部的表示是unicode編碼,因此,在做編碼轉換時,通常需要以unicode作為中間編碼,即先將其他編碼的字串解碼(decode)成unicode,再從unicode編碼(encode)成另一種編碼。
decode的作用是將其他編碼的字串轉換成unicode編碼,如str1.decode(‘gb2312’),表示將gb2312編碼的字串str1轉換成unicode編碼。
encode的作用是將unicode編碼轉換成其他編碼的字串,如str2.encode(‘gb2312’),表示將unicode編碼的字串str2轉換成gb2312編碼。
因此,轉碼的時候一定要先搞明白,字串str是什麼編碼,然後decode成unicode,然後再encode成其他編碼
程式碼中字串的預設編碼與程式碼檔案本身的編碼一致。
如:s=’中文’
如果是在utf8的檔案中,該字串就是utf8編碼,如果是在gb2312的檔案中,則其編碼為gb2312。
這種情況下,要進行編碼轉換,都需要先用decode方法將其轉換成unicode編碼,再使用encode方法將其轉換成其他編碼。通常,在沒有指定特定的編碼方式時,都是使用的系統預設編碼建立的程式碼檔案。
如果字串是這樣定義:s=u’中文’
則該字串的編碼就被指定為unicode了,即python的內部編碼,而與程式碼檔案本身的編碼無關。因此,對於這種情況做編碼轉換,只需要直接使用encode方法將其轉換成指定編碼即可。
如果一個字串已經是unicode了,再進行解碼則將出錯,因此通常要對其編碼方式是否為unicode進行判斷:
isinstance(s,unicode) #用來判斷是否為unicode
用非unicode編碼形式的str來encode會報錯
如何獲得系統的預設編碼?
#!/usr/bin/env python
#coding=utf-8
import sys
print sys.getdefaultencoding()
該段程式在英文WindowsXP上輸出為:ascii
問題17:
Python檔案操作的面試題
答案
1.如何用Python刪除一個檔案?
使用os.remove(filename)或者os.unlink(filename);
2.Python如何copy一個檔案?
shutil模組有一個copyfile函式可以實現檔案拷貝