From 80bf2a4330266d56e039f2f5c92bc1b0725801b4 Mon Sep 17 00:00:00 2001 From: zhaozhiming Date: Sun, 21 Jul 2024 18:17:50 +0800 Subject: [PATCH] Site updated: 2024-07-21 18:17:49 --- .../index.html | 307 ++++++++++++++++++ archives/2024/07/index.html | 14 + archives/2024/index.html | 28 +- archives/2024/page/2/index.html | 30 +- archives/2024/page/3/index.html | 194 +++++++++++ archives/index.html | 28 +- archives/page/10/index.html | 28 +- archives/page/11/index.html | 28 +- archives/page/12/index.html | 28 +- archives/page/13/index.html | 28 +- archives/page/14/index.html | 28 +- archives/page/15/index.html | 28 +- archives/page/16/index.html | 28 +- archives/page/17/index.html | 28 +- archives/page/18/index.html | 28 +- archives/page/19/index.html | 28 +- archives/page/2/index.html | 28 +- archives/page/20/index.html | 28 +- archives/page/21/index.html | 30 +- archives/page/22/index.html | 194 +++++++++++ archives/page/3/index.html | 33 +- archives/page/4/index.html | 28 +- archives/page/5/index.html | 28 +- archives/page/6/index.html | 28 +- archives/page/7/index.html | 28 +- archives/page/8/index.html | 28 +- archives/page/9/index.html | 28 +- atom.xml | 58 ++-- categories/ai/index.html | 28 +- categories/ai/page/2/index.html | 28 +- categories/ai/page/3/index.html | 28 +- categories/ai/page/4/index.html | 28 +- categories/ai/page/5/index.html | 14 + .../2024/07/next-generation-python-tools.jpg | Bin 0 -> 58050 bytes images/post/2024/07/ruff-compare.png | Bin 0 -> 46084 bytes images/post/2024/07/uv-compare.png | Bin 0 -> 45604 bytes index.html | 64 ++-- page/10/index.html | 62 ++-- page/11/index.html | 62 ++-- page/12/index.html | 62 ++-- page/13/index.html | 62 ++-- page/14/index.html | 63 ++-- page/15/index.html | 62 ++-- page/16/index.html | 62 ++-- page/17/index.html | 62 ++-- page/18/index.html | 60 ++-- page/19/index.html | 62 ++-- page/2/index.html | 64 ++-- page/20/index.html | 62 ++-- page/21/index.html | 62 ++-- page/22/index.html | 255 +++++++++++++++ page/3/index.html | 64 ++-- page/4/index.html | 64 ++-- page/5/index.html | 64 ++-- page/6/index.html | 65 ++-- page/7/index.html | 66 ++-- page/8/index.html | 65 ++-- page/9/index.html | 63 ++-- tags/miniconda/index.html | 187 +++++++++++ tags/poetry/index.html | 187 +++++++++++ tags/python/index.html | 19 ++ tags/ruff/index.html | 187 +++++++++++ tags/uv/index.html | 187 +++++++++++ 63 files changed, 2825 insertions(+), 1065 deletions(-) create mode 100644 2024/07/15/modern-python-project-tools-config/index.html create mode 100644 archives/2024/page/3/index.html create mode 100644 archives/page/22/index.html create mode 100644 images/post/2024/07/next-generation-python-tools.jpg create mode 100644 images/post/2024/07/ruff-compare.png create mode 100644 images/post/2024/07/uv-compare.png create mode 100644 page/22/index.html create mode 100644 tags/miniconda/index.html create mode 100644 tags/poetry/index.html create mode 100644 tags/ruff/index.html create mode 100644 tags/uv/index.html diff --git a/2024/07/15/modern-python-project-tools-config/index.html b/2024/07/15/modern-python-project-tools-config/index.html new file mode 100644 index 000000000..aa13eded9 --- /dev/null +++ b/2024/07/15/modern-python-project-tools-config/index.html @@ -0,0 +1,307 @@ + + + + + + 都 2024 年了,你还在用 pip 吗? - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+
+

都 2024 年了,你还在用 pip 吗?

+
+
Published on: +
+ +
+
+ +

编程语言 Python 随着 AI 的发展越来越受开发人员的喜爱,目前已经是最流行的编程语言,但由于 Python 是一门相对较老的语言,并且经过 Python2 到 Python3 这一漫长而复杂的迁移历程,使得一些 Python 开发人员可能还在使用一些过时的工具和库。今天我们就来带大家了解当前 Python 生态系统中最流行最实用的开发工具,让你彻底告别那些老古董

+ + +

Python 环境管理工具

Python 项目中虚拟环境的管理非常重要,虚拟环境可以帮助我们在不同项目中使用不同的 Python 版本和依赖库,避免不同项目之间的依赖冲突。Python 最早使用的是 virtualenv 工具来管理虚拟环境,到了 Python3.3 后自带了内置的创建虚拟环境模块 venv,但它们都存在一个问题,就是只能使用同一个 Python 版本创建虚拟环境。假如你在 A 项目想使用 Python3.10,而在 B 项目想使用 Python3.9,那么你就需要安装两个不同版本的 Python,然后分别使用 virtualenvvenv 来创建虚拟环境。

+

在使用了一些 Python 环境管理工具后,我比较推荐 Miniconda 这个工具,它是一个轻量级的 Conda 版本,可以帮助我们管理 Python 环境和依赖库,但我一般不使用它来管理依赖库,主要使用它来管理 Python 环境。

+

虽然 Miniconda 与其他 Python 环境管理工具(比如 pyenv)相比会重量级一些(包含了一些数据科学相关的库),而且功能也不是很纯粹(即有环境管理也有依赖管理等功能), 但是它的优势在于环境管理的功能非常出色,可以轻松地在不同操作系统上进行安装,同时适配 bash、fish、zsh 等不同的 shell 环境,而且跟 virtualenvvenv 一起使用也不会引起冲突。

+

Miniconda 使用示例

以下是 Miniconda 的一些常用命令:

+
    +
  • 创建 Python 环境: conda create -n myenv python=3.10
  • +
  • 切换 Python 环境: conda activate myenv
  • +
  • 展示所有 Python 环境: conda env list
  • +
  • 初始化 shell 环境: conda init <bash/fish/zsh>
  • +
+

Python 依赖管理工具

Python 项目中最常使用的工具应该要属 Pip 了,Pip 是 Python 的依赖管理工具,用于安装和管理 Python 依赖,虽然 Pip 功能强大,但在管理项目依赖时存在一些问题,比如新增依赖时需要手动修改 requirements.txt 文件,而且没有版本锁定功能,导致在不同环境中安装的依赖版本可能不一致。

+

其他编程语言比如 JS 则能有效地处理这种情况,使用它的依赖管理工具 npm 会产生一个 package.json 文件来管理项目依赖,然后生成一个 package-lock.json 文件来锁定依赖版本,确保在不同环境中安装相同的依赖版本。

+

为了解决这一问题,在 Python 中出现了不少依赖管理工具,Poetry 是其中一个比较流行的工具。Poetry 使用一个 pyproject.toml 文件来管理项目的所有依赖项和元数据,使项目配置更加简洁明了,它会自动处理依赖项的版本冲突,并且能够生成锁文件 poetry.lock,确保在不同环境中安装相同的依赖版本。

+

Poetry 安装

Poetry 有多种安装方式,最简单的是通过 Python 脚本进行安装,安装命令如下:

+
1
curl -sSL https://install.python-poetry.org | python3 -
+ +

安装完成后,可以通过 Miniconda 创建一个特定版本的 Python 环境,然后在这个环境中使用 Poetry,具体操作如下:

+
1
2
3
4
5
6
7
8
# 使用 conda 创建一个 Python 环境
conda create -n myenv python=3.10
# 切换到这个 Python 环境
conda activate myenv
# 构建一个 Poetry 虚拟环境
poetry env use python
# 进入该环境
poetry shell
+ +

Poetry 使用示例

在新项目中使用 Poetry 可以使用 poetry new <project_name> 来创建一个新的 Python 项目,如果是已有项目,可以在项目目录中使用 poetry init 命令来初始化项目,这两个命令都会生成一个 pyproject.toml 文件,用于管理项目的依赖项和元数据。

+

你可以使用 poetry install 来安装项目的所有依赖,如果是初次运行该命令,Poetry 会生成一个 poetry.lock 文件,用于锁定项目的依赖版本,确保在不同环境中安装相同的依赖版本。也可以使用 poetry add <package_name> 命令来单独添加一个依赖,这样 Poetry 会自动更新 pyproject.tomlpoetry.lock 文件。

+

我们通过一个简单的例子来看下 Poetry 对依赖管理的方法,假设我们在项目中安装了一个新的依赖 requests,那么可以使用如下命令:

+
1
poetry add requests
+ +

安装完成后,在 pyproject.toml 文件中只会添加 requests 这个依赖的信息:

+
1
2
[tool.poetry.dependencies]
requests = "^2.32.3"
+ +

但是在 poetry.lock 文件会中除了添加 requests 这个依赖外,还会添加 requests 这个依赖所需的其他依赖库的信息,可以使用以下命令查看依赖之间的关系:

+
1
2
3
4
5
6
$ poetry show --tree
requests 2.32.3 Python HTTP for Humans.
├── certifi >=2017.4.17
├── charset-normalizer >=2,<4
├── idna >=2.5,<4
└── urllib3 >=1.21.1,<3
+ +

可以看到 requests 是项目根目录下的依赖,而其他几个依赖是 requests 所需的依赖。

+

如果你用 Pip 来安装依赖,那么 pip install requests 后再用 pip freeze > requirements.txt 命令生成的 requirements.txt 文件会包含所有依赖的信息,使得你分不清哪些是项目的依赖,哪些是衍生的依赖。

+

Poetry 还允许你将 lock 文件导出成 requirements.txt 文件,这样你就可以使用 pip install -r requirements.txt 来安装项目的依赖,具体命令如下:

+
1
poetry export --without-hashes --format=requirements.txt --output requirements.txt
+ +

更多的 Poetry 使用方法可以参考官方文档

+

其他依赖管理工具

除了 Poetry 外,还有一些其他的 Python 依赖管理工具,比如 Pdm 就是不错的选择,它整体和 Poetry 类似,它还包含了 Python 环境管理的功能,这样就不需要和 Miniconda 配合也可直接使用,但是在流行度上比 Poetry 稍微逊色一些(截止时间 2024 年 7 月,Poetry 的 Github Star 数是 30K,而 Pdm 的 Github Star 数是 7.6K),可能是 Pdm 比 Poetry 发布时间晚的关系(Poetry 是 2018 年发布的,而 Pdm 是 2020 年发布的),后面如果 Pdm 发展的好的话,说不定会超过 Poetry。

+

另外一个值得推荐的 Python 依赖管理工具是 Uv,它是一款用 Rust 编写的极速 Python 包安装和解析工具,旨在作为 Pip 和 Pip-tools 的替代品,并逐步发展成为一个全面的 Python 项目和包管理器,下面是它和其他工具安装依赖的速度对比图:

+ + +

