百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

让你的Python代码更易读:7个提升函数可读性的实用技巧

wptr33 2025-07-08 23:41 15 浏览

如果你正在阅读这篇文章,很可能你已经用Python编程有一段时间了。今天,让我们聊聊可以提升你编程水平的一件事:编写易读的函数。

请想一想:我们花在阅读代码上的时间大约是写代码的10倍。所以,每当你创建一个清晰直观的函数时,其实是在为自己和团队节省时间和减少挫败感。

本文将带你了解七个实用技巧,帮助你把晦涩难懂的代码转变为清晰、易维护的函数。我们会通过前后对比示例,并解释这些改动为何重要。让我们开始吧!


1. 使用有描述性的函数名和参数名

函数名应该是清晰描述所执行动作的动词,参数名也应当具有描述性。

反面示例
看看这个函数,你能看出它干什么吗?

def process(d, t):
    return d * (1 + t/100)

"process" 这个名字很含糊,单字母参数"d"和"t"完全看不出用途。它是在计算折扣?加利息?不看代码其他部分,根本无法知道。

正面示例
这个版本就一目了然:我们正在对价格应用税率。

def apply_tax_to_price(price, tax_rate):
    return price * (1 + tax_rate/100)

函数名准确描述了所做的动作,参数名也清晰指出了每个值代表的含义。即使是不熟悉代码的人,也能一眼看懂。


2. 限制参数数量

参数过多的函数难以理解,也容易出错。如果需要传递多个相关值,应该有逻辑地进行分组。

反面示例
这个函数有9个参数:

def send_notification(user_id, email, phone, message, subject, 
                     priority, send_email, send_sms, attachment):
    # 代码实现...

调用这个函数时,你必须记住所有参数的顺序,非常容易出错。而且也不清楚哪些参数是必需的,哪些是可选的。

像 send_notification(42, "user@example.com", "+1234567890", "Hello", "Greeting", 2, True, False, None) 这样的调用,看不出每个值的含义,除非查阅函数定义。

正面示例
通过将相关参数分组,减少参数数量:

def send_notification(user, notification_config, message_content):
    """
    根据配置向用户发送通知。

    参数:
    - user: 包含联系信息的User对象
    - notification_config: 包含通知偏好的NotificationConfig对象
    - message_content: 包含主题、正文和附件的MessageContent对象
    """
    # 代码实现...

现在调用 send_notification(user, config, message) 时,每个参数的含义一目了然,也更灵活。如果将来需要添加新选项,只需在 NotificationConfig 类中扩展即可,无需更改函数签名。


3. 编写清晰且有用的文档字符串(Docstring)

好的文档字符串应说明函数的作用、输入输出及可能的副作用。不要只是重复函数名!

反面示例
这个文档字符串毫无意义:

def validate_email(email):
    """This function validates email."""
    # 代码实现...

它只是重复了函数名,没有任何附加信息。

我们不知道"validates"具体做什么:只是检查格式?验证域名存在?联系邮件服务器?也不知道返回什么,是否会抛出异常。

正面示例
这个文档字符串信息清晰有用:

def validate_email(email: str) -> bool:
    """
    检查邮箱地址格式是否有效。

    参数:
    - email: 要验证的邮箱字符串

    返回:
    - 如果邮箱格式有效返回True,否则返回False

    注意:
    - 本验证仅检查格式,不验证地址是否真实存在
    """
    # 代码实现...

具体说明了:

  • 只检查邮箱格式
  • 输入参数类型为字符串
  • 返回布尔值
  • 限定了功能范围(只检格式)
  • 类型注解进一步表明输入输出类型

4. 每个函数只做一件事

函数应专注于单一职责。如果你用“和”来描述一个函数的作用,那它很可能做得太多了。

反面示例
你一定会同意,这个函数确实做了太多事情:

def process_order(order):
    # 验证订单
    # 更新库存
    # 收款
    # 发送确认邮件
    # 更新分析数据

它同时处理验证、库存管理、支付、通知和数据分析。这样做的坏处:

  • 难以测试,需要模拟许多依赖
  • 难以维护,任何一处变化都影响整体
  • 复用性差,比如单独复用验证逻辑就做不到

正面示例
将其拆分为单一职责函数:

def process_order(order):
    """从验证到确认处理客户订单。"""
    validated_order = validate_order(order)
    update_inventory(validated_order)
    payment_result = charge_customer(validated_order)
    if payment_result.is_successful:
        send_confirmation_email(validated_order, payment_result)
        update_order_analytics(validated_order)
    return OrderResult(validated_order, payment_result)

