原文出处:http://www.worldhello.net/2010/12/08/2178.html

本文略有改动

1.1 安装setuptools

首先要安装setuptools工具。Debian/Ubuntu下可以直接使用apt安装:

$ sudo apt-get install python-setuptools

或者通过pip安装:

$ pip install setuptools

更新setuptools

$ pip install --upgrade setuptools

或者下载setuptoolswhl包来安装。可以在这里查看最新版本下载。下载完毕以后通过sh安装。

$ wget https://files.pythonhosted.org/packages/ec/51/f45cea425fd5cb0b0380f5b0f048ebc1da5b417e48d304838c02d6288a1e/setuptools-41.0.1-py2.py3-none-any.whl
$ pip install setuptools-41.0.1-py2.py3-none-any.whl

现在就可以使用pip命令来安装其他的 egg 或者 whl 包了。

1.2 制作自己的egg包

总是安装别人的 egg/whl 包,是不是也想制作自己的包呢?好,接下来我们就自己制作一个简单的包。 首先建立工程目录egg-demo,初始化一个 setup.py 文件:

$ mkdir egg-demo
$ cd egg-demo
$ touch setup.py
$ ls
setup.py

下面主要就是填充 setup.py。setup.py 其实是 python 工具包distutils的配置文件,setuptools就是基于distutils来做的。 在 setup.py 中通过setup函数来配置打包信息。首先要引入setuptools的函数setupsetuptoolssetup其实就是distutilssetup函数,填写 setup.py 为以下内容:

$ cat setup.py
#!/usr/bin/env python
#-*- coding:utf-8 -*- from setuptools import setup setup()

写到这里,一个空的 egg 配置文件就写好了。我们可以使用下面命令生成 egg 包:

$ python setup.py bdist_egg

或者生成 whl 包:

$ python setup.py bdist_wheel

下面看看究竟生成了什么:

$ ls -F
build/ dist/ setup.py UNKNOWN.egg-info/

可以看到多了三个文件夹。而在 dist 文件夹下,有一个 egg 文件:UNKNOWN-0.0.0-py3.6.egg。 产蛋成功!先看看这个 egg 文件是什么格式的:

$ file dist/UNKNOWN-0.0.0-py3.6.egg
dist/UNKNOWN-0.0.0-py3.6.egg: Zip archive data, at least v2.0 to extract

噢,原来就是一个zip压缩包呀!好,再来看看内部构造:

$ unzip -l dist/UNKNOWN-0.0.0-py3.6.egg
Archive: dist/UNKNOWN-0.0.0-py3.6.egg
Length Date Time Name
--------- ---------- ----- ----
181 2019-07-16 14:43 EGG-INFO/PKG-INFO
132 2019-07-16 14:43 EGG-INFO/SOURCES.txt
1 2019-07-16 14:43 EGG-INFO/dependency_links.txt
1 2019-07-16 14:43 EGG-INFO/top_level.txt
1 2019-07-16 14:43 EGG-INFO/zip-safe
--------- -------
316 5 files

同样的,可以对 whl 文件进行查看:

$ file dist/UNKNOWN-0.0.0-py3-none-any.whl
dist/UNKNOWN-0.0.0-py3-none-any.whl: Zip archive data, at least v2.0 to extract $ unzip -l dist/UNKNOWN-0.0.0-py3-none-any.whl
Archive: dist/UNKNOWN-0.0.0-py3-none-any.whl
Length Date Time Name
--------- ---------- ----- ----
171 2019-07-16 06:44 UNKNOWN-0.0.0.dist-info/METADATA
97 2019-07-16 06:44 UNKNOWN-0.0.0.dist-info/WHEEL
1 2019-07-16 06:44 UNKNOWN-0.0.0.dist-info/top_level.txt
296 2019-07-16 06:44 UNKNOWN-0.0.0.dist-info/RECORD
--------- -------
565 4 files

可以看到,whl 文件和 egg 文件还是有不同的。


只有一个EGG-INFO文件夹,内含五个 egg 信息文件,没了。 这个 egg 名称未知,版本 0.0.0。这是因为我们在setup里什么也没有设置。 显然,这个 egg 什么也不能做。 下面给它加点料。 在setup.py中,setup函数接收一系列属性作为配置参数。

  • name:name 是 egg 包的名称,也是寻找要打包的文件夹的名称,默认是 UNKNOWN。
  • version:版本号,默认 0.0.0
  • packages:这里要用到 setuptools 的另一个函数 find_packages,顾名思义,find_packages 用来将指定目录下的文件打包。
  • zip_safe:默认是 False,这样在每次生成 egg 包时都会检查项目文件的内容,确保无误。

还有一些描述性的属性,如 description,long_description,author,author_email,license,keywords,platform,url 等。 填充setup.py文件如下::

$ cat setup.py
#!/usr/bin/env python
#-*- coding:utf-8 -*- from setuptools import setup, find_packages setup(
name = "demo",
version="0.1.0",
packages = find_packages(),
zip_safe = False, description = "egg test demo.",
long_description = "egg test demo, haha.",
author = "amoblin",
author_email = "amoblin@ossxp.com", license = "GPL",
keywords = ("test", "egg"),
platforms = "Independant",
url = "",
)