得益于 Rust 的高性能,在速度上 Uv 完全碾压了其他工具,但可惜的是 Uv 目前还是使用 requirements.txt 来管理依赖,这样就无法保证可以在不同环境安装相同的依赖,而且 Uv 也不能和 Poetry 一起使用,因为 Uv 是按照兼容 Pip 的思路进行开发,而 Poetry 内部已经不用 Pip 做依赖管理了。

+

尽管 Uv 无法和 Poetry 一起使用,但是我们可以在 CI/CD 中使用 Uv 来加速依赖安装,比如我们先用 Poetry 导出 requirements.txt 文件,然后在 CI/CD 中使用 Uv 来安装依赖,这样就可以大大缩短依赖安装的时间。

+

Pip 是否过时

虽然这些依赖管理工具很强大,但一些小型项目可能更多开发人员还是会选择 Pip,因为 Pip 无需额外安装其他工具,一般有 Python 环境就可以直接使用。另外一点是编程语言的原生工具也在不断发展,以 JS 为例,npm 刚开始时也不支持 lock 文件,但后面参考了一些 JS 的流行工具,在社区的共同努力下慢慢完善了这个功能,所以 Pip 也有可能在未来的某个版本中加入类似 Poetry 的功能。

+

所以在 Pip 没有发展完善之前,我们可以使用 Poetry 这样的工具来解决依赖管理的问题,同时也能简化项目配置,提高开发效率。

+

代码规范工具

另外一种常用的工具是代码规范类工具,因为 Python 的语法比较灵活,所以在团队协作中可能会出现代码风格不一致的问题,为了解决这个问题,每种编程语言都会有一些代码规范工具,这类工具包括代码格式化工具、代码检查工具等。

+

在 Python 中代码格式化工具有 BlackYAPFautopep8 等,而代码检查工具有 Flake8Pylintmypy 等,这些工具都可以通过 Pip 来安装,然后在项目中使用。

+

而最近比较流行的一个代码规范工具 Ruff,它同时集成了代码格式化和代码检查功能,可以帮助我们更好地在项目中统一代码风格。Ruff 是 Uv 开发团队开发的另外一款工具,同样是使用 Rust 语言进行编写,从而使它的性能远远高于其他同类型的工具,下面是 Ruff 和其他工具的性能对比图:

+ + +

Ruff 内部集成了 Black 和 Flake8 等工具,下面我们就来介绍下 Ruff 如何安装及使用。

+

Ruff 安装

Ruff 可以通过 Pip 或 Poetry 进行安装,在项目中使用的话,推荐使用 Poetry 安装到 dev 开发依赖中,表示这个工具只在开发环境中使用,具体命令如下:

+
1
poetry add --dev ruff
+ +

如果想让 Ruff 在本地 IDE 中使用的话,建议是进行全局安装,这样就可以在任何项目中使用 Ruff,具体命令如下:

+
1
curl -LsSf https://astral.sh/ruff/install.sh | sh
+ +

安装完成后,可以通过 IDE 的插件来使用 Ruff,比如在 VSCode 中安装 Ruff 插件,然后在 VSCode 中使用快捷键 Ctrl+Shift+P 打开命令面板,输入 user config 打开用户配置文件,然后添加 Ruff 安装后的路径:

+
1
2
3
{
"ruff.path": ["/your/ruff/path"]
}
+ +

Ruff 使用示例

Ruff 主要有 2 个命令,分别是 ruff checkruff format,前者用于代码检查,后者用于代码格式化。

+

为了保证团队的代码风格一致,我们可以在 Git 的 hook 中添加 Ruff 检查命令,这样每个人在执行 git commit 命令时就会自动执行 Ruff 命令,如果检查失败则无法提交代码。

+

首先我们需要安装 pre-commit 工具,这个工具可以让我们在 Git hook 上轻松配置命令,同样将其安装到 dev 开发依赖即可:poetry add --dev pre-commit,然后在项目根目录下添加 .pre-commit-config.yaml 文件,内容如下:

+
1
2
3
4
5
6
7
8
9
10
11
repos:
- repo: https://github.com/astral-sh/ruff-pre-commit
# Ruff 版本
rev: v0.5.1
hooks:
# 执行 ruff check
- id: ruff
# 可选参数,自动修复代码
args: [--fix]
# 执行 ruff format
- id: ruff-format
+ +
    +
  • 在文件中我们添加 Ruff 工具来处理代码提交时的检查
  • +
  • 我们添加了 2 个工具,第一个工具 id 是 ruff 表示执行 ruff check 命令,第二个工具 id 是 ruff-format 表示执行 ruff format 命令
  • +
  • 第一个工具还有一个可选参数 args: [--fix],表示会自动修复检查出有误的代码,但也不是所有代码都能自动修复,有些代码还是需要手动修复的
  • +
+

最后我们在项目中执行 pre-commit install 命令,将文件内容添加到 Git hook 中:

+
1
2
$ pre-commit install
pre-commit installed at .git/hooks/pre-commit
+ +

配置完成后,我们故意写一些代码错误,然后提交代码, 检查结果如下:

+
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
$ git commit -m 'add some feature'
ruff.....................................................................Failed
- hook id: ruff
- exit code: 1

main.py:1:8: F401 [*] `requests` imported but unused
|
1 | import requests
| ^^^^^^^^ F401
|
= help: Remove unused import: `requests`

Found 1 error.
[*] 1 fixable with the `--fix` option.

ruff-format..............................................................Passed
+ +

可以看到代码检查任务失败了,报了 imported but unused 的错误,错误编号 F401,第二个任务代码格式化检查通过。

+

如果想在代码中不检查某行代码,可以在代码行后面加上 # noqa: {error_code},比如:

+
1
import requests # noqa: F401
+ +

这样就可以在 Ruff 检查时忽略这个错误。更多的 Ruff 使用方法可以参考官方文档

+

总结

今天我们介绍了 Python 中新一代的项目工具,新工具带来的好处是开发效率的提升,因为每个新工具都是在解决旧工具的不足之处,是在旧工具的基础上进行了优化和改进。作为一个热衷提高生产力的开发人员,可以在适当时机尝试使用这些新工具,如果觉得新工具不合适,也可以退回去重新使用旧工具,关键在于尝试。

+

关注我,一起学习各种人工智能和 GenAI 新技术,欢迎交流,如果你有什么想问想说的,欢迎在评论区留言。

+
+ +
+
+
+

赞赏

+ +
+
+ + + + + + + + + +
+

Comments

+
+
+
+ +
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/archives/2024/07/index.html b/archives/2024/07/index.html index cf240f623..e0a6b040a 100644 --- a/archives/2024/07/index.html +++ b/archives/2024/07/index.html @@ -129,6 +129,20 @@

2024

+ +

高级 RAG 检索策略之知识图谱

diff --git a/archives/2024/index.html b/archives/2024/index.html index a06839480..5c6a40b7c 100644 --- a/archives/2024/index.html +++ b/archives/2024/index.html @@ -129,6 +129,20 @@

2024

+ + - -
diff --git a/archives/2024/page/2/index.html b/archives/2024/page/2/index.html index e05602865..65d95a7db 100644 --- a/archives/2024/page/2/index.html +++ b/archives/2024/page/2/index.html @@ -129,6 +129,20 @@

2024

+ + - -
@@ -279,6 +279,8 @@

+ +

diff --git a/archives/2024/page/3/index.html b/archives/2024/page/3/index.html new file mode 100644 index 000000000..776c7869a --- /dev/null +++ b/archives/2024/page/3/index.html @@ -0,0 +1,194 @@ + + + + + + Archives: 2024 - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+ + + + + + + + + +

2024

+ + + + +
+ + + + + +
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/archives/index.html b/archives/index.html index ab3c16399..978cfb370 100644 --- a/archives/index.html +++ b/archives/index.html @@ -129,6 +129,20 @@

2024

+ + - -
diff --git a/archives/page/10/index.html b/archives/page/10/index.html index d483a7502..8890a6f08 100644 --- a/archives/page/10/index.html +++ b/archives/page/10/index.html @@ -129,6 +129,20 @@

2017

+ + - -
diff --git a/archives/page/11/index.html b/archives/page/11/index.html index 7615d697c..1d5a34bc2 100644 --- a/archives/page/11/index.html +++ b/archives/page/11/index.html @@ -129,6 +129,20 @@

2016

+ + - -
diff --git a/archives/page/12/index.html b/archives/page/12/index.html index c2a4b7d8c..bfa4e50ee 100644 --- a/archives/page/12/index.html +++ b/archives/page/12/index.html @@ -129,6 +129,20 @@

2015

+ + - -
diff --git a/archives/page/13/index.html b/archives/page/13/index.html index c22f2d064..29a87137f 100644 --- a/archives/page/13/index.html +++ b/archives/page/13/index.html @@ -129,6 +129,20 @@

2015

+ + - -
diff --git a/archives/page/14/index.html b/archives/page/14/index.html index 90c0c7655..425221323 100644 --- a/archives/page/14/index.html +++ b/archives/page/14/index.html @@ -129,6 +129,20 @@

2014

+ + - -
diff --git a/archives/page/15/index.html b/archives/page/15/index.html index ddd5f420f..6ace05cef 100644 --- a/archives/page/15/index.html +++ b/archives/page/15/index.html @@ -129,6 +129,20 @@

2014

+ + - -
diff --git a/archives/page/16/index.html b/archives/page/16/index.html index 5a63d8191..cdb713e18 100644 --- a/archives/page/16/index.html +++ b/archives/page/16/index.html @@ -129,6 +129,20 @@

2014

+ + - -
diff --git a/archives/page/17/index.html b/archives/page/17/index.html index 14521a311..ec422012f 100644 --- a/archives/page/17/index.html +++ b/archives/page/17/index.html @@ -129,6 +129,20 @@

2014

+ + - -
diff --git a/archives/page/18/index.html b/archives/page/18/index.html index f6fb3cdd9..e7845c701 100644 --- a/archives/page/18/index.html +++ b/archives/page/18/index.html @@ -129,6 +129,20 @@

2013

+ + - -
diff --git a/archives/page/19/index.html b/archives/page/19/index.html index a298c5c9e..5748d1d0d 100644 --- a/archives/page/19/index.html +++ b/archives/page/19/index.html @@ -129,6 +129,20 @@

2012

+ + - -
diff --git a/archives/page/2/index.html b/archives/page/2/index.html index 27fb19f56..58d783359 100644 --- a/archives/page/2/index.html +++ b/archives/page/2/index.html @@ -129,6 +129,20 @@

2024

+ + - -
diff --git a/archives/page/20/index.html b/archives/page/20/index.html index 9af431c1a..e9a5943d6 100644 --- a/archives/page/20/index.html +++ b/archives/page/20/index.html @@ -129,6 +129,20 @@

2012

+ + - -
diff --git a/archives/page/21/index.html b/archives/page/21/index.html index bebeda671..a7a8d2dda 100644 --- a/archives/page/21/index.html +++ b/archives/page/21/index.html @@ -129,6 +129,20 @@

2012

+ + - -
@@ -279,6 +279,8 @@

+ + diff --git a/archives/page/22/index.html b/archives/page/22/index.html new file mode 100644 index 000000000..172516f29 --- /dev/null +++ b/archives/page/22/index.html @@ -0,0 +1,194 @@ + + + + + + Archives - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+ + + + + + + + + +

