当前位置: 首页 > news >正文

Python常见面试题的详解4

1. 单例模式的实现方式

  • 要点:Python 有多种实现单例模式的方法。模块由于其特性天然支持单例,首次导入生成对象,后续导入直接复用。通过装饰器可以控制实例的创建,元类能借助 __call__ 方法管理实例化过程,重写类的 __new__ 方法也能保证实例的唯一性。
  • 示例

python

def singleton(cls):
    instances = {}
    def get_instance(*args, **kwargs):
        # 如果类尚未在实例字典中,就创建一个新实例
        if cls not in instances:
            instances[cls] = cls(*args, **kwargs)
        # 无论是否新创建,都返回对应的实例
        return instances[cls]
    return get_instance

@singleton
class MyClass:
    pass

2. Lambda 函数的概念

  • 要点:Lambda 函数是一种匿名函数,专门用于简化单行函数的定义。其语法为 lambda 参数: 表达式
  • 示例

python

# 使用 lambda 定义一个加法函数
add = lambda x, y: x + y
print(add(2, 3))  # 输出 5,展示函数的计算结果

3. 类型转换方法

  • 要点:Python 可通过内置函数实现类型转换,如 int(x) 用于转换为整数,float(x) 转换为浮点数,str(x) 转换为字符串,list(x) 转换为列表,tuple(x) 转换为元组等。
  • 示例

python

num_str = "123"
# 将字符串类型的数字转换为整数类型
num_int = int(num_str)  
print(num_int)  # 输出 123,验证转换结果

4. 反序迭代序列的方法

  • 要点:可以使用 reversed() 函数获取逆序迭代器,或者通过切片 [::-1] 生成逆序的新序列来实现反序迭代。
  • 示例

python

lst = [1, 2, 3]
# 使用 reversed() 函数反序迭代列表
for item in reversed(lst):
    print(item)  
# 输出 3, 2, 1,展示反序迭代的结果

5. Tuple 和 List 的相互转换

  • 要点:列表转元组可使用 tuple(list),元组转列表可使用 list(tuple)
  • 示例

python

my_list = [1, 2, 3]
# 将列表转换为元组
my_tuple = tuple(my_list)  
print(my_tuple)  # 输出 (1, 2, 3),展示转换后的元组

5. 删除列表重复元素的方法

  • 要点:利用 set() 可实现去重,但会丢失元素顺序;在 Python 3.7 及以上版本中,有序字典能保持插入顺序,可用于实现有序去重。
  • 示例(保持顺序)

python

def deduplicate(lst):
    seen = set()
    # 遍历列表,若元素不在已访问集合中,则保留并添加到集合中
    return [x for x in lst if not (x in seen or seen.add(x))]

6. 删除文件的方法

  • 要点:使用 os.remove() 函数可以删除指定的文件。
  • 示例

python

import os
# 尝试删除名为 "file.txt" 的文件
os.remove("file.txt")  

7. 生成随机数的方法

  • 要点:借助 Python 的 random 模块可以生成随机数,例如使用 random.randint(a, b) 生成指定范围内的随机整数。
  • 示例

python

import random
# 生成 1 到 100 之间的随机整数并输出
print(random.randint(1, 100))  

7. 发送邮件的方法

  • 要点:使用 smtplibemail 库来实现邮件发送功能,需要构建邮件内容、设置邮件主题、发件人和收件人等信息,并进行登录和发送操作。
  • 示例

python

import smtplib
from email.mime.text import MIMEText

# 创建邮件内容对象
msg = MIMEText("邮件内容")
msg["Subject"] = "主题"
msg["From"] = "发件人"
msg["To"] = "收件人"

# 使用上下文管理器连接邮件服务器并发送邮件
with smtplib.SMTP("smtp.example.com", 587) as server:
    server.login("user", "password")
    server.send_message(msg)

8. 静态代码分析工具

  • 要点:常见的 Python 静态代码分析工具包括 Pylint(用于检查代码风格和错误)、Flake8(结合了 PyFlakesPEP8 检查)、Mypy(用于静态类型检查)。

9. 交换元素使两序列和差最小

有两个序列 a,b, 大小都为 n,序列元素的值任意整形数, 无序; 要求: 通过交换 a,b 中的元素, 使[序列 a 元素的和]与[序列 b 元素的和]之间的差最小

  • 要点:要使交换序列 ab 中的元素后,两个序列元素和的差最小,我们可以采用以下步骤:

  1. 合并序列并计算总和:将序列 ab 合并为一个新序列 combined,然后计算这个新序列的总和 total

  2. 确定目标和:我们的目标是从 combined 中选出 n 个元素,使得这 n 个元素的和尽可能接近 total / 2

  3. 使用动态规划:通过动态规划的方法来寻找最接近目标和的 n 个元素的组合。

  4. 交换元素:根据动态规划得到的结果,确定哪些元素需要从一个序列交换到另一个序列,从而实现和差最小。

  • 示例