现在,每个任务都有专属函数:

  • 单一职责函数便于单独测试
  • 变更如邮件逻辑,只需改对应函数
  • 主函数结构像伪代码,整体流程一目了然

5. 使用类型注解增加清晰度

Python的类型提示让代码自文档化,有助于在运行前发现错误。

反面示例
这个函数虽然能用,但不够清晰:

def calculate_final_price(price, discount):
    return price * (1 - discount / 100)

discount的单位是什么?百分比还是小数?能否为负?返回值是什么?

没有类型注解,后续开发者可能会传错值或误用返回结果。

正面示例
类型注解让输入输出一目了然:

def calculate_final_price(price: float, discount_percentage: float) -> float:
    """
    计算应用折扣后的最终价格。

    参数:
    - price: 商品原价
    - discount_percentage: 要应用的折扣百分比(0-100)

    返回:
    - 折后价
    """
    return price * (1 - discount_percentage / 100)

参数名 discount_percentage 也表明应传入百分数(如20表示20%),不是小数(0.2)。文档字符串进一步说明了取值范围(0-100)。


6. 明智使用默认参数和关键字参数

默认参数让函数更灵活,但要注意正确使用。

反面示例
这个函数有不少问题:

def create_report(data, include_charts=True, format='pdf', output_path='report.pdf'):
    # 代码实现...

参数 format 与Python内置函数同名。硬编码的 output_path 意味着报告总会被覆盖。

所有参数都可以按位置传递,调用如 create_report(customer_data, False, 'xlsx') 不易理解那个False的含义。

正面示例
改进版如下:

def create_report(
    data: List[Dict[str, Any]],
    *,  # 强制后续参数用关键字
    include_charts: bool = True,
    format_type: Literal['pdf', 'html', 'xlsx'] = 'pdf',
    output_path: Optional[str] = None
) -> str:
    """
    根据提供的数据生成报告。

    参数:
    - data: 要包含在报告中的记录列表
    - include_charts: 是否生成图表
    - format_type: 报告输出格式
    - output_path: 报告保存路径(若为None,则使用默认路径)

    返回:
    - 生成的报告文件路径
    """
    if output_path is None:
        timestamp = datetime.now().strftime('%Y%m%d_%H%M%S')
        output_path = f"reports/report_{timestamp}.{format_type}"

    # 代码实现...

    return output_path

优势:

  • 用*强制后续参数必须用关键字,调用如 create_report(data, include_charts=False) 更清晰
  • 将 format 改为 format_type,避免与内置函数冲突
  • output_path 默认None,动态生成防止覆盖
  • 类型注解 Literal['pdf', 'html', 'xlsx'] 明确允许的格式类型

7. 用守卫子句(Guard Clause)做提前返回

用守卫子句提前处理边界情况,避免多层嵌套。

反面示例
以下函数嵌套条件太多,形成“金字塔型代码”:

def process_payment(payment):
    if payment.is_valid:
        if payment.amount > 0:
            if not payment.is_duplicate:
                # 真正的业务逻辑(被埋在多层内)
                return success_result
            else:
                return DuplicatePaymentError()
        else:
            return InvalidAmountError()
    else:
        return InvalidPaymentError()

主业务逻辑被深埋在嵌套之下,每加一个条件就多一层嵌套,代码越来越难以阅读。

正面示例
用守卫子句提前处理异常情况,主逻辑无需嵌套。

def process_payment(payment: Payment) -> PaymentResult:
    """
    处理支付事务。

    返回 PaymentResult 或抛出相应异常。
    """
    # 守卫子句做验证
    if not payment.is_valid:
        raise InvalidPaymentError("支付验证失败")

    if payment.amount <= 0:
        raise InvalidAmountError(f"无效支付金额: {payment.amount}")

    if payment.is_duplicate:
        raise DuplicatePaymentError(f"重复支付ID: {payment.id}")

    # 主要逻辑 - 无需嵌套
    transaction_id = submit_to_payment_processor(payment)
    update_payment_records(payment, transaction_id)
    notify_payment_success(payment)

    return PaymentResult(
        success=True,
        transaction_id=transaction_id,
        processed_at=datetime.now()
    )

每个验证单独清晰,出错即提前返回(或抛异常)。成功路径无嵌套,主业务逻辑显而易见。扩展性更强,新验证仅需加守卫子句,不必嵌套更深。


总结

