2018年最常見的36道Python面試題你能答對多少?附帶答案
Q 1:Python有哪些特點(diǎn)和優(yōu)點(diǎn)?
作為一門編程入門語言,Python主要有以下特點(diǎn)和優(yōu)點(diǎn):
- 可解釋
- 具有動(dòng)態(tài)特性
- 面向?qū)ο?/li>
- 簡明簡單
- 開源
- 具有強(qiáng)大的社區(qū)支持
當(dāng)然,實(shí)際上Python的優(yōu)點(diǎn)遠(yuǎn)不止如此。
Q 2:深拷貝和淺拷貝之間的區(qū)別是什么?
答:深拷貝就是將一個(gè)對象拷貝到另一個(gè)對象中,這意味著如果你對一個(gè)對象的拷貝做出改變時(shí),不會(huì)影響原對象。在Python中,我們使用函數(shù)deepcopy()執(zhí)行深拷貝,導(dǎo)入模塊copy,如下所示:
- >>> import copy
- >>> b=copy.deepcopy(a)

而淺拷貝則是將一個(gè)對象的引用拷貝到另一個(gè)對象上,所以如果我們在拷貝中改動(dòng),會(huì)影響到原對象。我們使用函數(shù)function()執(zhí)行淺拷貝,使用如下所示:>>> b=copy.copy(a)復(fù)制代碼
Q 3. 列表和元組之間的區(qū)別是?
答:二者的主要區(qū)別是列表是可變的,而元組是不可變的。舉個(gè)例子,如下所示:
- >>> mylist=[1,3,3]
- >>> mylist[1]=2
- >>> mytuple=(1,3,3)
- >>> mytuple[1]=2
- Traceback (most recent call last):
- File "<pyshell#97>", line 1, in <module>
- mytuple[1]=2
會(huì)出現(xiàn)以下報(bào)錯(cuò):
- TypeError: ‘tuple’ object does not support item assignment
從Q4到Q20都是針對新手的Python面試基礎(chǔ)試題,不過有經(jīng)驗(yàn)的人也可以看看這些問題,復(fù)習(xí)一下基礎(chǔ)概念。
Q 4. 解釋一下Python中的三元運(yùn)算子
不像C++,我們在Python中沒有?:,但我們有這個(gè):
- [on true] if [expression] else [on false]
如果表達(dá)式為True,就執(zhí)行[on true]中的語句。否則,就執(zhí)行[on false]中的語句。
下面是使用它的方法:
- >>> a,b=2,3
- >>> min=a if a<b else b
- >>> min
運(yùn)行結(jié)果:
- 2
- >>> print("Hi") if a<b else print("Bye")
運(yùn)行結(jié)果:
- Hi
Q 5. 在Python中如何實(shí)現(xiàn)多線程?
一個(gè)線程就是一個(gè)輕量級進(jìn)程,多線程能讓我們一次執(zhí)行多個(gè)線程。我們都知道,Python是多線程語言,其內(nèi)置有多線程工具包。
Python中的GIL(全局解釋器鎖)確保一次執(zhí)行單個(gè)線程。一個(gè)線程保存GIL并在將其傳遞給下個(gè)線程之前執(zhí)行一些操作,這會(huì)讓我們產(chǎn)生并行運(yùn)行的錯(cuò)覺。但實(shí)際上,只是線程在CPU上輪流運(yùn)行。當(dāng)然,所有的傳遞會(huì)增加程序執(zhí)行的內(nèi)存壓力。
Q 6. 解釋一下Python中的繼承
當(dāng)一個(gè)類繼承自另一個(gè)類,它就被稱為一個(gè)子類/派生類,繼承自父類/基類/超類。它會(huì)繼承/獲取所有類成員(屬性和方法)。
繼承能讓我們重新使用代碼,也能更容易的創(chuàng)建和維護(hù)應(yīng)用。Python支持如下種類的繼承:
- 單繼承:一個(gè)類繼承自單個(gè)基類
- 多繼承:一個(gè)類繼承自多個(gè)基類
- 多級繼承:一個(gè)類繼承自單個(gè)基類,后者則繼承自另一個(gè)基類
- 分層繼承:多個(gè)類繼承自單個(gè)基類
- 混合繼承:兩種或多種類型繼承的混合
Q 7. 什么是Flask?
Flask是Python編寫的一款輕量級Web應(yīng)用框架。其 WSGI 工具箱采用 Werkzeug ,模板引擎則使用 Jinja2。Flask使用 BSD 授權(quán)。其中兩個(gè)環(huán)境依賴是Werkzeug和jinja2,這意味著它不需要依賴外部庫。正因如此,我們將其稱為輕量級框架。
Flask會(huì)話使用簽名cookie讓用戶查看和修改會(huì)話內(nèi)容。它會(huì)記錄從一個(gè)請求到另一個(gè)請求的信息。不過,要想修改會(huì)話,用戶必須有密鑰Flask.secret_key。
Q 8. 在Python中是如何管理內(nèi)存的?
Python有一個(gè)私有堆空間來保存所有的對象和數(shù)據(jù)結(jié)構(gòu)。作為開發(fā)者,我們無法訪問它,是解釋器在管理它。但是有了核心API后,我們可以訪問一些工具。Python內(nèi)存管理器控制內(nèi)存分配。
另外,內(nèi)置垃圾回收器會(huì)回收使用所有的未使用內(nèi)存,所以使其適用于堆空間。
Q 9. 解釋Python中的help()和dir()函數(shù)
Help()函數(shù)是一個(gè)內(nèi)置函數(shù),用于查看函數(shù)或模塊用途的詳細(xì)說明:
- >>> import copy
- >>> help(copy.copy)
運(yùn)行結(jié)果為:
- Help on function copy in module copy:
- copy(x)
- Shallow copy operation on arbitrary Python objects.
- See the module’s __doc__ string for more info.
Dir()函數(shù)也是Python內(nèi)置函數(shù),dir() 函數(shù)不帶參數(shù)時(shí),返回當(dāng)前范圍內(nèi)的變量、方法和定義的類型列表;帶參數(shù)時(shí),返回參數(shù)的屬性、方法列表。
以下實(shí)例展示了 dir 的使用方法:
- >>> dir(copy.copy)
運(yùn)行結(jié)果為:
- [‘__annotations__’, ‘__call__’, ‘__class__’, ‘__closure__’, ‘__code__’, ‘__defaults__’, ‘__delattr__’, ‘__dict__’, ‘__dir__’, ‘__doc__’, ‘__eq__’, ‘__format__’, ‘__ge__’, ‘__get__’, ‘__getattribute__’, ‘__globals__’, ‘__gt__’, ‘__hash__’, ‘__init__’, ‘__init_subclass__’, ‘__kwdefaults__’, ‘__le__’, ‘__lt__’, ‘__module__’, ‘__name__’, ‘__ne__’, ‘__new__’, ‘__qualname__’, ‘__reduce__’, ‘__reduce_ex__’, ‘__repr__’, ‘__setattr__’, ‘__sizeof__’, ‘__str__’, ‘__subclasshook__’]
Q 10. 當(dāng)退出Python時(shí),是否釋放全部內(nèi)存?
答案是No。循環(huán)引用其它對象或引用自全局命名空間的對象的模塊,在Python退出時(shí)并非完全釋放。
另外,也不會(huì)釋放C庫保留的內(nèi)存部分。
Q 11. 什么是猴子補(bǔ)丁?
在運(yùn)行期間動(dòng)態(tài)修改一個(gè)類或模塊。
- >>> class A:
- def func(self):
- print("Hi")
- >>> def monkey(self):
- print "Hi, monkey"
- >>> m.A.func = monkey
- >>> a = m.A()
- >>> a.func()
運(yùn)行結(jié)果為:
- Hi, Monkey
Q 12. Python中的字典是什么?
字典是C++和Java等編程語言中所沒有的東西,它具有鍵值對。
- >>> roots={25:5,16:4,9:3,4:2,1:1}
- >>> type(roots)
- <class 'dict'>
- >>> roots[9]
運(yùn)行結(jié)果為:
- 3
字典是不可變的,我們也能用一個(gè)推導(dǎo)式來創(chuàng)建它。
- >>> roots={x**2:x for x in range(5,0,-1)}
- >>> roots
運(yùn)行結(jié)果:
- {25: 5, 16: 4, 9: 3, 4: 2, 1: 1}
Q 13. 請解釋使用*args和**kwargs的含義
當(dāng)我們不知道向函數(shù)傳遞多少參數(shù)時(shí),比如我們向傳遞一個(gè)列表或元組,我們就使用*args。
- >>> def func(*args):
- for i in args:
- print(i)
- >>> func(3,2,1,4,7)
運(yùn)行結(jié)果為:
- 3
- 2
- 1
- 4
- 7
在我們不知道該傳遞多少關(guān)鍵字參數(shù)時(shí),使用**kwargs來收集關(guān)鍵字參數(shù)。
- >>> def func(**kwargs):
- for i in kwargs:
- print(i,kwargs[i])
- >>> func(a=1,b=2,c=7)
運(yùn)行結(jié)果為:
- a.1
- b.2
- c.7
Q 14. 請寫一個(gè)Python邏輯,計(jì)算一個(gè)文件中的大寫字母數(shù)量
- >>> import os
- >>> os.chdir('C:\Users\lifei\Desktop')
- >>> with open('Today.txt') as today:
- count=0
- for i in today.read():
- if i.isupper():
- count+=1
- print(count)
運(yùn)行結(jié)果:
- 26
Q 15. 什么是負(fù)索引?
我們先創(chuàng)建這樣一個(gè)列表:
- >>> mylist=[0,1,2,3,4,5,6,7,8]
負(fù)索引和正索引不同,它是從右邊開始檢索。
- >>> mylist[-3]
運(yùn)行結(jié)果:
- 6
它也能用于列表中的切片:
- >>> mylist[-6:-1]
結(jié)果:
- [3, 4, 5, 6, 7]
Q 16. 如何以就地操作方式打亂一個(gè)列表的元素?
為了達(dá)到這個(gè)目的,我們從random模塊中導(dǎo)入shuffle()函數(shù)。
- >>> from random import shuffle
- >>> shuffle(mylist)
- >>> mylist
運(yùn)行結(jié)果:
- [3, 4, 8, 0, 5, 7, 6, 2, 1]
Q 17. 解釋Python中的join()和split()函數(shù)
Join()能讓我們將指定字符添加至字符串中。
- >>> ','.join('12345')
運(yùn)行結(jié)果:
- ‘1,2,3,4,5’
Split()能讓我們用指定字符分割字符串。
- >>> '1,2,3,4,5'.split(',')
運(yùn)行結(jié)果:
- [‘1’, ‘2’, ‘3’, ‘4’, ‘5’]
Q 18. Python區(qū)分大小寫嗎?
如果能區(qū)分像myname和Myname這樣的標(biāo)識符,那么它就是區(qū)分大小寫的。也就是說它很在乎大寫和小寫。我們可以用Python試一試:
- >>> myname='Ayushi'
- >>> Myname
- Traceback (most recent call last):
- File "<pyshell#3>", line 1, in <module>
運(yùn)行結(jié)果:
- MynameNameError: name ‘Myname’ is not defined
可以看到,這里出現(xiàn)了NameError,所以Python是區(qū)分大小寫的。
Q 19. Python中的標(biāo)識符長度能有多長?
在Python中,標(biāo)識符可以是任意長度。此外,我們在命名標(biāo)識符時(shí)還必須遵守以下規(guī)則:
- 只能以下劃線或者 A-Z/a-z 中的字母開頭
- 其余部分可以使用 A-Z/a-z/0-9
- 區(qū)分大小寫
- 關(guān)鍵字不能作為標(biāo)識符,Python中共有如下關(guān)鍵字:

