python中如何打包自己的Python项目?

python中如何打包自己的python项目?

在Python中打包自己的项目,最核心的思路是利用Python的包管理生态,尤其是

setuptools

这个工具链,来将你的代码、元数据和依赖项封装成一个可分发的格式,通常是

.whl

(wheel)或

.tar.gz

(source distribution)。这使得其他人,或者你自己在不同环境中,都能方便地安装和使用你的代码。

解决方案

要打包一个Python项目,现代且推荐的做法是围绕

pyproject.toml

文件进行配置,这让整个过程更加标准化和清晰。当然,如果你在维护一个老项目,可能还会遇到

setup.py

。但我们这里主要聊聊

pyproject.toml

首先,你需要一个合理的项目结构。我个人比较偏爱

src

布局,即把所有实际的Python代码放在一个名为

src

的子目录里,这样做的好处是能更好地模拟安装后的环境,避免一些常见的导入问题。比如:

my_project/├── src/│   └── my_package/│       ├── __init__.py│       └── main.py├── pyproject.toml├── README.md└── LICENSE

接下来,关键是创建

pyproject.toml

文件。这个文件定义了你的项目如何被构建以及它的元数据。一个基本的

pyproject.toml

可能看起来像这样:

立即学习“Python免费学习笔记(深入)”;

[build-system]requires = ["setuptools>=61.0"]build-backend = "setuptools.build_meta"[project]name = "my-awesome-package"version = "0.1.0"authors = [  { name="Your Name", email="your.email@example.com" },]description = "一个关于我的Python项目的简短描述。"readme = "README.md"requires-python = ">=3.8"classifiers = [    "Programming Language :: Python :: 3",    "License :: OSI Approved :: MIT License",    "Operating System :: OS Independent",]dependencies = [    "requests>=2.28.1",    "rich~=13.0",][project.urls]"Homepage" = "https://github.com/yourusername/my-awesome-package""Bug Tracker" = "https://github.com/yourusername/my-awesome-package/issues"[tool.setuptools.packages.find]where = ["src"] # 告诉setuptools去src目录找包

这里面,

[build-system]

部分告诉构建工具应该用什么来构建你的项目,通常是

setuptools

[project]

部分则包含了项目的核心元数据:名称、版本、作者、描述、依赖等等。

dependencies

字段非常重要,它列出了你的项目运行时所依赖的其他Python包。

[tool.setuptools.packages.find]

则是我在

src

布局下,告诉

setuptools

去哪里寻找我的Python包。

如果你有一些非Python文件需要包含在包里(比如配置文件、数据文件),

setuptools

会默认包含在

src

目录下的非Python文件。如果文件在其他地方,或者你需要更细致的控制,可能需要一个

MANIFEST.in

文件,但对于大多数项目,

pyproject.toml

的默认行为已经够用了。

配置好这些后,打包就非常简单了。确保你安装了

build

工具:

pip install build

。然后,在你的项目根目录(

pyproject.toml

所在的目录)运行:

python -m build

这个命令会在你的项目根目录下创建一个

dist/

目录,里面就会有你的

.whl

.tar.gz

文件。

.whl

是预编译的二进制分发包,安装更快;

.tar.gz

是源代码分发包,包含所有源代码和元数据。

打包过程本身并不复杂,真正的挑战往往在于如何合理组织代码、管理依赖,以及处理一些边缘情况,比如包含非Python资源。但只要遵循

pyproject.toml

的规范,大部分问题都能迎刃而解。

Python项目打包时,有哪些主流工具可以选择?它们之间有什么区别?

在Python的世界里,打包工具的选择其实还挺丰富的,但它们各有侧重,理解这些差异能帮助你根据项目需求做出最佳选择。

1.

setuptools

(搭配

pyproject.toml

setup.py

)

特点: 毫无疑问,它是Python生态的基石,几乎所有其他工具最终都可能依赖它来完成实际的构建工作。它提供了非常细致的控制能力,可以处理各种复杂的打包场景,比如C扩展、数据文件等。优点: 极度灵活,功能强大,社区支持广泛,兼容性最好。通过

pyproject.toml

配置,现代项目也能保持简洁。缺点: 配置起来相对复杂,尤其是当你需要处理一些非标准情况时。早期的

setup.py

脚本模式容易导致代码和配置混淆。我的看法: 对于需要最大兼容性、或者包含C扩展的库,

setuptools

依然是首选。它就像一把瑞士军刀,虽然有些功能不常用,但关键时刻总能派上用场。现在有了

pyproject.toml

,它的配置体验也大大改善了。

2.

Poetry

特点:

Poetry

不仅仅是一个打包工具,它更是一个完整的Python项目管理工具,集成了依赖管理、虚拟环境管理和打包发布。它非常注重确定性和易用性。优点: 极大地简化了依赖管理和虚拟环境操作。它使用