花时间编写清晰、易读的函数,你的代码将具备:

  • 更少的bug
  • 更容易测试
  • 更易供他人(或半年后的自己)维护
  • 自带文档功能
  • 更可能被复用,而不是重写

请记住,代码被阅读的次数远大于被书写的次数。希望你能从本文中收获几个关键要点!

相关推荐

SQL轻松入门(5):窗口函数(sql语录中加窗口函数的执行)

01前言标题中有2个字让我在初次接触窗口函数时,真真切切明白了何谓”高级”?说来也是一番辛酸史!话说,我见识了窗口函数的强大后,便磨拳擦掌的要试验一番,结果在查询中输入语句,返回的结果却是报错,Wh...

28个SQL常用的DeepSeek提示词指令,码住直接套用

自从DeepSeek出现后,极大地提升了大家平时的工作效率,特别是对于一些想从事数据行业的小白,只需要掌握DeepSeek的提问技巧,SQL相关的问题也不再是个门槛。...

从零开始学SQL进阶,数据分析师必备SQL取数技巧,建议收藏

上一节给大家讲到SQL取数的一些基本内容,包含SQL简单查询与高级查询,需要复习相关知识的同学可以跳转至上一节,本节给大家讲解SQL的进阶应用,在实际过程中用途比较多的子查询与窗口函数,下面一起学习。...

SQL_OVER语法(sql语句over什么含义)

OVER的定义OVER用于为行定义一个窗口,它对一组值进行操作,不需要使用GROUPBY子句对数据进行分组,能够在同一行中同时返回基础行的列和聚合列。...

SQL窗口函数知多少?(sql窗口怎么执行)

我们在日常工作中是否经常会遇到需要排名的情况,比如:每个部门按业绩来排名,每人按绩效排名,对部门销售业绩前N名的进行奖励等。面对这类需求,我们就需要使用sql的高级功能——窗口函数。...

如何学习并掌握 SQL 数据库基础:从零散查表到高效数据提取

无论是职场数据分析、产品运营,还是做副业项目,掌握SQL(StructuredQueryLanguage)意味着你能直接从数据库中提取、分析、整合数据,而不再依赖他人拉数,节省大量沟通成本,让你...

SQL窗口函数(sql窗口函数执行顺序)

背景在数据分析中,经常会遇到按某某条件来排名、并找出排名的前几名,用日常SQL的GROUPBY,ORDERBY来实现特别的麻烦,有时甚至实现不了,这个时候SQL窗口函数就能发挥巨大作用了,窗...

sqlserver删除重复数据只保留一条,使用ROW_NUMER()与Partition By

1.使用场景:公司的小程序需要实现一个功能:在原有小程序上,有一个优惠券活动表。存储着活动产品数据,但因为之前没有做约束,导致数据的不唯一,这会使打开产品详情页时,可能会出现随机显示任意活动问题。...

SQL面试经典问题(一)(sql经典面试题及答案)

以下是三个精心挑选的经典SQL面试问题及其详细解决方案,涵盖了数据分析、排序限制和数据清理等常见场景。这些问题旨在考察SQL的核心技能,适用于初学者到高级开发者的面试准备。每个问题均包含清晰的...

SQL:求连续N天的登陆人员之通用解答

前几天发了一个微头条:...

SQL四大排序函数神技(sql中的排序是什么语句)

在日常SQL开发中,排序操作无处不在。当大家需要排序时,是否只会想到ORDERBY?今天,我们就来揭秘SQL中四个强大却常被忽略的排序函数:ROW_NUMBER()、RANK()、DENSE_RAN...

四、mysql窗口函数之row_number()函数的使用

1、窗口函数之row_number()使用背景窗口函数中,排序函数rank(),dense_rank()虽说都是排序函数,但是各有用处,假如像上章节说的“同组同分”两条数据,我们不想“班级名次”出现“...

ROW_NUMBER()函数(rownumber函数与rank区别)

ROW_NUMBER()是SQL中的一个窗口函数(WindowFunction)...

Dify「模板转换」节点终极指南:动态文本生成进阶技巧(附代码)Jinja2引擎解析

这篇文章是关于Dify「模板转换」节点的终极指南,解析了基于Jinja2模板引擎的动态文本生成技巧,涵盖多源文本整合、知识检索结构化、动态API构建及个性化内容生成等六大应用场景,助力开发者高效利用模...

Python 最常用的语句、函数有哪些?

1.#coding=utf-8①代码中有中文字符,最好在代码前面加#coding=utf-8②pycharm不加可能不会报错,但是代码最终是会放到服务器上,放到服务器上的时候运行可能会报错。③...