20 个 Python 面试题来挑战你的知识
moboyou 2025-09-06 07:24 11 浏览
在本文中,我将我的一些笔记变成了20 个面试问题,涵盖了数据结构、核心编程概念和 Python 最佳实践。
希望你能完成其中的一些并重温你的 Python 技能。
事不宜迟,让我们直接进入。
1. 列表和元组有什么区别?你应该什么时候使用哪一个?
列表是可变数据结构,而元组是不可变数据结构。
Python 中的可变对象具有更改其值的能力。
列表是动态的:你可以向其中添加项目或覆盖和删除现有项目。
元组是固定大小的:它们没有方法append或extend方法。你也不能从中删除项目。
元组和列表都支持索引并允许使用in运算符检查其中的现有元素。
→ 在某些情况下,我认为元组可能有用。
- 如果你声明一个你知道永远不会更改的项目集合,或者你将只循环而不更改其值,请使用元组。
- 如果你寻找性能,元组比列表更快,因为它们是只读结构。如果你不需要写操作,请考虑使用元组。
- 如果你想防止意外写入不需要更改的数据,元组可以使你的代码更安全。
这是一个代码示例,显示了元组与列表的不同之处。
>>> numbers = [1, 2, 3, 4, 5]
>>> numbers[1] = 100
>>> print(numbers)
[1, 100, 3, 4, 5]
>>> names = ("john", "joe", "alice")
>>> names[0] = "bob")
---------------------------------------------------------------------------
TypeError Traceback (most recent call last)
<ipython-input-2-25012ce34a87> in <module>
----> 1 names[0] = "bob"
TypeError: 'tuple' object does not support item assignment2 — 多处理和多线程有什么区别?你应该什么时候使用哪个?
多处理和多线程是旨在加快代码速度的编程范例。
当你使用多处理时,你可以在进程上并行计算。进程是独立的,不相互通信:它们不共享相同的内存区域,并且相互之间有严格的隔离。在应用方面,多处理适用于 CPU 密集型工作负载。但是,它确实具有与进程数量成正比的大量内存占用。
另一方面,在多线程应用程序中,线程存在于单个进程中。因此,它们共享相同的内存区域:它们可以修改相同的变量并且可以相互干扰。虽然进程是严格并行执行的,但在 Python 中的给定时间点只执行一个线程,这是由于全局解释器锁 ( GIL )。多线程适用于受 IO 限制的应用程序,例如网页抓取或从数据库中获取数据。
→ 如果你想了解有关多线程和多处理的更多信息,我建议你阅读我之前关于多进程跟线程的文章关于多线程你知道多少呢?,该文章全面介绍了这两个概念。
3 — 模块、包和库之间有什么区别?
模块只是一个 Python 文件,旨在导入脚本或其他模块。它包含函数、类和全局变量。
包是模块的集合,它们在文件夹中组合在一起以提供一致的功能。包可以像模块一样被导入。它们通常有一个__init__.py文件告诉 Python 解释器按原样处理它们。
库是包的集合。
4 — python 中的多线程有什么问题?
全局解释器锁(或 GIL)可防止 Python 解释器同时执行多个线程。简而言之,GIL 强制在 Python 中的任何时间点只执行一个线程。
这代表了依赖多线程代码的 CPU 密集型应用程序的一个很大的性能瓶颈。
5 — 什么是装饰器?你能描述一下装饰器值得使用的情况吗?
装饰器是一个接收函数作为输入并返回函数作为输出的函数。装饰器的目标是在不改变其核心机制的情况下扩展输入函数的行为。
使用装饰器还可以防止你重复自己。它迫使你编写一次通用代码,然后将其用于需要它的每个功能。
装饰器大放异彩的典型用例是日志记录。
例如,想象一下,你希望将传递给程序中调用的每个函数的所有参数值记录到终端。你可以遍历每个函数定义并将其写下来,或者你可以只编写一个装饰器来执行此日志记录任务并将其应用于所有需要它的函数。
将装饰器应用于函数只需在该函数的定义上方添加一行即可。
#没有装饰器
def my_awesome_function():
# 做一些很棒的事情
# 带有装饰器
@my_awesome_decorator
def my_awesome_function():
# 做更棒的事情下面是一个代码示例,它创建了一个名为的装饰器,该装饰器log记录了传递给函数的参数的值。
import logging
logging.basicConfig(
format="%(asctime)s [%(levelname)s] %(name)s - %(message)s",
level=logging.INFO,
datefmt="%Y-%m-%d %H:%M:%S",
stream=sys.stdout,
)
logger = logging.getLogger("notebook")
def log(func):
def wrapper(*args, **kwargs):
output = func(*args, **kwargs)
msg = f"{func.__name__} was run with the following args: {args} and the following kwargs {kwargs}"
logger.info(msg)
return output
return wrapper
@log
def print_args(*args, **kwargs):
print(args)
print(kwargs)
>>> print_args(10, a=2, b="test")
(10,)
{'a': 2, 'b': 'test'}
2022-03-06 18:07:05,248 - notebook - INFO - print_args was run with the following args: (10,) and the following kwargs {'a': 2, 'b': 'test'}
>>> print_args(10, 100, a=2, b="test")
(10, 100)
{'a': 2, 'b': 'test'}
2022-03-06 18:07:05,562 - notebook - INFO - print_args was run with the following args: (10, 100) and the following kwargs {'a': 2, 'b': 'test'}
装饰器还可以用于其他目的,例如计时功能、验证输入数据、执行访问控制和身份验证、缓存等。
6 — 如何正确地将数据写入文件?否则会出什么问题?
使用上下文管理器是关键。
当你使用open没有上下文管理器的语句并且在关闭文件之前发生一些异常时(关闭文件是你在以这种方式打开文件时必须记住的事情)可能会发生内存问题并且文件可能会在此过程中损坏。
当你with用来打开一个文件并且发生异常时,Python 保证该文件是关闭的。
d = {"foo": 1}
# bad practice
f = open("./data.csv", "wb")
f.write("some data")
v = d["bar"] # KeyError
# f.close() never executes which leads to memory issues
f.close()
# good practice
with open("./data.csv", "wb") as f:
f.write("some data")
v = d["bar"]
# python still executes f.close() even if the KeyError exception occurs7 — 函数参数是按引用传递还是按值传递?
在 Python 中,所有函数参数都是通过引用传递的:这意味着如果将参数传递给函数,则函数将获得对同一对象的引用。
如果对象是可变的并且函数改变了它,则参数将在函数的外部范围内发生变异。让我们看一个例子:
>>> def append_number(numbers):
numbers.append(5)
>>> numbers = [1, 2, 3, 4]
>>> print(f"before: {numbers}"
[1, 2, 3, 4]
>>> append_number(numbers)
>>> numbers
[1, 2, 3, 4, 5]8 — 如何覆盖对象的打印方式?
使用 the__str__和__repr__dunder 方法。
这是一个示例,它演示了 Person 类中的实例在打印到控制台时如何被很好地格式化。
class Person:
def __init__(self, first_name, last_name, age):
self.first_name = first_name
self.last_name = last_name
self.age = age
def __str__(self):
return f"{self.first_name} {self.last_name} ({self.age})"
def __repr__(self):
return f"{self.first_name} {self.last_name} ({self.age})"
>>> person = Person("John", "Doe", 30) # thanks to __str__
John Doe (30)
>>> person # thanks to __repr__
John Doe (30)9 — 编写一个计算整数 n 阶乘的函数
递归是关键
def factorial(n):
if n == 0:
return 1
else:
return n * factorial(n-1)10 — is 和 == 运算符有什么区别?
==是一个测试相等性的运算符,而is是一个测试身份的运算符。
两个对象可以具有相同的值,但不一定相同(即具有相同的内存地址)。
请记住,这a is b是id(a) == id(b).
11 — 什么时候不应该使用 assert 语句?
assert语句对于内部测试和完整性检查很有用。
但是,它不应该用于执行数据验证或错误处理,因为出于性能原因,它通常在生产代码中被禁用。
想象一下,如果你使用断言检查管理员权限:这可能会在生产中引入很大的安全漏洞。
assert你可以抛出自定义错误,而不是使用该语句。
# Dangerous code!
def delete_product(user, product_id):
assert user.is_admin()
user.delete_product(product_id)
# Handle this properly by raising an error
def delete_product(user, product_id):
if not user.is_admin():
raise AuthError("User must have admin privileges")
else:
user.delete_product(product_id)12 — 什么是 Python 生成器?
Python 生成器是一个生成一系列项目的函数。
生成器看起来像典型的函数,但它们的行为是不同的。对于初学者,不使用return语句,而是使用yield语句。
然后,调用生成器函数不会运行该函数:它只会创建一个生成器对象。生成器的代码仅在next函数应用于生成器对象或生成器被迭代时执行(在这种情况下,next函数被隐式调用)
在生成器对象上调用函数的次数next等于yield在生成器函数中调用语句的次数。
你可以使用 for 循环或生成器表达式定义生成器。
>>> def repeat(n, message):
for _ in range(n):
yield message
repeat_hello_five_times = repeat(5, hello)
>>> for message in repeat_hello_five_times:
print(message)
"hello"
"hello"
"hello"
"hello"
"hello"
>>> repeat_hello_five_time = ("hello" for _ in range(5))
>>> repeat_hello_five_times
<generator object <genexpr> at 0x7fb64f2362d0>
>>> for message in repeat_hello_five_times:
print(message)
"hello"
"hello"
"hello"
"hello"
"hello" 13 — 类方法和静态方法有什么区别?什么时候应该使用哪个?
静态方法是一种对调用它的类或实例有任何了解的方法。这是一种逻辑上属于该类但没有隐式参数的方法。
可以在类或其任何实例上调用静态方法。
类方法是传递给调用它的类的方法,就像self传递给类中的其他实例方法一样。类方法的强制参数不是类实例:它实际上是类本身。
类方法的一个典型用例是提供另一种构造实例的方法:执行此操作的类方法称为类的工厂。
这是一个使用类方法的 Employee 类,该类方法创建实例的方式与类的主构造函数略有不同。
class Employee(object):
def __init__(self, first_name, last_name):
self.first_name = first_name
self.last_name = last_nale
@classmethod
def from_string(cls, name_str):
first_name, last_name = map(str, name_str.split(' '))
employee = cls(first_name, last_name)
return employee
ahmed = Employee.from_string('Ahmed Besbes')14 — 举一个例子说明你如何使用 zip 和枚举
该zip函数将多个迭代作为输入并将它们聚合到一个元组中。例如,如果你想同时遍历两个列表,这可能很有用。
>>> names = ["john", "bob", "alice"]
>>> ages = [10, 16, 20]
>>> for name, age in zip(names, ages):
print(name, age)
john 10
bob 16
alice 20该enumerate函数允许循环遍历一个可迭代对象并同时访问正在运行的索引和项目。
>>> names = ["john", "bob", "alice"]
>>> for index, name in enumerate(names):
print(index, name)
0 john
1 bob
2 alice15 — 你会如何在给定的函数中使用 *args 和 **kwargs?
*args 和 **kwargs 通过接受可变数量的参数使 Python 函数更加灵活。
- *args 在列表中传递可变数量的非关键字参数
- **kwargs 在字典中传递可变数量的关键字参数
这是一个函数示例,该函数采用可变数量的关键字参数,这些参数收集在名为的字典中data(请注意,它不需要命名kwargs)
16 — 给出一个使用 map 的函数式编程示例
>>> numbers = [1, 2, 3, 4, 5]
>>> numbers_times_2 = list(map(lambda n: n * 2, numbers))
>>> numbers_times_2
[2, 4, 6, 8, 10]17 — continue 和 break 语句有什么区别
该break语句终止包含它的循环。程序立即移动到循环外部范围内的代码段。
另一方面,该continue语句跳过当前迭代的其余代码并移至下一个迭代。
18 - 如何防止函数被调用不必要的时间?
使用缓存。
如果与给定输入关联的输出在一段时间内没有变化,则使用缓存对函数有意义。
一个典型的场景是查询一个 web 服务器:如果你第一次查询一个 URL,并且你知道响应不会改变,你可以缓存结果。
from cachetools import cached, TTLCache
cache = TTLCache(maxsize=100, ttl=86400)
@cached(cache)
def extract_article_content(url):
response = requests.get(url)
content = response.content
return content19 — 给出一些 PEP8 指南
- 每个缩进级别使用 4 个空格。
- 进口应按以下顺序分组:
- 标准库导入。
- 相关第三方进口。
- 本地应用程序/库特定的导入。
- 函数名和变量名应为小写并用下划线分隔
- 类名使用 CapWords 约定。
20 — 如何使用具有 2GB RAM 的计算机在 Python 中读取 8GB 文件?
此解决方案适用于任何大型(甚至更大)文件。
当你打开文件时,你需要做的就是将文件对象用作迭代器:在循环此文件对象时,你将一次获取一行,并且前面的行将从内存中清除(即它们是垃圾收集)。
这样,文件将永远不会完全加载到内存中。
with open("./large_dataset.txt") as input_file:
for line in input_file:
process_line(line)感谢阅读
这是我在面试中经常看到的一些问题的概述。我希望你从文章中学到了一些东西。20 个 Python 面试题来挑战你的知识
相关推荐
- Excel技巧:SHEETSNA函数一键提取所有工作表名称批量生产目录
-
首先介绍一下此函数:SHEETSNAME函数用于获取工作表的名称,有三个可选参数。语法:=SHEETSNAME([参照区域],[结果方向],[工作表范围])(参照区域,可选。给出参照,只返回参照单元格...
- Excel HOUR函数:“小时”提取器_excel+hour函数提取器怎么用
-
一、函数概述HOUR函数是Excel中用于提取时间值小时部分的日期时间函数,返回0(12:00AM)到23(11:00PM)之间的整数。该函数在时间数据分析、考勤统计、日程安排等场景中应用广泛。语...
- Filter+Search信息管理不再难|多条件|模糊查找|Excel函数应用
-
原创版权所有介绍一个信息管理系统,要求可以实现:多条件、模糊查找,手动输入的内容能去空格。先看效果,如下图动画演示这样的一个效果要怎样实现呢?本文所用函数有Filter和Search。先用filter...
- FILTER函数介绍及经典用法12:FILTER+切片器的应用
-
EXCEL函数技巧:FILTER经典用法12。FILTER+切片器制作筛选按钮。FILTER的函数的经典用法12是用FILTER的函数和切片器制作一个筛选按钮。像左边的原始数据,右边想要制作一...
- office办公应用网站推荐_office办公软件大全
-
以下是针对Office办公应用(Word/Excel/PPT等)的免费学习网站推荐,涵盖官方教程、综合平台及垂直领域资源,适合不同学习需求:一、官方权威资源1.微软Office官方培训...
- WPS/Excel职场办公最常用的60个函数大全(含卡片),效率翻倍!
-
办公最常用的60个函数大全:从入门到精通,效率翻倍!在职场中,WPS/Excel几乎是每个人都离不开的工具,而函数则是其灵魂。掌握常用的函数,不仅能大幅提升工作效率,还能让你在数据处理、报表分析、自动...
- 收藏|查找神器Xlookup全集|一篇就够|Excel函数|图解教程
-
原创版权所有全程图解,方便阅读,内容比较多,请先收藏!Xlookup是Vlookup的升级函数,解决了Vlookup的所有缺点,可以完全取代Vlookup,学完本文后你将可以应对所有的查找难题,内容...
- 批量查询快递总耗时?用Excel这个公式,自动计算揽收到签收天数
-
批量查询快递总耗时?用Excel这个公式,自动计算揽收到签收天数在电商运营、物流对账等工作中,经常需要统计快递“揽收到签收”的耗时——比如判断某快递公司是否符合“3天内送达”的服务承...
- Excel函数公式教程(490个实例详解)
-
Excel函数公式教程(490个实例详解)管理层的财务人员为什么那么厉害?就是因为他们精通excel技能!财务人员在日常工作中,经常会用到Excel财务函数公式,比如财务报表分析、工资核算、库存管理等...
- Excel(WPS表格)Tocol函数应用技巧案例解读,建议收藏备用!
-
工作中,经常需要从多个单元格区域中提取唯一值,如体育赛事报名信息中提取唯一的参赛者信息等,此时如果复制粘贴然后去重,效率就会很低。如果能合理利用Tocol函数,将会极大地提高工作效率。一、功能及语法结...
- Excel中的SCAN函数公式,把计算过程理清,你就会了
-
Excel新版本里面,除了出现非常好用的xlookup,Filter公式之外,还更新一批自定义函数,可以像写代码一样写公式其中SCAN函数公式,也非常强大,它是一个循环函数,今天来了解这个函数公式的计...
- Excel(WPS表格)中多列去重就用Tocol+Unique组合函数,简单高效
-
在数据的分析和处理中,“去重”一直是绕不开的话题,如果单列去重,可以使用Unique函数完成,如果多列去重,如下图:从数据信息中可以看到,每位参赛者参加了多项运动,如果想知道去重后的参赛者有多少人,该...
- Excel(WPS表格)函数Groupby,聚合统计,快速提高效率!
-
在前期的内容中,我们讲了很多的统计函数,如Sum系列、Average系列、Count系列、Rank系列等等……但如果用一个函数实现类似数据透视表的功能,就必须用Groupby函数,按指定字段进行聚合汇...
- Excel新版本,IFS函数公式,太强大了!
-
我们举一个工作实例,现在需要计算业务员的奖励数据,右边是公司的奖励标准:在新版本的函数公式出来之前,我们需要使用IF函数公式来解决1、IF函数公式IF函数公式由三个参数组成,IF(判断条件,对的时候返...
- Excel不用函数公式数据透视表,1秒完成多列项目汇总统计
-
如何将这里的多组数据进行汇总统计?每组数据当中一列是不同菜品,另一列就是该菜品的销售数量。如何进行汇总统计得到所有的菜品销售数量的求和、技术、平均、最大、最小值等数据?不用函数公式和数据透视表,一秒就...
- 一周热门
- 最近发表
-
- Excel技巧:SHEETSNA函数一键提取所有工作表名称批量生产目录
- Excel HOUR函数:“小时”提取器_excel+hour函数提取器怎么用
- Filter+Search信息管理不再难|多条件|模糊查找|Excel函数应用
- FILTER函数介绍及经典用法12:FILTER+切片器的应用
- office办公应用网站推荐_office办公软件大全
- WPS/Excel职场办公最常用的60个函数大全(含卡片),效率翻倍!
- 收藏|查找神器Xlookup全集|一篇就够|Excel函数|图解教程
- 批量查询快递总耗时?用Excel这个公式,自动计算揽收到签收天数
- Excel函数公式教程(490个实例详解)
- Excel(WPS表格)Tocol函数应用技巧案例解读,建议收藏备用!
- 标签列表
-
- 外键约束 oracle (36)
- oracle的row number (32)
- 唯一索引 oracle (34)
- oracle in 表变量 (28)
- oracle导出dmp导出 (28)
- 多线程的创建方式 (29)
- 多线程 python (30)
- java多线程并发处理 (32)
- 宏程序代码一览表 (35)
- c++需要学多久 (25)
- css class选择器用法 (25)
- css样式引入 (30)
- css教程文字移动 (33)
- php简单源码 (36)
- php个人中心源码 (25)
- php小说爬取源码 (23)
- 云电脑app源码 (22)
- html画折线图 (24)
- docker好玩的应用 (28)
- linux有没有pe工具 (34)
- 可以上传视频的网站源码 (25)
- 随机函数如何生成小数点数字 (31)
- 随机函数excel公式总和不变30个数据随机 (33)
- 所有excel函数公式大全讲解 (22)
- 有动图演示excel函数公式大全讲解 (32)