python

def minimal_diff(a, b):
    n = len(a)
    combined = a + b
    total = sum(combined)
    target = total // 2

    # 创建一个三维动态规划数组 dp
    # dp[i][j][k] 表示在前 i 个元素中选择 j 个元素,其和是否可以达到 k
    dp = [[[False] * (total + 1) for _ in range(n + 1)] for _ in range(2 * n + 1)]
    dp[0][0][0] = True

    # 填充动态规划数组
    for i in range(1, 2 * n + 1):
        for j in range(min(i, n) + 1):
            for k in range(total + 1):
                # 不选择第 i 个元素
                dp[i][j][k] = dp[i - 1][j][k]
                if j > 0 and k >= combined[i - 1]:
                    # 选择第 i 个元素
                    dp[i][j][k] = dp[i][j][k] or dp[i - 1][j - 1][k - combined[i - 1]]

    # 找到最接近目标和的和
    closest_sum = 0
    for k in range(target, -1, -1):
        if dp[2 * n][n][k]:
            closest_sum = k
            break

    # 回溯找出选择的元素
    selected = [False] * (2 * n)
    i, j, k = 2 * n, n, closest_sum
    while i > 0:
        if not dp[i - 1][j][k]:
            selected[i - 1] = True
            j -= 1
            k -= combined[i - 1]
        i -= 1

    # 根据选择的元素重新划分序列 a 和 b
    new_a = [combined[i] for i in range(2 * n) if selected[i]]
    new_b = [combined[i] for i in range(2 * n) if not selected[i]]

    return new_a, new_b

# 示例使用
a = [1, 2, 3]
b = [4, 5, 6]
new_a, new_b = minimal_diff(a, b)
print("序列 a 交换后的元素:", new_a)
print("序列 b 交换后的元素:", new_b)
print("序列 a 的和:", sum(new_a))
print("序列 b 的和:", sum(new_b))
print("和的差:", abs(sum(new_a) - sum(new_b)))
  • 说明:
  1. 动态规划数组 dpdp[i][j][k] 表示在前 i 个元素中选择 j 个元素,其和是否可以达到 k
  2. 填充动态规划数组:通过三重循环遍历所有可能的状态,更新 dp 数组的值。
  3. 找到最接近目标和的和:从目标和 target 开始递减搜索,找到第一个满足 dp[2 * n][n][k]Truek 值。
  4. 回溯找出选择的元素:根据 dp 数组的值,回溯找出哪些元素被选中。
  5. 重新划分序列:根据选择的元素,重新划分序列 ab

10. 正则表达式 <.*><.*?> 的区别

  • 要点<.*> 是贪婪匹配模式,会匹配从第一个 < 到最后一个 > 的所有内容;<.*?> 是非贪婪匹配模式,遇到第一个 > 就会结束匹配。
  • 示例

python

# 对于字符串 "<a><b></b></a>"
# 贪婪匹配 <.*> 会匹配整个字符串 "<a><b></b></a>"
# 非贪婪匹配 <.*?> 会分别匹配 "<a>" 和 "<b>" 等单独的标签

相关文章:

  • 【Day40 LeetCode】动态规划DP 回文子串问题
  • 如何在微信小程序中使用 Lottie 动画
  • 机器学习 网络安全
  • Hadoop 简介及其hdfs常用命令
  • LPDDR4、LPDDR5物理结构和信号定义区别
  • UniApp 的页面结构是怎样的?
  • 最新国内 ChatGPT Plus/Pro 获取教程
  • 【免费送书活动】《MySQL 9从入门到性能优化(视频教学版)》
  • 使用时间盲注与布尔盲注获取数据库名,表名及列名
  • 深度剖析观察者模式:从理论到实战的Java实现
  • 基于IOS实现各种倒计时功能
  • Linux udp poll函数
  • 行内元素和块级元素
  • New Game--(单调队列)
  • 如何设置linux系统时间?
  • USART串口协议
  • Java知识速记:Exception与Error的区别
  • c++:STL介绍
  • Rank-Analysis 预组队识别(英雄联盟)
  • Qwen2-VL 的重大省级,Qwen 发布新旗舰视觉语言模型 Qwen2.5-VL
  • b2b网站推广/百度免费发布信息
  • 建一个多用户团购网站需要多少钱/个人怎么接外贸订单
  • 菠菜网站搭建怎么做/营销策略包括哪些方面
  • 哈尔滨龙彩做网站多少钱/软文推广
  • 做网站推广广告/长春seo排名优化
  • 福田做网站价格/p2p万能搜索引擎