pyproject.toml

来定义所有内容,包括依赖锁定(

poetry.lock

),确保每次安装都得到相同的依赖版本。打包和发布命令也非常直观。缺点: 它的“全能”有时也意味着它的“固执”。如果你已经有了一套成熟的依赖管理流程,或者不喜欢它的某些默认行为,可能会觉得它有些限制。我的看法: 对于新的应用程序项目,尤其是那些需要严格依赖锁定的项目,

Poetry

是个非常棒的选择。它能让你的开发体验变得丝滑,减少很多因依赖冲突带来的头疼。但对于纯粹的库项目,我可能会考虑更轻量级的方案,或者依然用

setuptools

3.

Flit

特点:

Flit

专注于纯Python模块的打包,目标是提供一个极其简单、零配置的打包体验。它也使用

pyproject.toml

优点: 配置简单到令人发指,对于纯Python库,你只需要在

pyproject.toml

里写几行元数据,

Flit

就能帮你完成打包。它强制使用

src

布局,这本身也是一个好实践。缺点: 只能打包纯Python模块,不支持C扩展或其他复杂资源。我的看法: 如果你的项目是一个纯Python的库,没有C扩展,也不需要特别复杂的打包逻辑,

Flit

绝对值得一试。它的简洁性会让你爱不释手。

4.

Hatch

特点:

Hatch

是一个相对较新的项目管理工具,目标是成为Python项目的“一站式”解决方案,涵盖了虚拟环境、脚本运行、测试、打包和发布。它也基于

pyproject.toml

优点: 提供了非常全面的功能,并且设计上考虑了扩展性,允许用户自定义各种插件。它的配置比

Poetry

更灵活,但又比

setuptools

更集成。缺点: 作为一个较新的工具,社区成熟度可能不如

setuptools

Poetry

。功能丰富也意味着学习曲线可能稍长。我的看法:

Hatch

是一个很有潜力的工具,它试图在

Poetry

的集成性和

setuptools

的灵活性之间找到一个平衡点。如果你正在寻找一个现代、功能全面且可定制的项目管理工具,

Hatch

是一个值得关注的选项。

选择哪个工具,很大程度上取决于你的项目类型、团队偏好以及你对工具集成度的要求。对我而言,如果是纯库,

setuptools

with

pyproject.toml

提供足够的灵活性;如果是应用程序,

Poetry

的依赖管理优势很明显。

在Python项目中,如何有效地管理项目依赖和版本控制?

依赖管理和版本控制是项目健康运行的关键。我见过太多项目因为依赖问题而陷入泥潭,所以这块的处理,我觉得怎么强调都不过分。

1. 依赖声明:

pyproject.toml

dependencies

这是现代Python项目声明直接依赖的首选方式。在

pyproject.toml

[project]

部分,你可以列出项目运行所需的所有包及其版本要求。