2012

+ + + + +
+ + + + + +
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/archives/page/3/index.html b/archives/page/3/index.html index 43f130d81..1c7e2d589 100644 --- a/archives/page/3/index.html +++ b/archives/page/3/index.html @@ -127,6 +127,25 @@

+

2024

+ + + +
+ +

2023

- -
diff --git a/archives/page/4/index.html b/archives/page/4/index.html index baa963c47..b2fba0fd0 100644 --- a/archives/page/4/index.html +++ b/archives/page/4/index.html @@ -129,6 +129,20 @@

2023

+ + - -
diff --git a/archives/page/5/index.html b/archives/page/5/index.html index 43eafa1af..97a1aea52 100644 --- a/archives/page/5/index.html +++ b/archives/page/5/index.html @@ -129,6 +129,20 @@

2023

+ + - -
diff --git a/archives/page/6/index.html b/archives/page/6/index.html index 5b30e8c05..54cbee9ef 100644 --- a/archives/page/6/index.html +++ b/archives/page/6/index.html @@ -129,6 +129,20 @@

2023

+ + - -
diff --git a/archives/page/7/index.html b/archives/page/7/index.html index bd3cfaa5a..481e91fb6 100644 --- a/archives/page/7/index.html +++ b/archives/page/7/index.html @@ -129,6 +129,20 @@

2019

+ + - -
diff --git a/archives/page/8/index.html b/archives/page/8/index.html index ccfa5a8d3..a3fc5698d 100644 --- a/archives/page/8/index.html +++ b/archives/page/8/index.html @@ -129,6 +129,20 @@

2018

+ + - -
diff --git a/archives/page/9/index.html b/archives/page/9/index.html index 9ade06dce..e43295ced 100644 --- a/archives/page/9/index.html +++ b/archives/page/9/index.html @@ -129,6 +129,20 @@

2017

+ + - -
diff --git a/atom.xml b/atom.xml index 51ebeaf6b..03d5e783a 100644 --- a/atom.xml +++ b/atom.xml @@ -6,7 +6,7 @@ - 2024-07-12T01:18:49.490Z + 2024-07-19T03:26:19.579Z https://zhaozhiming.github.io/ @@ -16,6 +16,35 @@ Hexo + + 都 2024 年了,你还在用 pip 吗? + + https://zhaozhiming.github.io/2024/07/15/modern-python-project-tools-config/ + 2024-07-15T02:49:25.000Z + 2024-07-19T03:26:19.579Z + +

编程语言 Python 随着 AI 的发展越来越受开发人员的喜爱,目前已经是最流行的编程语言,但由于 Python 是一门相对较老的语言,并且经过 Python2 到 Python3 这一漫长而复杂的迁移历程,使得一些 Python 开发人员可能还在使用一些过时的工具和库。今天我们就来带大家了解当前 Python 生态系统中最流行最实用的开发工具,让你彻底告别那些老古董

Python 环境管理工具

Python 项目中虚拟环境的管理非常重要,虚拟环境可以帮助我们在不同项目中使用不同的 Python 版本和依赖库,避免不同项目之间的依赖冲突。Python 最早使用的是 virtualenv 工具来管理虚拟环境,到了 Python3.3 后自带了内置的创建虚拟环境模块 venv,但它们都存在一个问题,就是只能使用同一个 Python 版本创建虚拟环境。假如你在 A 项目想使用 Python3.10,而在 B 项目想使用 Python3.9,那么你就需要安装两个不同版本的 Python,然后分别使用 virtualenvvenv 来创建虚拟环境。

在使用了一些 Python 环境管理工具后,我比较推荐 Miniconda 这个工具,它是一个轻量级的 Conda 版本,可以帮助我们管理 Python 环境和依赖库,但我一般不使用它来管理依赖库,主要使用它来管理 Python 环境。

虽然 Miniconda 与其他 Python 环境管理工具(比如 pyenv)相比会重量级一些(包含了一些数据科学相关的库),而且功能也不是很纯粹(即有环境管理也有依赖管理等功能), 但是它的优势在于环境管理的功能非常出色,可以轻松地在不同操作系统上进行安装,同时适配 bash、fish、zsh 等不同的 shell 环境,而且跟 virtualenvvenv 一起使用也不会引起冲突。

Miniconda 使用示例

以下是 Miniconda 的一些常用命令:

  • 创建 Python 环境: conda create -n myenv python=3.10
  • 切换 Python 环境: conda activate myenv
  • 展示所有 Python 环境: conda env list
  • 初始化 shell 环境: conda init <bash/fish/zsh>

Python 依赖管理工具

Python 项目中最常使用的工具应该要属 Pip 了,Pip 是 Python 的依赖管理工具,用于安装和管理 Python 依赖,虽然 Pip 功能强大,但在管理项目依赖时存在一些问题,比如新增依赖时需要手动修改 requirements.txt 文件,而且没有版本锁定功能,导致在不同环境中安装的依赖版本可能不一致。

其他编程语言比如 JS 则能有效地处理这种情况,使用它的依赖管理工具 npm 会产生一个 package.json 文件来管理项目依赖,然后生成一个 package-lock.json 文件来锁定依赖版本,确保在不同环境中安装相同的依赖版本。

为了解决这一问题,在 Python 中出现了不少依赖管理工具,Poetry 是其中一个比较流行的工具。Poetry 使用一个 pyproject.toml 文件来管理项目的所有依赖项和元数据,使项目配置更加简洁明了,它会自动处理依赖项的版本冲突,并且能够生成锁文件 poetry.lock,确保在不同环境中安装相同的依赖版本。

Poetry 安装

Poetry 有多种安装方式,最简单的是通过 Python 脚本进行安装,安装命令如下:

1
curl -sSL https://install.python-poetry.org | python3 -

安装完成后,可以通过 Miniconda 创建一个特定版本的 Python 环境,然后在这个环境中使用 Poetry,具体操作如下:

1
2
3
4
5
6
7
8
# 使用 conda 创建一个 Python 环境
conda create -n myenv python=3.10
# 切换到这个 Python 环境
conda activate myenv
# 构建一个 Poetry 虚拟环境
poetry env use python
# 进入该环境
poetry shell

Poetry 使用示例

在新项目中使用 Poetry 可以使用 poetry new <project_name> 来创建一个新的 Python 项目,如果是已有项目,可以在项目目录中使用 poetry init 命令来初始化项目,这两个命令都会生成一个 pyproject.toml 文件,用于管理项目的依赖项和元数据。

你可以使用 poetry install 来安装项目的所有依赖,如果是初次运行该命令,Poetry 会生成一个 poetry.lock 文件,用于锁定项目的依赖版本,确保在不同环境中安装相同的依赖版本。也可以使用 poetry add <package_name> 命令来单独添加一个依赖,这样 Poetry 会自动更新 pyproject.tomlpoetry.lock 文件。

我们通过一个简单的例子来看下 Poetry 对依赖管理的方法,假设我们在项目中安装了一个新的依赖 requests,那么可以使用如下命令:

1
poetry add requests

安装完成后,在 pyproject.toml 文件中只会添加 requests 这个依赖的信息:

1
2
[tool.poetry.dependencies]
requests = "^2.32.3"

但是在 poetry.lock 文件会中除了添加 requests 这个依赖外,还会添加 requests 这个依赖所需的其他依赖库的信息,可以使用以下命令查看依赖之间的关系:

1
2
3
4
5
6
$ poetry show --tree
requests 2.32.3 Python HTTP for Humans.
├── certifi >=2017.4.17
├── charset-normalizer >=2,<4
├── idna >=2.5,<4
└── urllib3 >=1.21.1,<3

可以看到 requests 是项目根目录下的依赖,而其他几个依赖是 requests 所需的依赖。

如果你用 Pip 来安装依赖,那么 pip install requests 后再用 pip freeze > requirements.txt 命令生成的 requirements.txt 文件会包含所有依赖的信息,使得你分不清哪些是项目的依赖,哪些是衍生的依赖。

Poetry 还允许你将 lock 文件导出成 requirements.txt 文件,这样你就可以使用 pip install -r requirements.txt 来安装项目的依赖,具体命令如下:

1
poetry export --without-hashes --format=requirements.txt --output requirements.txt

更多的 Poetry 使用方法可以参考官方文档

其他依赖管理工具

除了 Poetry 外,还有一些其他的 Python 依赖管理工具,比如 Pdm 就是不错的选择,它整体和 Poetry 类似,它还包含了 Python 环境管理的功能,这样就不需要和 Miniconda 配合也可直接使用,但是在流行度上比 Poetry 稍微逊色一些(截止时间 2024 年 7 月,Poetry 的 Github Star 数是 30K,而 Pdm 的 Github Star 数是 7.6K),可能是 Pdm 比 Poetry 发布时间晚的关系(Poetry 是 2018 年发布的,而 Pdm 是 2020 年发布的),后面如果 Pdm 发展的好的话,说不定会超过 Poetry。

另外一个值得推荐的 Python 依赖管理工具是 Uv,它是一款用 Rust 编写的极速 Python 包安装和解析工具,旨在作为 Pip 和 Pip-tools 的替代品,并逐步发展成为一个全面的 Python 项目和包管理器,下面是它和其他工具安装依赖的速度对比图:

得益于 Rust 的高性能,在速度上 Uv 完全碾压了其他工具,但可惜的是 Uv 目前还是使用 requirements.txt 来管理依赖,这样就无法保证可以在不同环境安装相同的依赖,而且 Uv 也不能和 Poetry 一起使用,因为 Uv 是按照兼容 Pip 的思路进行开发,而 Poetry 内部已经不用 Pip 做依赖管理了。

尽管 Uv 无法和 Poetry 一起使用,但是我们可以在 CI/CD 中使用 Uv 来加速依赖安装,比如我们先用 Poetry 导出 requirements.txt 文件,然后在 CI/CD 中使用 Uv 来安装依赖,这样就可以大大缩短依赖安装的时间。

Pip 是否过时

虽然这些依赖管理工具很强大,但一些小型项目可能更多开发人员还是会选择 Pip,因为 Pip 无需额外安装其他工具,一般有 Python 环境就可以直接使用。另外一点是编程语言的原生工具也在不断发展,以 JS 为例,npm 刚开始时也不支持 lock 文件,但后面参考了一些 JS 的流行工具,在社区的共同努力下慢慢完善了这个功能,所以 Pip 也有可能在未来的某个版本中加入类似 Poetry 的功能。

所以在 Pip 没有发展完善之前,我们可以使用 Poetry 这样的工具来解决依赖管理的问题,同时也能简化项目配置,提高开发效率。

代码规范工具

另外一种常用的工具是代码规范类工具,因为 Python 的语法比较灵活,所以在团队协作中可能会出现代码风格不一致的问题,为了解决这个问题,每种编程语言都会有一些代码规范工具,这类工具包括代码格式化工具、代码检查工具等。

在 Python 中代码格式化工具有 BlackYAPFautopep8 等,而代码检查工具有 Flake8Pylintmypy 等,这些工具都可以通过 Pip 来安装,然后在项目中使用。

