为什么 'python3 -m venv myenv' 将旧版本的 pip 安装到 myenv 中,而不是我在系统上任何地方都能找到的任何版本的 pip?

Why 'python3 -m venv myenv' installs older version of pip into myenv than any version of pip I can find anywhere on the system?

这并没有给我带来任何无法通过激活虚拟环境和 运行 pip install -U pip 解决的问题,但我一直想知道旧版本的 pip 是从哪里来的。

我正在使用 OS X 10.7.5。当我使用 pyvenv-3.4 myenvpython3 -m venv myenv 创建虚拟环境时,虚拟环境中安装的 pip 版本是 6.0.8,但我已将我的全局 pip 升级到 6.1.1。

这是一个演示我的意思的终端会话:

$ python3 -m venv myenv
$ myenv/bin/pip -V
pip 6.0.8 from /Users/dust/Desktop/myenv/lib/python3.4/site-packages (python 3.4)

这是我希望发生的事情:

$ source myenv/bin/activate
(myenv)$ pip -V
UPDATED SYSTEM VERSION HERE WOULD BE NICE

除了在虚拟环境中创建的以外,我在其他任何地方都找不到 pip 6.0.8。

以下是我用来尝试解决这个问题的各种命令的输出:

$ which pip
/Library/Frameworks/Python.framework/Versions/3.4/bin/pip

$ which pip3
/Library/Frameworks/Python.framework/Versions/3.4/bin/pip3

$ pip -V
pip 6.1.1 from /Library/Frameworks/Python.framework/Versions/3.4/lib/python3.4/site-packages (python 3.4)

$ pip3 -V
pip 6.1.1 from /Library/Frameworks/Python.framework/Versions/3.4/lib/python3.4/site-packages (python 3.4)

我什至尝试使用 find:

$ find / -type f -name pip 2>&1 | awk '! /^f.*$/'
/Library/Frameworks/Python.framework/Versions/3.4/bin/pip
/usr/local/bin/pip

$ find / -type f -name pip3 2>&1 | awk '! /^f.*$/'
/Library/Frameworks/Python.framework/Versions/3.4/bin/pip3

我以为 /usr/local/bin/pip 可能是罪魁祸首,但不是:

$ /usr/local/bin/pip -V
pip 6.1.1 from /Library/Python/2.7/site-packages/pip-6.1.1-py2.7.egg (python 2.7)

嗯。也许 OS X python 有?

$ /usr/bin/python
>>> import pip
>>> pip.__version__
'6.1.1'

6.1.1 无论是 python 的哪个发行版都会报告,无论是 OS X 的 2.7.1,python.org 的 2.7.9,还是 python.org的3.4.3.

是否可以(或建议)更新放入虚拟环境的 pip 版本?

我面临同样的问题,运行 OSX 10.10.2python 3.4.2。最近,我在一台 debian wheezy 机器上用 python 3.4.3 创建了一个虚拟环境,最后得到的 pip 版本比可用版本还旧。不得不升级 pip.

我一直在手动将虚拟环境中的 pip6.0.8 升级到 6.1.1,因为我 o.c.d 以这种方式了解软件库版本 - 并且是的,我现在正在将我的 python 3 版本升级到 3.4.3。反正我的系统python3-pip是最新版本6.1.1,所以我也想知道为什么pyvenv会创建一个新的虚拟环境并加载旧的pip