Q 20. 怎么移除一個(gè)字符串中的前導(dǎo)空格?
字符串中的前導(dǎo)空格就是出現(xiàn)在字符串中***個(gè)非空格字符前的空格。我們使用方法Istrip()可以將它從字符串中移除。
- >>> ' Ayushi '.lstrip()
結(jié)果:
- ‘Ayushi ’
可以看到,該字符串既有前導(dǎo)字符,也有后綴字符,調(diào)用Istrip()去除了前導(dǎo)空格。如果我們想去除后綴空格,就用rstrip()方法。
- >>> ' Ayushi '.rstrip()
結(jié)果:
- ‘ Ayushi’
從Q 21到Q 35是為有Python經(jīng)驗(yàn)者準(zhǔn)備的進(jìn)階版Python面試題。
Q 21. 怎樣將字符串轉(zhuǎn)換為小寫?
我們使用lower()方法。
- >>> 'AyuShi'.lower()
結(jié)果:
- ‘ayushi’
使用upper()方法可以將其轉(zhuǎn)換為大寫。
- >>> 'AyuShi'.upper()
結(jié)果:
- ‘AYUSHI’
另外,使用isupper()和islower()方法檢查字符春是否全為大寫或小寫。
- >>> 'AyuShi'.isupper()
- False
- >>> 'AYUSHI'.isupper()
- True
- >>> 'ayushi'.islower()
- True
- >>> '@yu$hi'.islower()
- True
- >>> '@YU$HI'.isupper()
- True
那么,像@和$這樣的字符既滿足大寫也滿足小寫。
Istitle()能告訴我們一個(gè)字符串是否為標(biāo)題格式。
- >>> 'The Corpse Bride'.istitle()
- True
Q 22. Python中的pass語句是什么?
在用Python寫代碼時(shí),有時(shí)可能還沒想好函數(shù)怎么寫,只寫了函數(shù)聲明,但為了保證語法正確,必須輸入一些東西,在這種情況下,我們會(huì)使用pass語句。
- >>> def func(*args):
- pass
- >>>
同樣,break語句能讓我們跳出循環(huán)。
- >>> for i in range(7):
- if i==3: break
- print(i)
結(jié)果:
- 0
- 1
- 2
***,continue語句能讓我們跳到下個(gè)循環(huán)。
- >>> for i in range(7):
- if i==3: continue
- print(i)
結(jié)果:
- 0
- 1
- 2
- 4
- 5
- 6
Q 23. Python中的閉包是什么?
當(dāng)一個(gè)嵌套函數(shù)在其外部區(qū)域引用了一個(gè)值時(shí),該嵌套函數(shù)就是一個(gè)閉包。其意義就是會(huì)記錄這個(gè)值。
- >>> def A(x):
- def B():
- print(x)
- return B
- >>> A(7)()
結(jié)果:
- 7
Q 24. 解釋一下Python中的//,%和 ** 運(yùn)算符
- //運(yùn)算符執(zhí)行地板除法(向下取整除),它會(huì)返回整除結(jié)果的整數(shù)部分。
- >>> 7//2
- 3
這里整除后會(huì)返回3.5。
同樣地,執(zhí)行取冪運(yùn)算。ab會(huì)返回a的b次方。
- >>> 2**10
- 1024
***,%執(zhí)行取模運(yùn)算,返回除法的余數(shù)。
- >>> 13%7
- 6
- >>> 3.5%1.5
- 0.5
Q 25. 在Python中有多少種運(yùn)算符?解釋一下算數(shù)運(yùn)算符。
在Python中,我們有7種運(yùn)算符:算術(shù)運(yùn)算符、關(guān)系運(yùn)算符、賦值運(yùn)算符、邏輯運(yùn)算符、位運(yùn)算符、成員運(yùn)算符、身份運(yùn)算符。
我們有7個(gè)算術(shù)運(yùn)算符,能讓我們對數(shù)值進(jìn)行算術(shù)運(yùn)算:
1.加號(+),將兩個(gè)值相加
- >>> 7+8
- 15
2.減號(-),將***個(gè)值減去第二個(gè)值
- >>> 7-8
- -1
3.乘號(*),將兩個(gè)值相乘
- >>> 7*8
- 56
4.除號(/),用第二個(gè)值除以***個(gè)值
- >>> 7/8
- 0.875
- >>> 1/1
- 1.0
5.向下取整除、取模和取冪運(yùn)算,參見上個(gè)問題。
Q 26. 解釋一下Python中的關(guān)系運(yùn)算符
關(guān)系運(yùn)算符用于比較兩個(gè)值。
1.小于號(<),如果左邊的值較小,則返回True。
- >>> 'hi'<'Hi'
- False
2.大于號(>),如果左邊的值較大,則返回True。
- >>> 1.1+2.2>3.3
- True
3.小于等于號(<=),如果左邊的值小于或等于右邊的值,則返回Ture。
- >>> 3.0<=3
- True
4.大于等于號(>=),如果左邊的值大于或等于右邊的值,則返回True。
- >>> True>=False
- True
等于號(==),如果符號兩邊的值相等,則返回True。
- >>> {1,3,2,2}=={1,2,3}
- True
不等于號(!=),如果符號兩邊的值不相等,則返回True。
- >>> True!=0.1
- True
- >>> False!=0.1
- True
Q 27. 解釋一下Python中的賦值運(yùn)算符
這在Python面試中是個(gè)重要的面試問題。
我們將所有的算術(shù)運(yùn)算符和賦值符號放在一起展示:
- >>> a=7
- >>> a+=1
- >>> a
- 8
- >>> a-=1
- >>> a
- 7
- >>> a*=2
- >>> a
- 14
- >>> a/=2
- >>> a
- 7.0
- >>> a**=2
- >>> a
- 49
- >>> a//=3
- >>> a
- 16.0
- >>> a%=4
- >>> a
- 0.0
Q 28. 解釋一下Python中的邏輯運(yùn)算符
Python中有3個(gè)邏輯運(yùn)算符:and,or,not。
- >>> False and True
- False
- >>> 7<7 or True
- True
- >>> not 2==2
- False
Q 29. 解釋一下Python中的成員運(yùn)算符
通過成員運(yùn)算符‘in’和‘not in’,我們可以確認(rèn)一個(gè)值是否是另一個(gè)值的成員。
- >>> 'me' in 'disappointment'
- True
- >>> 'us' not in 'disappointment'
- True
Q 30. 解釋一下Python中的身份運(yùn)算符
這也是一個(gè)在Python面試中常問的問題。
通過身份運(yùn)算符‘is’和‘is not’,我們可以確認(rèn)兩個(gè)值是否相同。
- >>> 10 is '10'
- False
- >>> True is not False
- True
Q 31. 講講Python中的位運(yùn)算符
該運(yùn)算符按二進(jìn)制位對值進(jìn)行操作。
與(&),按位與運(yùn)算符:參與運(yùn)算的兩個(gè)值,如果兩個(gè)相應(yīng)位都為1,則該位的結(jié)果為1,否則為0
- >>> 0b110 & 0b010
- 2
2.或(|),按位或運(yùn)算符:只要對應(yīng)的二個(gè)二進(jìn)位有一個(gè)為1時(shí),結(jié)果位就為1。
- >>> 3|2
- 3
3.異或(^),按位異或運(yùn)算符:當(dāng)兩對應(yīng)的二進(jìn)位相異時(shí),結(jié)果為1
- >>> 3^2
- 1
4.取反(~),按位取反運(yùn)算符:對數(shù)據(jù)的每個(gè)二進(jìn)制位取反,即把1變?yōu)?,把0變?yōu)?
- >>> ~2
- -3
5.左位移(<<),運(yùn)算數(shù)的各二進(jìn)位全部左移若干位,由 << 右邊的數(shù)字指定了移動(dòng)的位數(shù),高位丟棄,低位補(bǔ)0
- >>> 1<<2
- 4
6.右位移(>>),把">>"左邊的運(yùn)算數(shù)的各二進(jìn)位全部右移若干位,>> 右邊的數(shù)字指定了移動(dòng)的位數(shù)
- >>> 4>>2
- 1
Q 32. 在Python中如何使用多進(jìn)制數(shù)字?
我們在Python中,除十進(jìn)制外還可以使用二進(jìn)制、八進(jìn)制和十六進(jìn)制。
二進(jìn)制數(shù)字由0和1組成,我們使用 0b 或 0B 前綴表示二進(jìn)制數(shù)。
- >>> int(0b1010)
- 10
2.使用bin()函數(shù)將一個(gè)數(shù)字轉(zhuǎn)換為它的二進(jìn)制形式。
- >>> bin(0xf)
- ‘0b1111’
3.八進(jìn)制數(shù)由數(shù)字 0-7 組成,用前綴 0o 或 0O 表示 8 進(jìn)制數(shù)。
- >>> oct(8)
- ‘0o10’
4.十六進(jìn)數(shù)由數(shù)字 0-15 組成,用前綴 0x 或者 0X 表示 16 進(jìn)制數(shù)。
- >>> hex(16)
- ‘0x10’
- >>> hex(15)
- ‘0xf’
Q 33. 怎樣獲取字典中所有鍵的列表?
使用 keys() 獲取字典中的所有鍵
- >>> mydict={'a':1,'b':2,'c':3,'e':5}
- >>> mydict.keys()
- dict_keys(['a', 'b', 'c', 'e'])
Q 34. 為何不建議以下劃線作為標(biāo)識符的開頭
因?yàn)镻ython并沒有私有變量的概念,所以約定速成以下劃線為開頭來聲明一個(gè)變量為私有。所以如果你不想讓變量私有,就不要使用下劃線開頭。
Q 35. 怎樣聲明多個(gè)變量并賦值?
一共有兩種方式:
- >>> a,b,c=3,4,5 #This assigns 3, 4, and 5 to a, b, and c respectively
- >>> a=b=c=3 #This assigns 3 to a, b, and c
Q 36. 元組的解封裝是什么?
首先我們來看解封裝:
- >>> mytuple=3,4,5
- >>> mytuple
- (3, 4, 5)
這將 3,4,5 封裝到元組 mytuple 中。
現(xiàn)在我們將這些值解封裝到變量 x,y,z 中:
- >>> x,y,z=mytuple
- >>> x+y+z
得到結(jié)果12。