而最近比较流行的一个代码规范工具 Ruff,它同时集成了代码格式化和代码检查功能,可以帮助我们更好地在项目中统一代码风格。Ruff 是 Uv 开发团队开发的另外一款工具,同样是使用 Rust 语言进行编写,从而使它的性能远远高于其他同类型的工具,下面是 Ruff 和其他工具的性能对比图:

Ruff 内部集成了 Black 和 Flake8 等工具,下面我们就来介绍下 Ruff 如何安装及使用。

Ruff 安装

Ruff 可以通过 Pip 或 Poetry 进行安装,在项目中使用的话,推荐使用 Poetry 安装到 dev 开发依赖中,表示这个工具只在开发环境中使用,具体命令如下:

1
poetry add --dev ruff

如果想让 Ruff 在本地 IDE 中使用的话,建议是进行全局安装,这样就可以在任何项目中使用 Ruff,具体命令如下:

1
curl -LsSf https://astral.sh/ruff/install.sh | sh

安装完成后,可以通过 IDE 的插件来使用 Ruff,比如在 VSCode 中安装 Ruff 插件,然后在 VSCode 中使用快捷键 Ctrl+Shift+P 打开命令面板,输入 user config 打开用户配置文件,然后添加 Ruff 安装后的路径:

1
2
3
{
"ruff.path": ["/your/ruff/path"]
}

Ruff 使用示例

Ruff 主要有 2 个命令,分别是 ruff checkruff format,前者用于代码检查,后者用于代码格式化。

为了保证团队的代码风格一致,我们可以在 Git 的 hook 中添加 Ruff 检查命令,这样每个人在执行 git commit 命令时就会自动执行 Ruff 命令,如果检查失败则无法提交代码。

首先我们需要安装 pre-commit 工具,这个工具可以让我们在 Git hook 上轻松配置命令,同样将其安装到 dev 开发依赖即可:poetry add --dev pre-commit,然后在项目根目录下添加 .pre-commit-config.yaml 文件,内容如下:

1
2
3
4
5
6
7
8
9
10
11
repos:
- repo: https://github.com/astral-sh/ruff-pre-commit
# Ruff 版本
rev: v0.5.1
hooks:
# 执行 ruff check
- id: ruff
# 可选参数,自动修复代码
args: [--fix]
# 执行 ruff format
- id: ruff-format
  • 在文件中我们添加 Ruff 工具来处理代码提交时的检查
  • 我们添加了 2 个工具,第一个工具 id 是 ruff 表示执行 ruff check 命令,第二个工具 id 是 ruff-format 表示执行 ruff format 命令
  • 第一个工具还有一个可选参数 args: [--fix],表示会自动修复检查出有误的代码,但也不是所有代码都能自动修复,有些代码还是需要手动修复的

最后我们在项目中执行 pre-commit install 命令,将文件内容添加到 Git hook 中:

1
2
$ pre-commit install
pre-commit installed at .git/hooks/pre-commit

配置完成后,我们故意写一些代码错误,然后提交代码, 检查结果如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
$ git commit -m 'add some feature'
ruff.....................................................................Failed
- hook id: ruff
- exit code: 1

main.py:1:8: F401 [*] `requests` imported but unused
|
1 | import requests
| ^^^^^^^^ F401
|
= help: Remove unused import: `requests`

Found 1 error.
[*] 1 fixable with the `--fix` option.

ruff-format..............................................................Passed

可以看到代码检查任务失败了,报了 imported but unused 的错误,错误编号 F401,第二个任务代码格式化检查通过。

如果想在代码中不检查某行代码,可以在代码行后面加上 # noqa: {error_code},比如:

1
import requests # noqa: F401

这样就可以在 Ruff 检查时忽略这个错误。更多的 Ruff 使用方法可以参考官方文档

总结

今天我们介绍了 Python 中新一代的项目工具,新工具带来的好处是开发效率的提升,因为每个新工具都是在解决旧工具的不足之处,是在旧工具的基础上进行了优化和改进。作为一个热衷提高生产力的开发人员,可以在适当时机尝试使用这些新工具,如果觉得新工具不合适,也可以退回去重新使用旧工具,关键在于尝试。

关注我,一起学习各种人工智能和 GenAI 新技术,欢迎交流,如果你有什么想问想说的,欢迎在评论区留言。

]]>
+ + + 介绍新一代 Python 项目常用工具,以及如何在项目中使用 + + + + + + + + + + + + + + + + +
+ 高级 RAG 检索策略之知识图谱 @@ -511,31 +540,4 @@ - - 使用 LlamaIndex 结合 Eleasticsearch 进行 RAG 检索增强生成 - - https://zhaozhiming.github.io/2024/01/13/llamaindex-eleasticsearch-rga-practice/ - 2024-01-13T07:12:46.000Z - 2024-01-18T03:59:05.315Z - -

检索增强生成(Retrieval-Augmented Generation,RAG)是一种结合了检索(Retrieval)和生成(Generation)的技术,它有效地解决了大语言模型(LLM)的一些问题,比如幻觉、知识限制等。随着 RAG 技术的发展,RAG 涉及到的向量技术受到了大家的关注,向量数据库也慢慢被大家所了解,一些老牌的数据库厂商也纷纷表示支持向量检索,比如 Elasticsearch 也在最近的版本增加了向量检索的支持。本文将介绍 Elasticsearch 和 RAG 中相关的 Embedding 模型的部署,以及在 LLM 框架 LLamaIndex 中如何使用 Elasticsearch 进行文档索引入库和检索。

RAG 介绍

在使用 LLM 时我们经常会遇到这样一些情况,比如当我们的问题超出 LLM 的知识范围时,它要么解释说这个问题超出它的知识范围(这是 LLM 的知识限制),要么它会很自信地瞎编一些答案(这是我们所说的 LLM 幻觉)。

为了应对 LLM 的这些问题,RAG(检索增强生成)技术应运而生,RAG 的主要原理是将文档向量化后进行存储,在提出问题时将问题进行向量检索,检索出相关的文档,然后再将文档作为问题的上下文,一起发送给 LLM,让 LLM 来生成问题的答案,有了相关文档的支持,LLM 在内容的生成上就会参考这些文档,这样就可以有效地解决 LLM 的幻觉问题。同时,RAG 可以让 LLM 更快地了解到最新的信息,通常要让 LLM 了解到更新的信息,需要对 LLM 进行重新训练,训练方式不管是预训练还是微调,成本都是比较高的,而 RAG 只需要将最新的文档加入到数据库中即可,这样 LLM 就可以通过向量检索的方式来获取最新的信息。

关键字检索和语义检索

RAG 的相关技术包括向量检索,也称为语义检索,它不同于传统的关键字检索,关键字检索依赖于在文档中查找与查询中使用的确切词汇匹配的单词或短语,它通常只关注字面上的匹配,而不考虑查询的上下文或语义含义,而语义检索旨在理解查询的意图和上下文含义,不仅仅是文字匹配,它通过分析词语的语义关系(如同义词、词义消歧)来提高检索的相关性。

举一个简单的例子,比如我们输入苹果 2024 新品发布,关键字检索可能返回关于苹果公司 2024 年的任何新闻发布,但也可能包括与水果苹果相关的新品种发布的信息,语义检索则会查找出关于苹果公司最新电子产品发布的新闻,而忽略与水果苹果相关的内容。

Elasitcsearch(以下简称 ES) 虽然一开始只是全文搜索引擎,也就是关键字检索,但是随着向量检索技术的发展,ES 也开始支持向量检索,这让 ES 成为了一个既可以做关键字检索,又可以做语义检索的数据库。下面我们就来介绍 ES 数据库的部署。

Elasticsearch 部署

部署 ES 最简单的方式是通过 Docker,首先需要安装 Docker,可以参考 Docker 的官方安装文档

Docker 安装完成后开始安装 ES,我们需要使用 ES 的最新版本,因为最新的版本包括了向量检索的功能,目前最新的版本是8.11.3,安装启动命令如下:

1
docker run -d --name es -p 9200:9200 -p 9300:9300 -e "discovery.type=single-node" elasticsearch:8.11.3

使用docker run命令启动 ES 服务,-d参数表示以后台方式运行,--name参数表示容器的名称,-p参数表示端口映射,-e参数表示环境变量,elasticsearch:8.11.3表示使用elasticsearch相关版本的镜像。如果你是单机部署的话,可以不需要映射9300端口,这个端口主要用于 ES 集群内部节点之间的通信。

修改 ES 用户密码

ES 默认的配置会开启安全认证,这意味着在访问 ES 时需要通过用户名和密码认证,因此我们需要先获取到 ES 的用户名和密码,ES 的默认用户是elastic,如果不清楚该用户的密码,可以通过以下命令来重置用户密码:

1
2
3
4
5
6
7
8
9
10
11
12
# 进入 ES 容器
$ docker exec -it es bash
# 重置密码
$ bin/elasticsearch-reset-password -u elastic -i
This tool will reset the password of the [elastic] user.
You will be prompted to enter the password.
Please confirm that you would like to continue [y/N]y


Enter password for [elastic]:
Re-enter password for [elastic]:
Password for the [elastic] user successfully reset.

在 ES 容器中我们通过elasticsearch-reset-password命令来重置elastic用户的密码,重置完成后,我们可以通过在浏览器中输入https://localhost:9200来访问 ES(注意 url 地址是 https,不是 http,后面会讲如何关闭 https),首次访问时会提示你输入用户名和密码:

输入用户名和密码后,我们就可以看到 ES 相关的 JSON 信息。

关闭 ES SSL 认证

ES 为了加强系统的安全性,会默认开启 SSL 认证,在访问 ES 时需要使用 HTTPS 协议,但如果我们只是本地使用的话不太需要这种级别的安全认证,因此我们可以关闭 ES 的 SSL 认证,关闭 SSL 认证的需要修改 ES 的配置文件elascitsearch.yml。修改该文件我们先要将 ES 默认的配置文件拷贝到本地磁盘,然后修改配置文件,最后在 ES 容器启动时挂载修改后的配置文件。

首先我们将刚才启动的 ES 容器中的配置目录拷贝到本地磁盘,然后原来的 ES 容器就可以关闭了,命令如下:

1
2
3
4
# 拷贝配置文件
docker cp es:/usr/share/elasticsearch/config ./config
# 关闭 ES 容器
docker rm -f es

config文件夹包含了elascitsearch.yml和其他配置文件,然后我们修改elascitsearch.yml文件来关闭 SSL 认证,修改内容如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
# Enable encryption for HTTP API client connections, such as Kibana, Logstash, and Agents
xpack.security.http.ssl:
- enabled: true
+ enabled: false
keystore.path: certs/http.p12

# Enable encryption and mutual authentication between cluster nodes
xpack.security.transport.ssl:
- enabled: true
+ enabled: false
verification_mode: certificate
keystore.path: certs/transport.p12
truststore.path: certs/transport.p12

修改完成后,我们需要重新运行一个新的 ES 容器,并将修改后的配置文件挂载到容器中,命令如下:

1
docker run -d --name es -p 9200:9200 -p 9300:9300 -v "$PWD/config":/usr/share/elasticsearch/config -e "discovery.type=single-node" elasticsearch:8.11.3

等容器启动后,我们就可以通过http://localhost:9200来访问 ES 了。这里要注意的是因为重新部署了 ES 容器,所以刚才修改的用户密码也会失效,需要重新重置用户密码。