egg-demo目录下建立和上述 name 名称相同的目录demodemo目录下写__init__.py文件:

$ mkdir demo
$ cat demo/__init__.py
#!/usr/bin/env python
#-*- coding:utf-8 -*- def test():
print "Hello, I'm amoblin." if __name__ == '__main__':
test()

再次生成 egg 包以后查看 egg 包信息:

$ python setup.py bdist_egg
$ unzip -l dist/demo-0.1.0-py3.6.egg
Archive: dist/demo-0.1.0-py3.6.egg
Length Date Time Name
--------- ---------- ----- ----
227 2019-07-16 14:50 EGG-INFO/PKG-INFO
164 2019-07-16 14:50 EGG-INFO/SOURCES.txt
1 2019-07-16 14:50 EGG-INFO/dependency_links.txt
1 2019-07-16 14:50 EGG-INFO/not-zip-safe
5 2019-07-16 14:50 EGG-INFO/top_level.txt
111 2019-07-16 14:49 demo/__init__.py
--------- -------
509 6 files

可以看到,多了一个文件夹demo,里面有我们写的__init__.py。 奉行敏捷原则,先安装了体验一下再说:

$ sudo python setup.py install
running install
running bdist_egg
running egg_info
writing demo.egg-info/PKG-INFO
writing dependency_links to demo.egg-info/dependency_links.txt
writing top-level names to demo.egg-info/top_level.txt
reading manifest file 'demo.egg-info/SOURCES.txt'
writing manifest file 'demo.egg-info/SOURCES.txt'
installing library code to build/bdist.linux-x86_64/egg
running install_lib
running build_py
creating build/bdist.linux-x86_64/egg
creating build/bdist.linux-x86_64/egg/demo
copying build/lib/demo/__init__.py -> build/bdist.linux-x86_64/egg/demo
byte-compiling build/bdist.linux-x86_64/egg/demo/__init__.py to __init__.cpython-36.pyc
creating build/bdist.linux-x86_64/egg/EGG-INFO
copying demo.egg-info/PKG-INFO -> build/bdist.linux-x86_64/egg/EGG-INFO
copying demo.egg-info/SOURCES.txt -> build/bdist.linux-x86_64/egg/EGG-INFO
copying demo.egg-info/dependency_links.txt -> build/bdist.linux-x86_64/egg/EGG-INFO
copying demo.egg-info/not-zip-safe -> build/bdist.linux-x86_64/egg/EGG-INFO
copying demo.egg-info/top_level.txt -> build/bdist.linux-x86_64/egg/EGG-INFO
creating 'dist/demo-0.1.0-py3.6.egg' and adding 'build/bdist.linux-x86_64/egg' to it
removing 'build/bdist.linux-x86_64/egg' (and everything under it)
Processing demo-0.1.0-py3.6.egg
removing '/usr/local/lib/python3.6/dist-packages/demo-0.1.0-py3.6.egg' (and everything under it)
creating /usr/local/lib/python3.6/dist-packages/demo-0.1.0-py3.6.egg
Extracting demo-0.1.0-py3.6.egg to /usr/local/lib/python3.6/dist-packages
demo 0.1.0 is already the active version in easy-install.pth Installed /usr/local/lib/python3.6/dist-packages/demo-0.1.0-py3.6.egg
Processing dependencies for demo==0.1.0
Finished processing dependencies for demo==0.1.0

在这一步,也可以直接进入到 dist 文件夹中,使用 pip install demo-0.1.0-py3.6.egg 命令来安装。还更加方便,因为在卸载的时候也可以使用 pip remove 命令来卸载

OK!安装完毕!接下来我们就可以直接通过import来使用啦!

$ python -c "from demo import test;test()"
Hello, I'm amoblin.

成功输出!这说明安装正确。我们的一个 egg 包诞生了。 一般情况下,我们的源程序都放在 src 目录下,所以接下来将 demo 文件夹移动到 src 里。但这样也要修改setup.py文件,修改find_packages函数中参数为'src',同时增加package_dir参数:

packages=find_packages('src'),
package_dir = {'':'src'}

这样告诉 setuptools 在 src 目录下找包,而不是原来默认的工程根目录。

1.3 egg 文件卸载

以 python3.6 版本为例,egg 文件一般安装在/usr/local/lib/python3.6/dist-packages/目录下,该目录下还有一个easy-install.pth文件,用于存放安装的 egg 信息:

$ cd /usr/local/lib/python3.6/dist-packages
$ cat easy-install.pth|grep demo
./demo-0.1.0-py3.6.egg
$ ls -F|grep demo
demo-0.1.0-py3.6.egg/

卸载 egg 文件很简单,首先将包含此 egg 的行从 easy-install.pth 中删除,然后删除 egg 文件夹即可。

