python pyproject.toml

news2026/4/29 14:45:11
聊聊Python的build这玩意儿其实不算新面孔早在Python打包工具链里就默默存在了很久只不过近几年才因为更好的规范性和可扩展性被推到台前。简单说它是Python官方推荐的打包流程前端工具——不是替代setuptools而是把那些繁琐的配置和构建选择封装成一套更统一的调用方式。1它是什么build这个工具本质上是PEP 517和PEP 518这两个提案的产物。以前Python打包基本依赖setup.py直接运行但那种方式把配置、构建和安装逻辑揉在一起不同项目用的构建后端不一样比如setuptools、flit、poetry调用方式也七零八落。build提供了一个标准化的入口无论项目用哪个后端只要它符合PEP 517规范就都能通过python -m build来生成发行包。你可能会觉得这和直接跑python setup.py sdist bdist_wheel有什么区别区别在于中间隔了一层抽象。build不关心你的构建后端是什么它只负责根据pyproject.toml里的声明去调用对应的后端。就像去餐厅吃饭你不需要知道后厨用的是什么锅只要告诉服务员你要什么菜就行。2他能做什么build的主要任务就是生成两种常见的分发格式源码分发包sdist和二进制包wheel。这个过程中它会做几件具体的事情读取pyproject.toml的[build-system]部分确定用哪个构建后端比如setuptools在临时干净的虚拟环境中安装构建依赖运行后端提供的build_wheel和build_sdist接口把产出的.tar.gz和.whl文件放到指定目录默认是dist/实际工作中你经常需要先构建项目再测试或分发。比如团队里有一个项目的依赖比较复杂build可以确保每次构建都在隔离环境中完成不会污染当前环境的依赖。这对于CI/CD流水线尤其重要——每次都在干净的沙箱里构建能避免本地环境遗留的包对产出造成影响。另外build在纯源码分发和wheel分发之间也做了很实用的区分。有些项目依赖C扩展在本地构建wheel可能需要编译器这时候build可以只生成sdist让最终用户在自己的平台上编译。而在打包纯Python项目时直接生成wheel会更快用户安装起来也方便。3怎么使用使用build其实很简单基本上是三步走先安装它pip install build然后在项目根目录有pyproject.toml的地方运行python -m build它会自动在当前目录寻找pyproject.toml构建完毕后把文件放到dist/目录下。如果只想构建其中一种格式也可以加参数python -m build --sdist只生成源码包python -m build --wheel只生成wheel包有个经常被忽略的细节build默认不会生成sdist如果你同时需要两种格式直接用不带参数的命令是最保险的。另外如果项目的构建后端需要安装额外的依赖才能运行build会自动处理用户不需要手动去装那些依赖。还有一个实用技巧如果你在做持续集成可以加上--no-isolation参数跳过临时环境直接在当前环境构建。但建议只在调试时用正式构建还是要用隔离模式避免环境差异导致的问题。4最佳实践说说我个人在项目中摸索出来的几个习惯第一pyproject.toml一定要配置好[build-system]部分。很多新手只配置了元数据却忘了声明构建后端。一个典型的配置长这样[build-system] requires [setuptools61.0, wheel] build-backend setuptools.build_meta这里requires是构建这个项目需要预先安装的依赖build-backend指定了调用的后端。如果用的是flit或poetry对应的后端名称会不一样。第二尽量在你的持续集成脚本里用build而不是直接调后端的命令。比如在GitHub Actions中-name:Build packagerun:|pip install build python -m build --wheel这样做的好处是将来如果换了构建后端只需要改pyproject.toml构建命令完全不需要动。第三做版本发布时多验证一下构建的产物的质量。比如在构建后可以立刻把生成的wheel安装到临时环境测试pipinstalldist/*.whl# 然后执行一些导入测试python-cfrom mypackage import __version__; print(__version__)这样能避免发布后才发现漏文件或版本号错误。第四如果项目的源码里包含非Python文件比如数据文件、模板记得在pyproject.toml里配置好include或使用MANIFEST.in。很多开发者在这上面栽过跟头构建出来的包少了配置文件。5和同类技术对比把build跟其他打包工具放在一起看更能体会它的定位差异。最直接的对比对象是python setup.py系列命令。那套老办法的问题是它把构建逻辑硬编码在setup.py里而且只能配合setuptools用。如果你的项目改用poetry管理依赖setup.py就没什么用了。而build是完全中立的调度器。跟poetry内置的构建功能poetry build相比poetry主要是依赖管理器附带打包功能它的构建逻辑是锁死的只能用自己的后端。而build可以和任何符合PEP 517规范的后端配合你可以在项目里用poetry管理依赖但在构建时依然可以用python -m build。跟flit自带的构建命令flit build类似build更专注于构建这个环节本身。很多时候你发现自己用poetry或flit只是为了它的构建功能其实完全可以只用一个简单的构建后端加build这样项目依赖更少也更灵活。还有一个轻量级的工具叫tqdm的打包方式不过那更偏向于微服务部署场景了。实际项目选型的时候可以这样考虑如果团队里已经用了poetry来管理依赖和版本号那么poetry build就已经够用了没必要额外再加build。如果项目配置简单用setuptools加上pyproject.toml就能搞定那build就是最轻量的选择。如果项目比较复杂# # 关于Python的pyproject.toml聊聊我的理解前段时间帮一个朋友整理他的Python项目他还在用setup.py项目里requirements.txt、setup.cfg、MANIFEST.in各种文件混在一起看着就头疼。我建议他试试pyproject.toml他问我这东西到底好在哪。其实这个问题挺有意思的因为pyproject.toml经历了好几个版本的演变不同阶段它在项目里的角色也不太一样。它到底是什么简单说pyproject.toml是一个用TOML格式写的配置文件放在项目根目录用来告诉Python世界这个项目该怎么构建、怎么打包、需要哪些依赖。TOML这门语言有点像INI但比INI多了层级结构读起来很直观——键值对、数组、表基本就这三板斧。这个文件最初来自PEP 5182016年目的是解决一个长期困扰大家的问题一个Python项目到底应该用什么工具来构建。以前没有统一标准有人用setuptools有人用flit有人写poetry每个工具都要自己的配置方式。pyproject.toml的出现相当于给了大家一个统一的入口不管底层用哪个构建工具配置都放在同一个地方。后来PEP 621又进一步规范了pyproject.toml的格式让项目元数据名字、版本、作者这些信息也有了统一的存放位置。现在最新版的pip、build这些工具都能直接识别pyproject.toml。它到底能做什么说到能力我觉得可以从三个层面来看。第一层也是最基础的替代setup.py和setup.cfg。虽然写一个简单的setup.py难不到哪里去但要维护一个包含复杂配置的setup.py那种感觉就像在写重复代码。我见过一个项目setup.py里用了一大堆条件判断来处理不同平台的依赖读起来简直像在读天书。pyproject.toml用声明式的配置方式把构建工具的配置放进了独立的条目下面清晰得多。第二层管理项目依赖。传统的做法是requirements.txt配上pip freeze但这种方式的问题是谁来确定这些依赖是项目本身需要的还是只是开发过程中临时装的pyproject.toml支持区分不同场景的依赖——运行需要的、测试需要的、文档需要的、开发工具链需要的。这个类比打包行李出门住酒店真正需要的底裤牙刷运行依赖和那些看着好玩但可能用不上的东西可选依赖放在不同的口袋里需要的时候才拿出来。第三层定义构建后端。这是pyproject.toml最核心的革新之处。以前你要用某个构建工具得手动装好然后调用它的命令行。现在只需要在pyproject.toml里声明一下“我用Setuptools作为构建后端”pip就会自动处理剩下的事情。这就好比你跟外卖平台说“我要吃宫保鸡丁”平台自己会去联系对应的餐厅、安排配送你不需要知道厨房里发生了什么。怎么上手用实际用起来并不复杂。假设新建一个项目叫my-tool。在项目根目录创建一个pyproject.toml文件里面至少要有一个[build-system]条目指定构建后端。选择用Setuptools的话可能还需要配套的setup.cfg或者也能集中到pyproject.toml里。不过现在越来越多的项目选择用Poetry或者Flit这两种工具都支持把几乎所有配置写在pyproject.toml里不需要额外配置文件。比如说用Poetry创建一个项目[build-system] requires [poetry-core] build-backend poetry.core.masonry.api [tool.poetry] name my-tool version 0.1.0 description 一个简单的示例项目 authors [你的名字 youexample.com] [tool.poetry.dependencies] python ^3.8 requests ^2.28.0 [tool.poetry.group.dev.dependencies] pytest ^7.0.0这里有几个有意思的地方。requires定义了构建时需要的包build-backend指定了实际干活的构建后端。[tool.poetry]开头的是Poetry自己的配置空间因为pyproject.toml也允许不同工具用tool.工具名来定义自己的配置。运行python -m build或者poetry build就能生成wheel包和sdist包了。如果用Poetry还能用poetry add requests自动更新pyproject.toml。一些值得注意的做法Pyproject.toml虽然灵活但用得不好也会制造麻烦。有几个经验可以参考。每个项目最好只用一种构建工具。不要把Poetry和Setuptools的配置混在一起写容易冲突。选定了就坚持下去除非有特别好的理由要切换。注意Python版本的兼容性。如果你的项目需要支持Python 3.6或更老的版本那么pyproject.toml可能会带来一些限制因为老版本的pip对它的支持不够完善。不过2023年的现在大部分用户已经用的是比较新的Python了。私有包索引需要配置清楚。如果公司内部有PyPI镜像或者用GitLab的包 registry需要在pyproject.toml里加上相关的配置。比如Poetry的方式是在[[tool.poetry.source]]里面配置。依赖版本范围要合理。初学者容易把依赖写死成1.2.3导致用户没法升级补丁版本。推荐用加的方式比如requests2.28.0,3.0.0。这背后的逻辑是希望用户能用上最新的安全修复但也不想看到一个不兼容的大版本更新时项目炸掉。和其他配置方式的对比最后聊聊pyproject.toml跟其他方案的比较。vs setup.py setup.cfg传统方式。setup.py实际上是一个可执行的Python脚本意味着你可以在里面写任意的Python代码来做构建相关的判断。这既是优势也是劣势——灵活性高但容易写出难以维护的配置。pyproject.toml牺牲了这种灵活性换来了可读性和工具的一致性。对于绝大部分项目来说收益大于损失。vs requirements.txtrequirements.txt本质上是一个裸的依赖列表没有版本控制的粒度也没有区分场景的能力。pyproject.toml的依赖管理更结构化。但值得注意的是requirements.txt并没有被取代。很多项目依然会生成一个requirements.txt文件只是它现在是从pyproject.toml锁定的版本中导出来的用于部署环境。角色变了从一个主配置文件变成了一个部署工件。vs Pipfile / PipenvPipenv是早期试图统一Python依赖管理的尝试用Pipfile和Pipfile.lock。但它的设计复杂度过高社区接受度一直不太高。pyproject.toml的设计思路更轻量更接近符合直觉的工作流。vs setup.py的区别setup.py是可以带逻辑的。比如根据操作系统不同选择不同的依赖这在pyproject.toml里需要用构建工具提供的机制来实现比如Setuptools的extras_require 平台标记。一般来说现在不太建议在构建流程里写太复杂的逻辑这样做的好处是构建的可复制性更强。说到底pyproject.toml不是一个银弹它解决的是Python项目长期以来在构建和依赖管理上缺少统一标准的问题。就像每个人家里都会有一个固定的地方放钥匙和钱包——pyproject.toml给了Python项目一个约定俗成的说明书的位置。比如有多个构建阶段或前后端混编那build的隔离构建环境能帮你省去很多脏活累活。说到底build解决的不是能不能构建的问题而是怎么构建得更好的问题。它让构建过程和特定工具解耦这在多人协作和维护长期项目时价值会越来越明显。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2565818.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…