ES 监控工具

想要查看 ES 中的数据,如果是使用命令行工具的话可能不太方便,因此我们需要一个 GUI 工具,这里推荐elasticvue,一个基于浏览器的 ES GUI 工具,安装也非常简单,同样是使用 docker 来进行安装:

1
docker run -p 9080:8080 --name elasticvue -d cars10/elasticvue

然后我们在浏览器中输入http://localhost:9080来访问 elasticvue,进到首页后点击ADD ELASTICSEARCH CLUSTER按钮,可以看到如下界面:

根据上图上半部分的Configure提示,需要修改 ES 的配置文件elascitsearch.yml以接入 elasticvue,修改内容可以参考图中的Configure部分,修改完后重启 ES 容器即可。

1
docker restart es

然后在 elasticvue 中添加 ES 集群,输入 ES 的地址http://localhost:9200,选择Basic auth输入用户名和密码,这样就可以连上我们的 ES 服务了。

Embedding 模型部署

向量检索的核心是向量,而向量是由 Embedding 模型生成的,我们可以使用一些线上的 Embedding 模型,比如 OpenAI 的 Embedding 模型,也可以自己部署 Embedding 模型。这里我们选择部署自己的 Embedding 模型,我们使用 BAAI/bge-base-en-v1.5 模型,这是一个英文 Embedding 模型,可以用于英文的向量生成。

我们使用 FastChat 来部署 Embedding 模型,FastChat 是一个模型训练、部署、评估的开发平台,不仅支持 LLM 模型,还支持 Embedding 模型,下面来介绍如何使用 FastChat 部署 Embedding 模型。

首先我们要安装 FastChat,然后通过 FastChat 来部署一个兼容 OpenAI API 的 Embedding API 服务,安装命令如下:

1
pip3 install "fschat[model_worker,api]"

安装完成后,先使用 FastChat 的命令行工具来启动 controller 服务,命令如下:

1
2
3
4
5
6
$ python3 -m fastchat.serve.controller --host 0.0.0.0
2024-01-14 18:29:43 | INFO | controller | args: Namespace(dispatch_method='shortest_queue', host='0.0.0.0', port=21001, ssl=False)
2024-01-14 18:29:43 | ERROR | stderr | INFO: Started server process [1154]
2024-01-14 18:29:43 | ERROR | stderr | INFO: Waiting for application startup.
2024-01-14 18:29:43 | ERROR | stderr | INFO: Application startup complete.
2024-01-14 18:29:43 | ERROR | stderr | INFO: Uvicorn running on http://0.0.0.0:21001 (Press CTRL+C to quit)

然后重新打开一个终端,使用 FastChat 的命令行工具来启动 worker 服务,命令如下:

1
2
3
4
5
6
7
$ python3 -m fastchat.serve.model_worker --model-path BAAI/bge-base-en-v1.5 --host 0.0.0.0
2024-01-14 18:32:39 | INFO | model_worker | Loading the model ['bge-base-en-v1.5'] on worker 339a9e30 ...
2024-01-14 18:32:40 | INFO | model_worker | Register to controller
2024-01-14 18:32:40 | ERROR | stderr | INFO: Started server process [1229]
2024-01-14 18:32:40 | ERROR | stderr | INFO: Waiting for application startup.
2024-01-14 18:32:40 | ERROR | stderr | INFO: Application startup complete.
2024-01-14 18:32:40 | ERROR | stderr | INFO: Uvicorn running on http://0.0.0.0:21002 (Press CTRL+C to quit)

执行命令后,FastChat 会自动从 huggingface 上下载 BAAI/bge-base-en-v1.5 模型,下载完成后就会启动 worker 服务,worker 服务会自动连接到 controller 服务。

我们再打开一个终端,使用 FastChat 的命令行工具来启动 兼容 OpenAI API 的 API 服务,命令如下:

1
2
3
4
5
$ python3 -m fastchat.serve.openai_api_server --host 0.0.0.0 --port 8000
2024-01-14 18:37:10 | ERROR | stderr | INFO: Started server process [1405]
2024-01-14 18:37:10 | ERROR | stderr | INFO: Waiting for application startup.
2024-01-14 18:37:10 | ERROR | stderr | INFO: Application startup complete.
2024-01-14 18:37:10 | ERROR | stderr | INFO: Uvicorn running on http://0.0.0.0:8000(Press CTRL+C to quit)

服务启动后,我们可以访问http://localhost:8000/docs来查看 API 服务的 swagger 文档:

可以看到图中的/v1/embeddings接口就是我们需要调用的 Embedding 接口,我们可以通过 curl 命令来测试一下该接口,命令如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
curl -X 'POST' \
'https://localhost:8000/v1/embeddings' \
-H 'accept: application/json' \
-H 'Content-Type: application/json' \
-d '{
"model": "bge-base-en-v1.5",
"input": "hello"
}'

# 显示结果
{
"object": "list",
"data": [
{
"object": "embedding",
"embedding": [0.013750563375651836, …], # 向量数据
"index": 0
}
],
"model": "bge-base-en-v1.5",
"usage": {
"prompt_tokens": 3,
"total_tokens": 3
}
}

我们在请求参数中输入模型名称和需要被向量化的文本,命令执行完成后,我们可以看到返回的结果包含了 embedding 后的向量数据,并且返回格式跟 OpenAI API 的格式是一样的。

FastChat 更多的相关部署内容可以参考 FastChat 的文档

LlamaIndex 文件加载与检索

LlamaIndex 是继 LangChain 之后另外一个 LLM 应用开发框架,整体功能以 RAG 为主,现在也慢慢在开发一些 Agent 相关的功能。该框架的主要编程语言是 Python,具有广泛的社区支持和贡献,包括众多的 forks 和 stars,表明其在开发社区中的受欢迎程度和实用性。

下面我们来介绍使用 LlamaIndex 结合 ES 进行文档加载与检索,在开始编写代码之前,我们需要安装 LlamaIndex 和 ES 的 Python 包,命令如下:

1
pip install llama-index elasticsearch

Embedding 自定义类

安装完依赖包后,我们开始编写相关代码,首先我们需要创建一个自定义的 Embedding 类,这个 Embedding 类会调用我们刚才部署的 Embedding API 接口来实现文本的向量化,代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
from llama_index.embeddings.base import BaseEmbedding, Embedding
from llama_index.bridge.pydantic import PrivateAttr
from typing import Any, List

class CustomEmbeddings(BaseEmbedding):
"""Custom class for embeddings.

Args:
model_name (str): Mode for embedding.
url(str): Url for embedding model.
"""

_model_name: str = PrivateAttr()
_url: str = PrivateAttr()

def __init__(self, model_name: str, url: str, **kwargs: Any) -> None:
self._model_name = model_name
self._url = url
super().__init__(**kwargs)

@classmethod
def class_name(cls) -> str:
return "custom_embedding"

def _aget_query_embedding(self, query: str) -> Embedding:
return get_embedding(text=query, model_uid=self._model_name, url=self._url)

def _aget_text_embedding(self, text: str) -> Embedding:
return get_embedding(text=text, model_uid=self._model_name, url=self._url)

def _get_query_embedding(self, query: str) -> Embedding:
return get_embedding(text=query, model_uid=self._model_name, url=self._url)

def _get_text_embedding(self, text: str) -> Embedding:
return get_embedding(text=text, model_uid=self._model_name, url=self._url)

def _get_text_embeddings(self, texts: List[str]) -> List[Embedding]:
return get_embeddings(
list_of_text=texts, model_uid=self._model_name, url=self._url
)
  • 使用 LlamaIndex 实现自定义的 Embedding 类,需要继承 BaseEmbedding 类,并实现相关的方法
  • 这里我们实现了_aget_query_embedding_aget_text_embedding_get_query_embedding_get_text_embedding_get_text_embeddings这几个方法,这几个方法会调用其他公共方法来实现文本转向量的功能。

我们再来看一下get_embeddingget_embeddings这两个方法的实现,代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
import requests

def send_request(model_uid: str, text: str, url: str):
url = f"{url}/v1/embeddings"
request_body = {"model": model_uid, "input": text}
response = requests.post(url, json=request_body)
if response.status_code != 200:
raise RuntimeError(
f"Failed to create the embeddings, detail: {_get_error_string(response)}"
)
return response.json()

def get_embedding(text: str, model_uid: str, url: str) -> Embedding:
"""Get embedding."""
text = text.replace("\n", " ")
response_data = send_request(model_uid, text, url)
return response_data["data"][0]["embedding"]

def get_embeddings(
list_of_text: List[str], model_uid: str, url: str
) -> List[Embedding]:
"""Get embeddings."""
assert len(list_of_text) <= 2048, "The batch size should not be larger than 2048."

list_of_text = [text.replace("\n", " ") for text in list_of_text]
response_data = send_request(model_uid, list_of_text, url)
return [d["embedding"] for d in response_data["data"]]

  • get_embeddingget_embeddings都使用send_request来获取文本的向量数据,不同的地方在于一个参数是字符串,一个参数是字符串数组
  • send_request方法会发起 HTTP 请求调用 Embedding API 接口来实现文本向量化
  • 参考之前的 API 返回结果,Embedding 向量保存在一个数组中
  • get_embedding获取返回结果的第一个向量数据,get_embeddings获取所有的向量数据

向量化文档

有了自定义 Embedding 类,我们就可以使用 LlamaIndex 来实现文档的向量存储了,首先我们连接 ES 数据库,代码如下:

1
2
3
4
5
6
7
8
9
10
from llama_index.vector_stores import ElasticsearchStore
from llama_index import StorageContext

es_url = "http://{username}:{password}@localhost:9200"
index_name = "my_index"
store = ElasticsearchStore(
es_url=es_url,
index_name=index_name,
)
storage_context = StorageContext.from_defaults(vector_store=store)
  • 新建一个 ES store 来连接 ES,需要指定 ES 的地址和索引名称
  • ES 如果开启了安全认证,需要在 ES 的地址中添加用户名和密码
  • 使用 LlamaIndex 的 StorageContext 来集成 ES 的 store

我们再定义带有 Embedding 模型的 ServiceContext,代码如下:

1
2
3
4
5
6
7
8
9
10
from llama_index import ServiceContext
from custom_embedding import CustomEmbeddings

embedding_model_url = "http://localhost:8000"
embedding_model_name = "bge-base-en-v1.5"
service_context = ServiceContext.from_defaults(
embed_model=CustomEmbeddings(
url=embedding_model_url, model_name=embedding_model_name
),
)
  • embedding_model_url 是我们刚才部署的 Embedding API 的地址,model_name 是模型名称

接着我们来将文档转换为 LlamaIndex 的 Document 对象,我们可以使用 LlamaIndex 的示例文档paul_graham_essay来做演示,这篇文章是 Paul Graham 关于他个人生涯和工作经历的回顾,代码如下:

1
2
3
4
from llama_index import SimpleDirectoryReader

data_path = "./data" # paul_graham_essay.txt 所在的目录
documents = SimpleDirectoryReader(data_path).load_data()
  • SimpleDirectoryReader 对象可以对文件夹中的文件进行解析,txt 文件的解析不需要额外的依赖,但如果是其他格式的文件,比如 pdf,则需要安装相关的依赖 pypdf