[project]dependencies = [    "requests>=2.28.1,<3.0",  # 明确版本范围    "numpy~=1.23.0",          # 兼容版本,例如 1.23.x    "pandas",                 # 不指定版本,但通常不推荐在库中使用]

版本指定策略:

==1.2.3

: 精确版本,确保每次都安装特定版本。适用于应用程序,追求确定性。

>=1.2.3

: 最低版本要求。适用于库,允许用户使用更新的版本。

~=1.2.3

(或

~1.2.3

): 兼容版本,例如

~=1.2.3

意味着

>=1.2.3

<1.3.0

。这是一个很好的折衷方案,允许小版本更新,同时避免引入大的不兼容变更。

>=1.2.3,<2.0.0

: 显式指定一个版本范围,通常用于兼容某个大版本系列。我个人倾向于在库项目中多用

~=

>=X.Y.Z,<X+1.0.0

,而在应用程序中,会更倾向于精确锁定或由

Poetry

pip-tools

等工具生成锁文件。

2. 依赖锁定:

poetry.lock

requirements.txt

仅仅声明依赖是不够的,你还需要锁定它们及其所有传递性依赖的具体版本。这确保了在不同时间、不同机器上安装项目时,都能得到完全相同的依赖环境,避免“在我机器上没问题”的尴尬。

Poetry

poetry.lock

如果你使用

Poetry

,它会自动为你生成一个

poetry.lock

文件。这个文件包含了所有直接和间接依赖的精确版本和哈希值,确保了高度的确定性。

pip-tools

生成的

requirements.txt

如果你不使用

Poetry

,但仍希望锁定依赖,

pip-tools

是一个非常棒的工具。你可以在

requirements.in

中声明你的直接依赖,然后用

pip-compile

命令生成一个详细的

requirements.txt

文件,里面包含了所有依赖的精确版本。

  # requirements.in  requests  rich  # 生成 requirements.txt  pip-compile requirements.in

然后安装时使用

pip install -r requirements.txt

3. 虚拟环境(Virtual Environments):这是Python依赖管理的基础。永远不要在全局Python环境中安装项目依赖。使用

venv

conda

Poetry

自带的虚拟环境,为每个项目创建一个隔离的Python环境。这可以防止不同项目之间的依赖冲突。

# 使用 venvpython -m venv .venvsource .venv/bin/activate  # Linux/macOS.venvScriptsactivate     # Windowspip install -r requirements.txt

4. 版本控制(Semantic Versioning – SemVer):为你的项目本身以及你所依赖的第三方库选择合理的版本控制策略,尤其是语义化版本(

MAJOR.MINOR.PATCH

)非常重要。

MAJOR

版本:不兼容的API变更。

MINOR

版本:向下兼容的新功能。

PATCH

版本:向下兼容的bug修复。遵循SemVer能让你的用户更好地理解你的版本更新意味着什么,并帮助他们在升级时做出明智的决策。

5. 挑战与实践:

依赖冲突: 当你的项目依赖的两个库又分别依赖同一个第三方库的不同版本时,就可能出现冲突。这时,你需要权衡,看能否升级其中一个依赖,或者寻找替代方案。锁文件(

poetry.lock

requirements.txt

)能帮助你及早发现这些冲突。环境隔离: 始终强调虚拟环境的重要性。它能让你在本地开发时避免“依赖地狱”。自动化: 将依赖管理集成到CI/CD流程中,例如,在每次构建时都检查依赖是否最新,或者自动更新锁文件。

对我来说,依赖管理是一个持续的过程。它不是一次性的配置,而是随着项目演进需要不断审视和调整的。使用合适的工具和策略,可以大大减少这方面的摩擦。

如何将我的Python包发布到PyPI或一个私有仓库?

将你的Python包发布出去,无论是公开的PyPI还是私有的仓库,都是为了让其他人(或你自己)能够方便地安装和使用它。这个过程通常涉及构建、认证和上传三个主要步骤。

1. 发布到PyPI(Python Package Index)

PyPI是Python官方的公共包索引,是大多数开源Python包的首选发布平台。

a. 准备工作:

注册PyPI账户: 如果你还没有,需要在 PyPI官网 注册一个账户。同时,为了安全起见,强烈建议开启双重认证(2FA)。创建API Token: 不要使用你的PyPI用户名和密码直接上传。在PyPI账户设置中生成一个API Token,并为其分配一个适当的权限(例如,仅允许上传到你的特定项目)。将这个Token保存好,因为它只会显示一次。安装

twine

twine

是一个安全的包上传工具,它能确保你的包在上传过程中加密。

  pip install twine

b. 构建你的包:确保你已经按照前面的“解决方案”部分,使用

python -m build

命令构建了你的包。这会在

dist/

目录下生成

.whl

.tar.gz

文件。

c. 上传包:使用

twine

dist/

目录下的所有包文件上传到PyPI。

twine upload dist/*

运行这个命令后,

twine

会提示你输入用户名和密码。

用户名: 输入

__token__

(注意是两个下划线)。密码: 输入你之前生成的API Token。

如果一切顺利,你的包就会被上传到PyPI,并且你可以在PyPI网站上看到它。之后,任何人都可以通过

pip install your-awesome-package

来安装你的包了。

2. 发布到私有仓库

有时,你的包不适合公开,或者你希望在企业内部共享。这时,私有仓库就派上用场了。常见的私有仓库解决方案有:

Artifactory (JFrog Artifactory)Nexus Repository Manager (Sonatype Nexus)GitLab Package RegistryGitHub Packagesdevpi (一个轻量级的PyPI兼容服务器)

a. 配置私有仓库:这通常涉及到在你的私有仓库服务中创建一个新的Python仓库,并获取其URL和认证凭据(API Key, 用户名/密码等)。

b. 构建你的包:与发布到PyPI一样,先用

python -m build

构建你的包。

c. 上传包:你仍然可以使用

twine

来上传,但需要指定私有仓库的URL。

twine upload --repository-url https://your-private-repo.com/pypi/your-repo/ dist/*
twine

会提示你输入私有仓库的用户名和密码。具体认证方式取决于你的私有仓库配置,可能是一个API Key,也可能是普通的用户名和密码。

d. 从私有仓库安装:一旦包上传成功,其他人就可以通过

pip

从你的私有仓库安装它。

pip install --index-url https://your-private-repo.com/pypi/your-repo/ your-awesome-package

如果私有仓库需要认证,你可能还需要在URL中包含凭据,或者通过

~/.pip/pip.conf

(或

pip.ini

)文件进行配置:

# ~/.pip/pip.conf[global]index-url = https://username:password@your-private-repo.com/pypi/your-repo/

或者,为了安全,使用

--extra-index-url

来指定私有仓库,同时保留PyPI作为主索引,这样可以安装私有包,也能安装公共包:

pip install --extra-index-url https://username:password@your-private-repo.com/pypi/your-repo/ your-awesome-package

安全提示:

永远不要将API Token或密码硬编码到脚本或版本控制中。使用环境变量、CI/CD工具的秘密管理功能,或者

~/.pypirc

文件来存储凭据。

~/.pypirc

文件可以配置不同的仓库别名和认证信息,例如:

  [distutils]  index-servers =      pypi      my-private-repo  [pypi]  username = __token__  password = pypi-api-token  [my-private-repo]  repository = https://your-private-repo.com/pypi/your-repo/  username = repo-username  password = repo-password

然后你可以使用

twine upload --repository my-private-repo dist/*

发布包是项目生命周期中非常重要的一环,它将你的代码从本地环境推向更广阔的舞台。理解这些工具和流程,能让你更自信地管理和分发你的Python项目。

以上就是python中如何打包自己的Python项目?的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1371342.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月14日 11:21:57
下一篇 2025年12月14日 11:22:05

相关推荐

  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何解决本地图片在使用 mask JS 库时出现的跨域错误?

    如何跨越localhost使用本地图片? 问题: 在本地使用mask js库时,引入本地图片会报跨域错误。 解决方案: 要解决此问题,需要使用本地服务器启动文件,以http或https协议访问图片,而不是使用file://协议。例如: python -m http.server 8000 然后,可以…

    2025年12月24日
    200
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • CSS元素设置em和transition后,为何载入页面无放大效果?

    css元素设置em和transition后,为何载入无放大效果 很多开发者在设置了em和transition后,却发现元素载入页面时无放大效果。本文将解答这一问题。 原问题:在视频演示中,将元素设置如下,载入页面会有放大效果。然而,在个人尝试中,并未出现该效果。这是由于macos和windows系统…

    2025年12月24日
    200
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 如何用HTML/JS实现Windows 10设置界面鼠标移动探照灯效果?

    Win10设置界面中的鼠标移动探照灯效果实现指南 想要在前端开发中实现类似于Windows 10设置界面的鼠标移动探照灯效果,有两种解决方案:CSS 和 HTML/JS 组合。 CSS 实现 不幸的是,仅使用CSS无法完全实现该效果。 立即学习“前端免费学习笔记(深入)”; HTML/JS 实现 要…

    2025年12月24日
    000
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000
  • 如何用前端实现 Windows 10 设置界面的鼠标移动探照灯效果?

    如何在前端实现 Windows 10 设置界面中的鼠标移动探照灯效果 想要在前端开发中实现 Windows 10 设置界面中类似的鼠标移动探照灯效果,可以通过以下途径: CSS 解决方案 DEMO 1: Windows 10 网格悬停效果:https://codepen.io/tr4553r7/pe…

    2025年12月24日
    000
  • 如何用前端技术实现Windows 10 设置界面鼠标移动时的探照灯效果?

    探索在前端中实现 Windows 10 设置界面鼠标移动时的探照灯效果 在前端开发中,鼠标悬停在元素上时需要呈现类似于 Windows 10 设置界面所展示的探照灯效果,这其中涉及到了元素外围显示光圈效果的技术实现。 CSS 实现 虽然 CSS 无法直接实现探照灯效果,但可以通过以下技巧营造出类似效…

    2025年12月24日
    000
  • 使用CSS mask属性指定图片URL时,为什么浏览器无法加载图片?

    css mask属性未能加载图片的解决方法 使用css mask属性指定图片url时,如示例中所示: mask: url(“https://api.iconify.design/mdi:apple-icloud.svg”) center / contain no-repeat; 但是,在网络面板中却…

    2025年12月24日
    000
  • 如何用CSS Paint API为网页元素添加时尚的斑马线边框?

    为元素添加时尚的斑马线边框 在网页设计中,有时我们需要添加时尚的边框来提升元素的视觉效果。其中,斑马线边框是一种既醒目又别致的设计元素。 实现斜向斑马线边框 要实现斜向斑马线间隔圆环,我们可以使用css paint api。该api提供了强大的功能,可以让我们在元素上绘制复杂的图形。 立即学习“前端…

    2025年12月24日
    000
  • 图片如何不撑高父容器?

    如何让图片不撑高父容器? 当父容器包含不同高度的子元素时,父容器的高度通常会被最高元素撑开。如果你希望父容器的高度由文本内容撑开,避免图片对其产生影响,可以通过以下 css 解决方法: 绝对定位元素: .child-image { position: absolute; top: 0; left: …

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信