由于升级 pip,我没有注意到在任何虚拟环境中发生任何不好的事情(但另一方面,我也没有注意到任何好事)显然新的 pip 更快-- 没有注意到,并且在成功安装时输出更少的垃圾,因为用户不关心 -- 也没有注意到,可能是因为我是那些不关心的人之一,并且还带有状态- 能够启动拿铁艺术的艺术咖啡机!!! -- 仍在等待 sudo pip install latte 完成 :(

所以,为了回答你的问题,升级绝对是可能的,而且可能是可取的,因为显然新的 pip 修复了一些错误并且运行速度更快,但我想加速不是那么重要,并且错误修复不会影响那么多人(我在使用旧 pip 时从未遇到过错误)。

当你创建一个新的虚拟环境时,你 可以 link 使用标志 --system-site-packages 到系统站点包,就像这样

pyvenv myenv --system-site-packages

这将 link 到您系统范围的 pip 版本,并且会消除在每个虚拟环境中手动升级 pip 的烦恼,但是如果您这样做,那么您的虚拟环境是虚拟的吗?

更新:在我上面的咆哮之后,我进入了 venv 包的源代码进行挖掘。 pip 是通过文件 __init__.py 第 248 行

中名为 _setup_pip 的方法设置的
def _setup_pip(self, context):
        """Installs or upgrades pip in a virtual environment"""
        # We run ensurepip in isolated mode to avoid side effects from
        # environment vars, the current directory and anything else
        # intended for the global Python environment
        cmd = [context.env_exe, '-Im', 'ensurepip', '--upgrade',
                                                    '--default-pip']
        subprocess.check_output(cmd, stderr=subprocess.STDOUT)

所以,venv 似乎是使用 subprocess 模块从 shell 调用 ensurepip

google-fu 的一分钟给了我来自 documentation for ensurepip 的这个。

ensurepip.version()

Returns a string specifying the bundled version of pip that will be installed when bootstrapping an environment.

因此,从命令行输入以下代码:

$ python3 -c 'import ensurepip; print(ensurepip.version())' 
6.0.8

显示我当前的 pip 将由 ensurepip 引导。

我想在 ensurepip 升级之前,每次新安装我们都只能使用旧版本的 pip,因为我找不到升级随附的 pip 版本的方法ensurepip

较新

如果你想"hotpatch"你安装的python,只需修改ensurepip/__init__.py中列出的版本并替换ensurepip/_bundled中的两个文件。您可以从安装 python 的目录中通过 运行 find * | grep ensurepip 找到此位置。在带有 Homebrew 的 macOS 上,这是位置:/usr/local/Cellar/python/3.7.1/Frameworks/Python.framework/Versions/3.7/lib/python3.7/ensurepip

您还需要删除包含 .pyc 个文件的 ensurepip/__pycache__ 目录。

我较早的构建时修复:

在从源代码构建之前,您可以通过修补 Python 来更新 pip 和 setuptools 的捆绑版本。以下补丁会将 pip 和 setuptools 的捆绑版本更新为当前可用的当前版本。您需要使用以下选项调用 configure--with-ensurepip=upgrade

这些 whl 文件是从 PYPI 下载的:

https://pypi.org/project/pip/#files

https://pypi.org/project/setuptools/#files

diff -ru Python-3.7.1/Lib/ensurepip/__init__.py Python-3.7.1.new/Lib/ensurepip/__init__.py
--- Python-3.7.1/Lib/ensurepip/__init__.py  2018-10-20 06:04:19.000000000 +0000
+++ Python-3.7.1.new/Lib/ensurepip/__init__.py  2018-11-27 02:36:19.301655008 +0000
@@ -8,9 +8,9 @@
 __all__ = ["version", "bootstrap"]


-_SETUPTOOLS_VERSION = "39.0.1"
+_SETUPTOOLS_VERSION = "40.6.2"

-_PIP_VERSION = "10.0.1"
+_PIP_VERSION = "18.1"

 _PROJECTS = [
     ("setuptools", _SETUPTOOLS_VERSION),
Only in Python-3.7.1/Lib/ensurepip/_bundled: pip-10.0.1-py2.py3-none-any.whl
Only in Python-3.7.1.new/Lib/ensurepip/_bundled: pip-18.1-py2.py3-none-any.whl
Only in Python-3.7.1/Lib/ensurepip/_bundled: setuptools-39.0.1-py2.py3-none-any.whl
Only in Python-3.7.1.new/Lib/ensurepip/_bundled: setuptools-40.6.2-py2.py3-none-any.whl

如果幸运的话,您可以使用 Python 补丁来提升它。

你的Python过时了吗?这是使用 macports,但 brewapt 中的任何一个都可以。

port outdated

python36                       3.6.9 < 3.6.10

让我们解决这个问题:

sudo port upgrade python36

现在...鼓声,在我新创建的 venv 中,Python 补丁似乎带来了最新的 pip

pip --version 显示 20.0.2,它之前在 18,这正是它在venv.

这是在花费了非常长的时间试图找出如何更新我的用户级别 Python 3.6 版本的 pip 之后,在 mac 上,当然,如果你退出你在 Python 2.7 和 Python 3.6 的 virtualenv,通过 macports 安装,似乎没有 pip 命令可以使用,直到你建立一个 venv.

通过查看 /opt/local/Library/Frameworks/Python.framework/Versions/3.6,macports 放东西的地方,我怀疑 Utknonos 正在做某事,所以+1.

确实应该有一些更明显的东西,比如 pip-system-selfupdate 命令或 python36 附带的 pip36。特别是考虑到 pip upgrade 的搜索将 return 大量点击 this 主题。

注:我更新后,虽然Python已经更新了

(venv)$ pwd

/opt/local/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6

(venv)$ grep PIP_VERSION ensurepip/__init__.py
_PIP_VERSION = "18.1"

(venv) $python --version
Python 3.6.10

(venv) $pip --version
pip 20.0.2 from .../venv/lib/python3.6/site-packages/pip (python 3.6)