我们将以上的对象组装在一起,示例代码如下:

1
2
3
4
5
6
7
from llama_index import VectorStoreIndex

index = VectorStoreIndex.from_documents(
documents,
storage_context=storage_context,
service_context=service_context,
)
  • 使用 VectorStoreIndex 集成 storage_context 和 service_context,并加载 documents
  • 不需要担心 ES 的索引是否已创建,如果没有该索引,LlamaIndex 会自动创建

代码执行后,我们就可以在 ES 中看到索引的文档了,我们通过 elasticvue 来查看索引的文档,如下图所示:

除了可以对整个文件夹进行加载外,我们还可以在已有的索引中添加新的文档,代码如下:

1
2
3
4
5
6
7
8
filepath = "./data/paul_graham_essay.txt"
index = VectorStoreIndex.from_vector_store(
vector_store=store,
storage_context=storage_context,
service_context=service_context,
)
document = SimpleDirectoryReader(input_files=[filepath]).load_data()[0]
index.insert(document)
  • 在 VectorStoreIndex 中传入 ES store 来加载已有的 ES 索引
  • SimpleDirectoryReader 也可以传入单个文件路径,这样就可以加载单个文件
  • 使用 VectorStoreIndex 的 insert 方法来添加新的文档

问题检索与生成

接下来我们再使用 LlamaIndex 来对问题进行检索,代码如下:

1
2
3
4
5
6
7
8
9
query_engine = index.as_query_engine()
response = query_engine.query("What did the author do growing up?")
print(f"response: {response}")

# 显示结果
# response: The author took philosophy courses, but found them boring.
# As a result, the author decided to switch to AI and started teaching
# themselves Lisp, which was regarded as the language of AI at the time.
# The author also reverse-engineered SHRDLU for their undergraduate thesis.
  • LlamaIndex 默认使用 OpenAI 的 LLM 来做生成,因此在执行代码之前,需要将 OPENAI_API_KEY 环境变量设置为你的 API KEY

我们询问了一个关于作者成长经历的问题,LlamaIndex 会先使用向量检索来检索相关的文档,然后再使用 LLM 来生成答案,我们可以看到 LlamaIndex 生成的答案是正确的。

如果我们将 LlamaIndex 中的 LLM 取消,那么 response 的结果会变成结合了相关文档的提示词模板,如下所示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
service_context = ServiceContext.from_defaults(
llm=None, # 取消LLM
embed_model=CustomEmbeddings(
url=embedding_model_url, model_name=embedding_model_name
),
)
......
response = query_engine.query("What did the author do growing up?")
print(f"response: {response}")

# 显示结果
# LLM is explicitly disabled. Using MockLLM.
# response: Context information is below.
# ---------------------
# file_path: data/paul_graham_essay.txt
#
# I don't think it was entirely blabla ......
# ---------------------
# Given the context information and not prior knowledge, answer the query.
# Query: What did the author do growing up?
# Answer:
  • 只需在 ServiceContext 中添加参数llm=none即可取消默认的 OpenAI LLM
  • 其他代码与原来的一样

可以看到同样的问题,不使用 LLM 的情况下返回的结果是一个包含了相关文档的提示词模板。

在 response 对象中,我们还可以通过response.source_nodes可以获取到检索到的文档信息,文档的 JSON 信息如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
{
"py/object": "llama_index.schema.NodeWithScore",
"py/state": {
"__dict__": {
"node": {
"py/object": "llama_index.schema.TextNode",
"py/state": {
"__dict__": {
"metadata": {"file_path": "data/paul_graham_essay.txt",},
"hash": "72baf405cfa89677a1a409d46d58dab2f4c183adcba5602d8b01a27a05d9a7a5",
"text": "blabla...",
"start_char_idx": 53611,
"end_char_idx": 57967
},
}
},
"score": 1.0
},
}
},
{
"py/object": "llama_index.schema.NodeWithScore",
"py/state": {"__dict__": {},}
}
  • 可以看到 LlamaIndex 根据问题检索出 2 个 Node(可以把 Node 理解成文档的分块)
  • 每个 Node 有文本内容 text,匹配分数 score 等属性

LlamaIndex 默认是使用向量检索,我们也可以将其替换为其他检索方式,代码如下:

1
2
3
4
5
6
7
8
9
10
11
from llama_index.vector_stores.types import VectorStoreQueryMode

# 使用关键字检索
query_engine = index.as_query_engine(
vector_store_query_mode=VectorStoreQueryMode.TEXT_SEARCH,
)

# 使用混合检索
query_engine = index.as_query_engine(
vector_store_query_mode=VectorStoreQueryMode.HYBRID,
)

更多的 LlamaIndex 用法可以参考官方文档

总结

RAG 是 LLM 技术的一个重要方向,它不仅可以解决 LLM 中存在的一些问题,而且可以帮助我们打造更高质量的 LLM 应用。本文从 ES 和 Embedding 模型的部署一步步展开,结合 LLM 框架 LlamaIndex 来实现 RAG 的检索增强生成,并介绍了在实践过程中相关的原理和注意事项。希望本文能够帮助大家更好地理解 RAG 技术,如果对文章内容有疑问或者建议,欢迎在评论区留言。

关注我,一起学习各种人工智能和 AIGC 新技术,欢迎交流,如果你有什么想问想说的,欢迎在评论区留言。

]]>
- - - 介绍 Elasticsearch 部署、Embedding 模型部署以及如何使用 LlamaIndex 进行 RAG - - - - - - - - - - - - - - -
- diff --git a/categories/ai/index.html b/categories/ai/index.html index 5a749dd07..b89d47712 100644 --- a/categories/ai/index.html +++ b/categories/ai/index.html @@ -129,6 +129,20 @@

2024

+ + - -
diff --git a/categories/ai/page/2/index.html b/categories/ai/page/2/index.html index 1d1de409f..fd3aaa6f9 100644 --- a/categories/ai/page/2/index.html +++ b/categories/ai/page/2/index.html @@ -129,6 +129,20 @@

2024

+ + -
diff --git a/categories/ai/page/3/index.html b/categories/ai/page/3/index.html index 4d73d971f..dacc0c6a1 100644 --- a/categories/ai/page/3/index.html +++ b/categories/ai/page/3/index.html @@ -129,6 +129,20 @@

2023

+ + - -
diff --git a/categories/ai/page/4/index.html b/categories/ai/page/4/index.html index 624d0f392..f6f50d480 100644 --- a/categories/ai/page/4/index.html +++ b/categories/ai/page/4/index.html @@ -129,6 +129,20 @@

2023

+ + - -
diff --git a/categories/ai/page/5/index.html b/categories/ai/page/5/index.html index 43cadf459..95067e5dc 100644 --- a/categories/ai/page/5/index.html +++ b/categories/ai/page/5/index.html @@ -129,6 +129,20 @@

2023

+ +

部署 LangChain 和 ChatGLM2 来打造自有知识库问答系统