把 python 程序打包成 egg 或者 whl 安装包的更多相关文章

  1. 如何将 Python 程序打包成 .exe 文件?

    有不少订阅本公众号的朋友都不是玩 Python,甚至都不是计算机相关专业的,当我给他们一个 Python 程序时,他们是完全不知道该怎么运行的. 于是我想是不是可以将我的程序打包成可执行文件,直接运行 ...

  2. Python 程序打包成 exe 可执行文件

    Python 程序打包工具 Python 是一个脚本语言,被解释器解释执行.它的发布方式: .py 文件:对于开源项目或者源码没那么重要的,直接提供源码,需要使用者自行安装 Python 并且安装依赖 ...

  3. python + pyinstaller 实现将python程序打包成exe文件直接运行

    pyinstaller 我们在平常学习使用python的时候经常会自己编写一些小程序来使用,虽然python是跨平台的语言,但是如果我们想要在一个没有python以及很多库环境的电脑上使用我们的小程序 ...

  4. 使用py2exe将python程序打包成exe程序

    近日帮朋友写了个python小程序,从互联网上抓取一些需要的文章到本地.为了运行方便,希望能转换成exe程序在windows下定期执行.从百度上找了些文章,发现py2exe的应用比较多,遂使用之. 1 ...

  5. python程序打包成.exe----pyinstaller工具

    1. 环境 windows 2. 安装 准备文件:PyWin32 or pypiwin32 运行如下安装命令:  pip install pyinstaller==3.0 不要使用3.2版本,编译完成 ...

  6. 将Python 程序打包成 .exe格式入门

    PyInstaller PyInstaller 是一个十分有用的第三方库,可以用来打包 python 应用程序,打包完的程序就可以在没有安装 Python 解释器的机器上运行了. 它能够在 Windo ...

  7. python实战===python程序打包成exe

    推荐PyInstaller项目www.pyinstaller.org   安装方法: 先跑pip install pywin32再跑pip install pyinstaller即可 可用一句命令打包 ...

  8. 将 Python 程序打包成 .exe 文件

    1.简介 做了一个excel的风控模板,里面含有宏,我用python的第三方xlwings部署到linux后发现,linux环境并不支持xlwings. Python 程序都是脚本的方式,一般是在解析 ...

  9. python程序打包成.exe

    安装pyinstaller 方法一:使用pip install pyinstaller 方法二:如果是下载github上的包之后手动安装 包下载 亲测可用:Pyinstaller下载地址,GitHub ...

随机推荐

  1. WPF Calendar 日历控件 样式自定义

    原文:WPF Calendar 日历控件 样式自定义 粗略的在代码上做了些注释 blend 生成出来的模版 有的时候 会生成 跟 vs ui界面不兼容的代码 会导致可视化设计界面 报错崩溃掉 但是确不 ...

  2. 零元学Expression Blend 4 - Chapter 9 用实例了解布局容器系列-「Canvas」

    原文:零元学Expression Blend 4 - Chapter 9 用实例了解布局容器系列-「Canvas」 本系列将教大家以实做案例认识Blend 4 的布局容器,此章介绍的布局容器是Blen ...

  3. TIFF图片简介

    每个TIFF文件都是从指示字节顺序的两个字节开始的.“II”表示小字节在先.“MM”表示大字节在先字节顺序.后面的两个字节表示数字42.数字42是“为了其深刻的哲学意义"而选择的. 42的读 ...

  4. Android零基础入门第79节:Intent 属性详解(上)

    Android应用将会根据Intent来启动指定组件,至于到底启动哪个组件,则取决于Intent的各属性.本期将详细介绍Intent的各属性值,以及 Android如何根据不同属性值来启动相应的组件. ...

  5. 机器学习Machine Learning(ML)

    什么是机器学习 定义 对于某个任务T和表现的衡量P,当计算机程序在该任务T的表现上,经过P的衡量,随着经验E而增长,称计算机能够通过经验E来学习该任务.(Tom Mitchell) 举例而言,在跳棋游 ...

  6. 创建第一个ASP.NET MVC项目

    创建 新建->项目->展开Web->ASP.NET Web应用程序->MVC->确认 ASP.NET MVC应用程序的目录结构 /Controllers该目录保存处理UR ...

  7. Qt实现网络播放器

        写了这么多的博客,关于网络的还不算多,经常有人询问一些关于网络传输.制作在线试听及下载音乐.构造及解析数据等的一些问题,今天就在这里一并讲解.   网络操作:     主要涉及:QNetwor ...

  8. Codility---MaxProductOfThree

    Task description A non-empty zero-indexed array A consisting of N integers is given. Theproduct of t ...

  9. 管理python虚拟环境的工具virtuelenvwrapper

    virtuelenvwrapper出现的原因 virtualenv 的一个最大的缺点就是: ​ 每次开启虚拟环境之前要去虚拟环境所在目录下的 bin 目录下 source 一下 activate,这就 ...

  10. Babel是什么?

    要是官方文档写得好的话,我也许就不用自己做个笔记. 官方文档 Babel 是一个工具链,主要用于将 ECMAScript 2015+ 版本的代码转换为向后兼容的 JavaScript 语法,以便能够运 ...