diff --git a/images/post/2024/07/next-generation-python-tools.jpg b/images/post/2024/07/next-generation-python-tools.jpg new file mode 100644 index 0000000000000000000000000000000000000000..b93e5e727413e1f5afc9ea63e4b87924a083898c GIT binary patch literal 58050 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ξ}!JyGpLb)u=fxouapMNd_Km6oN; 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 literal 0 HcmV?d00001 diff --git a/images/post/2024/07/uv-compare.png b/images/post/2024/07/uv-compare.png new file mode 100644 index 0000000000000000000000000000000000000000..cfbd581ad475a193a59d861af1c944b702895614 GIT binary patch literal 45604 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≺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 F{txiqxY7Us literal 0 HcmV?d00001 diff --git a/index.html b/index.html index 98aecd9b0..05281ed1f 100644 --- a/index.html +++ b/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 都 2024 年了,你还在用 pip 吗? +

+
+
+
Published on: +
+ +
+ + + +

编程语言 Python 随着 AI 的发展越来越受开发人员的喜爱,目前已经是最流行的编程语言,但由于 Python 是一门相对较老的语言,并且经过 Python2 到 Python3 这一漫长而复杂的迁移历程,使得一些 Python 开发人员可能还在使用一些过时的工具和库。今天我们就来带大家了解当前 Python 生态系统中最流行最实用的开发工具,让你彻底告别那些老古董

+ + + + +
+ +
+
-

- -
-

- 用 API 方式免费使用 GPT 写小说 -

-
-
-
Published on: -
-
Tags: - -chatgpt llm gpt -
-
- - - -

最近 OpenAI 宣布将降低网页版 ChatGPT 的使用门槛,允许没有账号的用户使用,这一好消息使我们可以在网页上免费访问 ChatGPT,但和 API 相比仍然缺乏灵活性,通过 API 开发人员可以编写代码来与 ChatGPT 进行自动化交互。今天我们将介绍如何利用 OpenAI 的这一免费功能,将其转化为 API 的形式来进行使用,并介绍如何通过这种方式来写一篇小说。

- - - - -
-
diff --git a/page/10/index.html b/page/10/index.html index aa056a6bf..00bdebf94 100644 --- a/page/10/index.html +++ b/page/10/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,32 @@

历年文章

+
+

+ 物极必反 +

+
+
+
Published on: +
+
Tags: + +thinking +
+
+ + +

物极必反是很多人都明白的一个道理,在软件开发中也一样,比如过度设计,这也是为什么Fackbook推崇“完成比完美更重要”的原因。

+ + + + +
+ +
+
- - -
-

- 『我』的产品说明书 -

-
-
-
Published on: -
-
Tags: - -teamwork -
-
- - -

最近使用引导者方法——的产品说明书,为团队做了一次沟通交流,让大家彼此加深了了解,同时从中得到了团队成员的喜好、特长等信息,大家觉得这种方式挺有意思,所以今天在这里分享一下。

- - - - -
-
diff --git a/page/11/index.html b/page/11/index.html index 31a53c570..a458862bf 100644 --- a/page/11/index.html +++ b/page/11/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,32 @@

历年文章

+
+

+ 『我』的产品说明书 +

+
+
+
Published on: +
+
Tags: + +teamwork +
+
+ + +

最近使用引导者方法——的产品说明书,为团队做了一次沟通交流,让大家彼此加深了了解,同时从中得到了团队成员的喜好、特长等信息,大家觉得这种方式挺有意思,所以今天在这里分享一下。

+ + + + +
+ +
+
- - -
-

- 科学上网利器Shadowsocks -

-
-
-
Published on: -
-
Tags: - -ss vps -
-
- - -

前段时间AppStore出了一款神器Surge,可以让iOS像其他平台使用Shadowsocks(以下简称ss)一样地轻松科学上网,因为以前都是用买的SSH或VPN科学上网,所以对自建的VPN服务这一块没有太多关注,甚至错过了ss这种成熟的工具。最近试用了过后觉得非常方便,所以在这里记录一下。

- - - - -
-
diff --git a/page/12/index.html b/page/12/index.html index 04ffa6e95..05f912cba 100644 --- a/page/12/index.html +++ b/page/12/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,32 @@

历年文章

+
+

+ 科学上网利器Shadowsocks +

+
+
+
Published on: +
+
Tags: + +ss vps +
+
+ + +

前段时间AppStore出了一款神器Surge,可以让iOS像其他平台使用Shadowsocks(以下简称ss)一样地轻松科学上网,因为以前都是用买的SSH或VPN科学上网,所以对自建的VPN服务这一块没有太多关注,甚至错过了ss这种成熟的工具。最近试用了过后觉得非常方便,所以在这里记录一下。

+ + + + +
+ +
+
- - -
-

- 使用Spring-data进行Redis操作 -

-
-
-
Published on: -
-
Tags: - -redis spring -
-
- - -

Redis相信大家都听说过,它是一个开源的key-value缓存数据库,有很多Java的客户端支持,比较有名的有Jedis,JRedis等(见这里)。当然我们可以使用客户端的原生代码实现redis的操作,但实际上在spring中就已经集成了这些客户端的使用,下面我们就以Jedis为例来介绍一下Spring中关于Redis的配置。

- - - - -
-
diff --git a/page/13/index.html b/page/13/index.html index e4fcfc9ba..636b0aaac 100644 --- a/page/13/index.html +++ b/page/13/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,32 @@

历年文章

+
+

+ 使用Spring-data进行Redis操作 +

+
+
+
Published on: +
+
Tags: + +redis spring +
+
+ + +

Redis相信大家都听说过,它是一个开源的key-value缓存数据库,有很多Java的客户端支持,比较有名的有Jedis,JRedis等(见这里)。当然我们可以使用客户端的原生代码实现redis的操作,但实际上在spring中就已经集成了这些客户端的使用,下面我们就以Jedis为例来介绍一下Spring中关于Redis的配置。

+ + + + +
+ +
+
- - -
-

- 使用Docker打造自己的开发环境 -

-
-
-
Published on: -
-
Tags: - -docker -
-
- - -

作为这个时代的程序员真的很幸福,每天不但有一些改变世界的产品出现,而且提高程序员开发效率的工具也是层出不穷。之前介绍过如何使用Vagrant来创建Ceph环境,使用Vagrant可以很方便的管理我们的虚拟机,同时来定制开发团队的开发环境,现在又出现了Docker,让我们有了一个更好的选择。

- - - - -
-
diff --git a/page/14/index.html b/page/14/index.html index c61803c63..97d9d8210 100644 --- a/page/14/index.html +++ b/page/14/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,32 @@

历年文章

+
+

+ 使用Docker打造自己的开发环境 +

+
+
+
Published on: +
+
Tags: + +docker +
+
+ + +

作为这个时代的程序员真的很幸福,每天不但有一些改变世界的产品出现,而且提高程序员开发效率的工具也是层出不穷。之前介绍过如何使用Vagrant来创建Ceph环境,使用Vagrant可以很方便的管理我们的虚拟机,同时来定制开发团队的开发环境,现在又出现了Docker,让我们有了一个更好的选择。

+ + + + +
+ +
+
- - -
-

- 让你的安卓开发更容易(二)——Genymotion -

-
-
-
Published on: -
-
Tags: - -android genymotion -
-
- - -

以前听过一个笑话,说是一个App好不容易拿到100万的融资,但是没几天就花完了,问创始人这钱怎么花的?创始人说:没什么,就是每个型号的Android手机各买了一个来做测试,钱就花完了。

-

Android开发需要有强大的模拟器来避免这种尴尬,Genymotion是一个Android模拟器,比起Google官方的AVD(Android Virtual Devices),它有着启动快速,安装方便,简单上手的特点。

- - - - -
-
diff --git a/page/15/index.html b/page/15/index.html index 5af3bda62..14c14a1d6 100644 --- a/page/15/index.html +++ b/page/15/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 让你的安卓开发更容易(二)——Genymotion +

+
+
+
Published on: +
+
Tags: + +android genymotion +
+
+ + +

以前听过一个笑话,说是一个App好不容易拿到100万的融资,但是没几天就花完了,问创始人这钱怎么花的?创始人说:没什么,就是每个型号的Android手机各买了一个来做测试,钱就花完了。

+

Android开发需要有强大的模拟器来避免这种尴尬,Genymotion是一个Android模拟器,比起Google官方的AVD(Android Virtual Devices),它有着启动快速,安装方便,简单上手的特点。

+ + + + +
+ +
+ - - diff --git a/page/16/index.html b/page/16/index.html index 74f3e2865..5fe3ba62b 100644 --- a/page/16/index.html +++ b/page/16/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,31 @@

历年文章

+ +
- - - diff --git a/page/17/index.html b/page/17/index.html index 36a54b864..cbdd3f5ff 100644 --- a/page/17/index.html +++ b/page/17/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+ +
- - - diff --git a/page/18/index.html b/page/18/index.html index 307b51720..297f732d1 100644 --- a/page/18/index.html +++ b/page/18/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,31 @@

历年文章

+ +
- - -
-

- Jacoco的原理 -

-
-
-
Published on: -
-
Tags: - -jacoco -
-
- -

覆盖率计数器

Jacoco使用一系列的不同的计数器来做覆盖率的度量计算。所有这些计数器都是从java的class文件中获取信息,这些class文件可以(可选)包含调试的信息在里面。即使在没有源码的情况下,这种方法也可以实时有效地对应用程序进行度量和分析。在大部分情况下,收集到的信息可以映射到源码,可视化到每一行代码的粒度。但这种方法还是有一些限制。这些class文件必须使用调试信息来编译,这样才可以计算行的覆盖率和提供出源码的高亮。但不是所有的JAVA语言的结构都可以直接编译成一致的二进制代码。在这种情况下,java 编译器会创建所谓的“合成”代码,会导致产生一些不期望得到的覆盖率结果。

- - - - -
-
diff --git a/page/19/index.html b/page/19/index.html index 90834f47c..33ff18d58 100644 --- a/page/19/index.html +++ b/page/19/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,31 @@

历年文章

+
+

+ Jacoco的原理 +

+
+
+
Published on: +
+
Tags: + +jacoco +
+
+ +

覆盖率计数器

Jacoco使用一系列的不同的计数器来做覆盖率的度量计算。所有这些计数器都是从java的class文件中获取信息,这些class文件可以(可选)包含调试的信息在里面。即使在没有源码的情况下,这种方法也可以实时有效地对应用程序进行度量和分析。在大部分情况下,收集到的信息可以映射到源码,可视化到每一行代码的粒度。但这种方法还是有一些限制。这些class文件必须使用调试信息来编译,这样才可以计算行的覆盖率和提供出源码的高亮。但不是所有的JAVA语言的结构都可以直接编译成一致的二进制代码。在这种情况下,java 编译器会创建所谓的“合成”代码,会导致产生一些不期望得到的覆盖率结果。

+ + + + +
+ +
+
- - -
-

- 在JavaScript中进行文件处理,第三部分:处理事件和错误 -

-
-
-
Published on: -
-
Tags: - -javascript -
-
- -

译注:原文是《JavaScript高级程序设计》的作者Nicholas Zakas写的,本翻译纯属为自己学习而做,仅供参考。原文链接:这里

-
-

FileReader对象用来读取浏览器可以访问的文件的内容。在我前一篇blog中,你学习到了如何使用FileReader对象轻松读取文件,并将文件内容转换为各种形式。FileReader在很多方面和XMLHttpRequest非常相似。

- - - - -
-
diff --git a/page/2/index.html b/page/2/index.html index 118126156..0aabc50cc 100644 --- a/page/2/index.html +++ b/page/2/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 用 API 方式免费使用 GPT 写小说 +

+
+
+
Published on: +
+
Tags: + +chatgpt llm gpt +
+
+ + + +

最近 OpenAI 宣布将降低网页版 ChatGPT 的使用门槛,允许没有账号的用户使用,这一好消息使我们可以在网页上免费访问 ChatGPT,但和 API 相比仍然缺乏灵活性,通过 API 开发人员可以编写代码来与 ChatGPT 进行自动化交互。今天我们将介绍如何利用 OpenAI 的这一免费功能,将其转化为 API 的形式来进行使用,并介绍如何通过这种方式来写一篇小说。

+ + + + +
+ +
+
- - -
-

- 使用 LlamaIndex 结合 Eleasticsearch 进行 RAG 检索增强生成 -

-
-
-
Published on: -
- -
- - - -

检索增强生成(Retrieval-Augmented Generation,RAG)是一种结合了检索(Retrieval)和生成(Generation)的技术,它有效地解决了大语言模型(LLM)的一些问题,比如幻觉、知识限制等。随着 RAG 技术的发展,RAG 涉及到的向量技术受到了大家的关注,向量数据库也慢慢被大家所了解,一些老牌的数据库厂商也纷纷表示支持向量检索,比如 Elasticsearch 也在最近的版本增加了向量检索的支持。本文将介绍 Elasticsearch 和 RAG 中相关的 Embedding 模型的部署,以及在 LLM 框架 LLamaIndex 中如何使用 Elasticsearch 进行文档索引入库和检索。

- - - - -
-
diff --git a/page/20/index.html b/page/20/index.html index 7d213851a..b08d4e78d 100644 --- a/page/20/index.html +++ b/page/20/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 在JavaScript中进行文件处理,第三部分:处理事件和错误 +

+
+
+
Published on: +
+
Tags: + +javascript +
+
+ +

译注:原文是《JavaScript高级程序设计》的作者Nicholas Zakas写的,本翻译纯属为自己学习而做,仅供参考。原文链接:这里

+
+

FileReader对象用来读取浏览器可以访问的文件的内容。在我前一篇blog中,你学习到了如何使用FileReader对象轻松读取文件,并将文件内容转换为各种形式。FileReader在很多方面和XMLHttpRequest非常相似。

+ + + + +
+ +
+
- - -
-

- jenkins的2个问题 -

-
-
-
Published on: -
-
Tags: - -jenkins -
-
- -

最近CI服务器从老版本的hudson升级为jenkins,遇到了2个问题,记录一下:

- - - - -
-
diff --git a/page/21/index.html b/page/21/index.html index c6566f590..208d96193 100644 --- a/page/21/index.html +++ b/page/21/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,31 @@

历年文章

+
+

+ jenkins的2个问题 +

+
+
+
Published on: +
+
Tags: + +jenkins +
+
+ +

最近CI服务器从老版本的hudson升级为jenkins,遇到了2个问题,记录一下:

+ + + + +
+ +
+
- - -
-

- 在junit中添加fail--有test失败即build failed -

-
-
-
Published on: -
-
Tags: - -ant junit -
-
- -

项目使用jenkins做持续集成,ant来构建,发现在跑junit单元测试的时候,如果有test case失败了,ci的状态是黄色的unstable,而不是红色的failed,看起来很不爽。个人觉得build只有两种状态最好,绿色stable和红色failed,黄色让人看起来很困惑,是要fix好呢还是不fix也可以呢?

- - - - -
-
@@ -438,6 +438,8 @@

+ + diff --git a/page/22/index.html b/page/22/index.html new file mode 100644 index 000000000..f17ff52eb --- /dev/null +++ b/page/22/index.html @@ -0,0 +1,255 @@ + + + + + + Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+ + + + + + + +
+ +
+
+ +
+

+ 在junit中添加fail--有test失败即build failed +

+
+
+
Published on: +
+
Tags: + +ant junit +
+
+ +

项目使用jenkins做持续集成,ant来构建,发现在跑junit单元测试的时候,如果有test case失败了,ci的状态是黄色的unstable,而不是红色的failed,看起来很不爽。个人觉得build只有两种状态最好,绿色stable和红色failed,黄色让人看起来很困惑,是要fix好呢还是不fix也可以呢?

+ + + + +
+ +
+ + + + +
+
+
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/page/3/index.html b/page/3/index.html index 285125a03..e1bc3d525 100644 --- a/page/3/index.html +++ b/page/3/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 使用 LlamaIndex 结合 Eleasticsearch 进行 RAG 检索增强生成 +

+
+
+
Published on: +
+ +
+ + + +

检索增强生成(Retrieval-Augmented Generation,RAG)是一种结合了检索(Retrieval)和生成(Generation)的技术,它有效地解决了大语言模型(LLM)的一些问题,比如幻觉、知识限制等。随着 RAG 技术的发展,RAG 涉及到的向量技术受到了大家的关注,向量数据库也慢慢被大家所了解,一些老牌的数据库厂商也纷纷表示支持向量检索,比如 Elasticsearch 也在最近的版本增加了向量检索的支持。本文将介绍 Elasticsearch 和 RAG 中相关的 Embedding 模型的部署,以及在 LLM 框架 LLamaIndex 中如何使用 Elasticsearch 进行文档索引入库和检索。

+ + + + +
+ +
+
- - -
-

- 使用 API 生成 Stable Diffustion 文字和二维码隐藏图片 -

-
-
-
Published on: -
- -
- - - -

Stable Diffusion 前段时间有几个比较火的效果,一个是将文字隐藏在图片中,放大看时是一张正常图片,缩小看却可以看到图片中的隐藏文字,另外一个效果与前者类似,但是图片中隐藏的是一个二维码,通过扫描图片可以进入二维码中的网址。由于出图效果好,很多人想要根据自己的需求制作这种图片,甚至有人在网上出售这种图片的定制服务。今天我们就来介绍下如何使用 API 的方式 在 Stable Diffusion 中实现这种效果。

- - - - -
-
diff --git a/page/4/index.html b/page/4/index.html index 31a1d3bfd..c803558ef 100644 --- a/page/4/index.html +++ b/page/4/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 使用 API 生成 Stable Diffustion 文字和二维码隐藏图片 +

+
+
+
Published on: +
+ +
+ + + +

Stable Diffusion 前段时间有几个比较火的效果,一个是将文字隐藏在图片中,放大看时是一张正常图片,缩小看却可以看到图片中的隐藏文字,另外一个效果与前者类似,但是图片中隐藏的是一个二维码,通过扫描图片可以进入二维码中的网址。由于出图效果好,很多人想要根据自己的需求制作这种图片,甚至有人在网上出售这种图片的定制服务。今天我们就来介绍下如何使用 API 的方式 在 Stable Diffusion 中实现这种效果。

+ + + + +
+ +
+
- - -
-

- 使用 Docker 部署 AI 环境 -

-
-
-
Published on: -
- -
- - - -

之前给大家介绍了主机安装方式——如何在 Ubuntu 操作系统下安装部署 AI 环境,但随着容器化技术的普及,越来越多的程序以容器的形式进行部署,通过容器的方式不仅可以简化部署流程,还可以随时切换不同的环境。实际上很多云服务厂商也是这么干的,用一台带有 NVIDIA 显卡的机器来部署多个容器,然后通过容器的方式来提供给用户使用,这样就可以充分利用显卡资源了。今天给大家介绍一下如何使用 Docker 的方式来部署我们之前部署过的 AI 环境。

- - - - -
-
diff --git a/page/5/index.html b/page/5/index.html index 153a31a86..71e50ddb2 100644 --- a/page/5/index.html +++ b/page/5/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 使用 Docker 部署 AI 环境 +

+
+
+
Published on: +
+ +
+ + + +

之前给大家介绍了主机安装方式——如何在 Ubuntu 操作系统下安装部署 AI 环境,但随着容器化技术的普及,越来越多的程序以容器的形式进行部署,通过容器的方式不仅可以简化部署流程,还可以随时切换不同的环境。实际上很多云服务厂商也是这么干的,用一台带有 NVIDIA 显卡的机器来部署多个容器,然后通过容器的方式来提供给用户使用,这样就可以充分利用显卡资源了。今天给大家介绍一下如何使用 Docker 的方式来部署我们之前部署过的 AI 环境。

+ + + + +
+ +
+
- - -
-

- 大语言模型代码生成工具 -

-
-
-
Published on: -
-
Tags: - -code ai -
-
- - - -

随着大语言模型技术的发展,人工智能的能力应用到了各个领域,包括与开发人员息息相关的代码生成领域,这个领域并不新鲜,但借助大型语言模型的力量,它被赋予了全新的生命力和可能性。在数字化的世界中,代码是一种语言,是人类与机器沟通的桥梁。借助大型语言模型的力量,我们可以让这座桥梁更加强大,更加智能。今天我们将深入探讨大型语言模型在代码生成方面的应用。

- - - - -
-
diff --git a/page/6/index.html b/page/6/index.html index 340fc293a..c5ab2a6df 100644 --- a/page/6/index.html +++ b/page/6/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+
+

+ 大语言模型代码生成工具 +

+
+
+
Published on: +
+
Tags: + +code ai +
+
+ + + +

随着大语言模型技术的发展,人工智能的能力应用到了各个领域,包括与开发人员息息相关的代码生成领域,这个领域并不新鲜,但借助大型语言模型的力量,它被赋予了全新的生命力和可能性。在数字化的世界中,代码是一种语言,是人类与机器沟通的桥梁。借助大型语言模型的力量,我们可以让这座桥梁更加强大,更加智能。今天我们将深入探讨大型语言模型在代码生成方面的应用。

+ + + + +
+ +
+
- - -
-

- 使用 TestCafe 进行 Web 自动化测试 -

-
-
-
Published on: -
- -
- - - -

Web 自动化测试,又叫 UI 自动化测试,国外叫 End-to-End Test(e2e, 端到端测试),可以让重复繁琐的手工测试( 俗称点点点)通过程序自动执行,可以极大地提升测试人员的效率(想象一下以前如果回归测试要测试一天的话,使用自动化测试可能只需要不到 1 小时的时间)。

-

今天介绍一个自动化测试工具新贵——TestCafe,它的功能和开发团队都很棒,但一直比较低调所以没有其他同类工具出名,如果你对自动化测试感兴趣,不妨和我一起来了解它。

- - - - -
-
diff --git a/page/7/index.html b/page/7/index.html index cd26d902a..5cba53a06 100644 --- a/page/7/index.html +++ b/page/7/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,34 @@

历年文章

+
+

+ 使用 TestCafe 进行 Web 自动化测试 +

+
+
+
Published on: +
+ +
+ + + +

Web 自动化测试,又叫 UI 自动化测试,国外叫 End-to-End Test(e2e, 端到端测试),可以让重复繁琐的手工测试( 俗称点点点)通过程序自动执行,可以极大地提升测试人员的效率(想象一下以前如果回归测试要测试一天的话,使用自动化测试可能只需要不到 1 小时的时间)。

+

今天介绍一个自动化测试工具新贵——TestCafe,它的功能和开发团队都很棒,但一直比较低调所以没有其他同类工具出名,如果你对自动化测试感兴趣,不妨和我一起来了解它。

+ + + + +
+ +
+
- - -
-

- 基于 Expo 的 React Native 消息推送 -

-
-
-
Published on: -
- -
- - - -

ExpoReact Native 开发的一个神器,正如 Expo 官网上所说,Expo 之于 React Native 就像 Rails 之于 Ruby,它提供了很多超越原生 React Native API 的功能,包括二维码扫描、存储、内部浏览器等,甚至还可以使用 Expo 进行 APP 的打包,完全不需要使用 XCode 和 Android Studio。

-

而消息推送则是 APP 应用非常常见的一个功能,今天就来介绍一下基于 Expo 的 React Native 消息推送功能是如何开发的吧。

- - - - -
-
diff --git a/page/8/index.html b/page/8/index.html index af139a375..45503e224 100644 --- a/page/8/index.html +++ b/page/8/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,34 @@

历年文章

+
+

+ 基于 Expo 的 React Native 消息推送 +

+
+
+
Published on: +
+ +
+ + + +

ExpoReact Native 开发的一个神器,正如 Expo 官网上所说,Expo 之于 React Native 就像 Rails 之于 Ruby,它提供了很多超越原生 React Native API 的功能,包括二维码扫描、存储、内部浏览器等,甚至还可以使用 Expo 进行 APP 的打包,完全不需要使用 XCode 和 Android Studio。

+

而消息推送则是 APP 应用非常常见的一个功能,今天就来介绍一下基于 Expo 的 React Native 消息推送功能是如何开发的吧。

+ + + + +
+ +
+
- - - diff --git a/page/9/index.html b/page/9/index.html index 5e1972702..cf027054b 100644 --- a/page/9/index.html +++ b/page/9/index.html @@ -133,6 +133,10 @@

最新文章

@@ -170,7 +170,7 @@

标签云

@@ -180,6 +180,33 @@

历年文章

+ +
- - -
-

- 物极必反 -

-
-
-
Published on: -
-
Tags: - -thinking -
-
- - -

物极必反是很多人都明白的一个道理,在软件开发中也一样,比如过度设计,这也是为什么Fackbook推崇“完成比完美更重要”的原因。

- - - - -
-
diff --git a/tags/miniconda/index.html b/tags/miniconda/index.html new file mode 100644 index 000000000..8a8473f7d --- /dev/null +++ b/tags/miniconda/index.html @@ -0,0 +1,187 @@ + + + + + + Tag: miniconda - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+ + + + + + + + + +

2024

+ + + + +
+ + + +
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/tags/poetry/index.html b/tags/poetry/index.html new file mode 100644 index 000000000..e47ae9460 --- /dev/null +++ b/tags/poetry/index.html @@ -0,0 +1,187 @@ + + + + + + Tag: poetry - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+ + + + + + + + + +

2024

+ + + + +
+ + + +
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/tags/python/index.html b/tags/python/index.html index 8130c1110..545ac67ab 100644 --- a/tags/python/index.html +++ b/tags/python/index.html @@ -127,6 +127,25 @@

+

2024

+ + + +
+ +

2015

diff --git a/tags/ruff/index.html b/tags/ruff/index.html new file mode 100644 index 000000000..cb17b94a4 --- /dev/null +++ b/tags/ruff/index.html @@ -0,0 +1,187 @@ + + + + + + Tag: ruff - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+ + + + + + + + + +

2024

+ + + + +
+ + + +
+ +
+ + + + + + + + + +
+
+ + + + + diff --git a/tags/uv/index.html b/tags/uv/index.html new file mode 100644 index 000000000..b8e045f62 --- /dev/null +++ b/tags/uv/index.html @@ -0,0 +1,187 @@ + + + + + + Tag: uv - Hacker and Geeker's Way + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ +
+
+ + + + + + + + + +

2024

+ + + + +
+ + + +
+ +
+ + + + + + + + + +
+
+ + + + +