mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-25 23:11:02 +08:00
256 lines
12 KiB
Markdown
256 lines
12 KiB
Markdown
[#]: collector: "lujun9972"
|
||
[#]: translator: "amwps290"
|
||
[#]: reviewer: "wxy"
|
||
[#]: publisher: "wxy"
|
||
[#]: url: "https://linux.cn/article-13117-1.html"
|
||
[#]: subject: "A hands-on tutorial of SQLite3"
|
||
[#]: via: "https://opensource.com/article/21/2/sqlite3-cheat-sheet"
|
||
[#]: author: "Klaatu https://opensource.com/users/klaatu"
|
||
|
||
SQLite3 实践教程
|
||
======
|
||
|
||
> 开始使用这个功能强大且通用的数据库吧。
|
||
|
||
![](https://img.linux.net.cn/data/attachment/album/202102/14/131146jsx2kvyobwxwswct.jpg)
|
||
|
||
应用程序经常需要保存数据。无论你的用户是创建简单的文本文档、复杂的图形布局、游戏进度还是错综复杂的客户和订单号列表,软件通常都意味着生成数据。有很多方法可以存储数据以供重复使用。你可以将文本转储为 INI、[YAML][2]、XML 或 JSON 等配置格式,可以输出原始的二进制数据,也可以将数据存储在结构化数据库中。SQLite 是一个自包含的、轻量级数据库,可轻松创建、解析、查询、修改和传输数据。
|
||
|
||
- 下载 [SQLite3 备忘录][3]
|
||
|
||
SQLite 专用于 [公共领域][4],[从技术上讲,这意味着它没有版权,因此不需要许可证][5]。如果你需要许可证,则可以 [购买所有权担保][6]。SQLite 非常常见,大约有 1 万亿个 SQLite 数据库正在使用中。在每个基于 Webkit 的 Web 浏览器,现代电视机,汽车多媒体系统以及无数其他软件应用程序中,Android 和 iOS 设备, macOS 和 Windows 10 计算机,大多数 Linux 系统上都包含多个这种数据库。
|
||
|
||
总而言之,它是用于存储和组织数据的一个可靠而简单的系统。
|
||
|
||
### 安装
|
||
|
||
你的系统上可能已经有 SQLite 库,但是你需要安装其命令行工具才能直接使用它。在 Linux上,你可能已经安装了这些工具。该工具提供的命令是 `sqlite3` (而不仅仅是 sqlite)。
|
||
|
||
如果没有在你的 Linux 或 BSD 上安装 SQLite,你则可以从软件仓库中或 ports 树中安装 SQLite,也可以从源代码或已编译的二进制文件进行[下载并安装][7]。
|
||
|
||
在 macOS 或 Windows 上,你可以从 [sqlite.org][7] 下载并安装 SQLite 工具。
|
||
|
||
### 使用 SQLite
|
||
|
||
通过编程语言与数据库进行交互是很常见的。因此,像 Java、Python、Lua、PHP、Ruby、C++ 以及其他编程语言都提供了 SQLite 的接口(或“绑定”)。但是,在使用这些库之前,了解数据库引擎的实际情况以及为什么你对数据库的选择很重要是有帮助的。本文向你介绍 SQLite 和 `sqlite3` 命令,以便你熟悉该数据库如何处理数据的基础知识。
|
||
|
||
### 与 SQLite 交互
|
||
|
||
你可以使用 `sqlite3` 命令与 SQLite 进行交互。 该命令提供了一个交互式的 shell 程序,以便你可以查看和更新数据库。
|
||
|
||
```
|
||
$ sqlite3
|
||
SQLite version 3.34.0 2020-12-01 16:14:00
|
||
Enter ".help" for usage hints.
|
||
Connected to a transient in-memory database.
|
||
Use ".open FILENAME" to reopen on a persistent database.
|
||
sqlite>
|
||
```
|
||
|
||
该命令将使你处于 SQLite 的子 shell 中,因此现在的提示符是 SQLite 的提示符。你以前使用的 Bash 命令在这里将不再适用。你必须使用 SQLite 命令。要查看 SQLite 命令列表,请输入 `.help`:
|
||
|
||
```
|
||
sqlite> .help
|
||
.archive ... Manage SQL archives
|
||
.auth ON|OFF SHOW authorizer callbacks
|
||
.backup ?DB? FILE Backup DB (DEFAULT "main") TO FILE
|
||
.bail ON|off Stop after hitting an error. DEFAULT OFF
|
||
.binary ON|off Turn BINARY output ON OR off. DEFAULT OFF
|
||
.cd DIRECTORY CHANGE the working directory TO DIRECTORY
|
||
[...]
|
||
```
|
||
|
||
这些命令中的其中一些是二进制的,而其他一些则需要唯一的参数(如文件名、路径等)。这些是 SQLite Shell 的管理命令,不是用于数据库查询。数据库以结构化查询语言(SQL)进行查询,许多 SQLite 查询与你从 [MySQL][8] 和 [MariaDB][9] 数据库中已经知道的查询相同。但是,数据类型和函数有所不同,因此,如果你熟悉另一个数据库,请特别注意细微的差异。
|
||
|
||
### 创建数据库
|
||
|
||
启动 SQLite 时,可以打开内存数据库,也可以选择要打开的数据库:
|
||
|
||
```
|
||
$ sqlite3 mydatabase.db
|
||
```
|
||
|
||
如果还没有数据库,则可以在 SQLite 提示符下创建一个数据库:
|
||
|
||
```
|
||
sqlite> .open mydatabase.db
|
||
```
|
||
|
||
现在,你的硬盘驱动器上有一个空文件,可以用作 SQLite 数据库。 文件扩展名 `.db` 是任意的。你也可以使用 `.sqlite` 或任何你想要的后缀。
|
||
|
||
### 创建一个表
|
||
|
||
数据库包含一些<ruby>表<rt>table</rt></ruby>,可以将其可视化为电子表格。有许多的行(在数据库中称为<ruby>记录<rt>record</rt></ruby>)和列。行和列的交集称为<ruby>字段<rt>field</rt></ruby>。
|
||
|
||
结构化查询语言(SQL)以其提供的内容而命名:一种以可预测且一致的语法查询数据库内容以接收有用的结果的方法。SQL 读起来很像普通的英语句子,即使有点机械化。当前,你的数据库是一个没有任何表的空数据库。
|
||
|
||
你可以使用 `CREATE` 来创建一个新表,你可以和 `IF NOT EXISTS` 结合使用。以便不会破坏现在已有的同名的表。
|
||
|
||
你无法在 SQLite 中创建一个没有任何字段的空表,因此在尝试 `CREATE` 语句之前,必须考虑预期表将存储的数据类型。在此示例中,我将使用以下列创建一个名为 `member` 的表:
|
||
|
||
* 唯一标识符
|
||
* 人名
|
||
* 记录创建的时间和日期
|
||
|
||
#### 唯一标识符
|
||
|
||
最好用唯一的编号来引用记录,幸运的是,SQLite 认识到这一点,创建一个名叫 `rowid` 的列来为你自动实现这一点。
|
||
|
||
无需 SQL 语句即可创建此字段。
|
||
|
||
#### 数据类型
|
||
|
||
对于我的示例表中,我正在创建一个 `name` 列来保存 `TEXT` 类型的数据。为了防止在没有指定字段数据的情况下创建记录,可以添加 `NOT NULL` 指令。
|
||
|
||
用 `name TEXT NOT NULL` 语句来创建。
|
||
|
||
SQLite 中有五种数据类型(实际上是 _储存类别_):
|
||
|
||
* `TEXT`:文本字符串
|
||
* `INTEGER`:一个数字
|
||
* `REAL`:一个浮点数(小数位数无限制)
|
||
* `BLOB`:二进制数据(例如,.jpeg 或 .webp 图像)
|
||
* `NULL`:空值
|
||
|
||
#### 日期和时间戳
|
||
|
||
SQLite 有一个方便的日期和时间戳功能。它本身不是数据类型,而是 SQLite 中的一个函数,它根据所需的格式生成字符串或整数。 在此示例中,我将其保留为默认值。
|
||
|
||
创建此字段的 SQL 语句是:`datestamp DATETIME DEFAULT CURRENT_TIMESTAMP`。
|
||
|
||
### 创建表的语句
|
||
|
||
在 SQLite 中创建此示例表的完整 SQL:
|
||
|
||
```
|
||
sqlite> CREATE TABLE
|
||
...> IF NOT EXISTS
|
||
...> member (name TEXT NOT NULL,
|
||
...> datestamp DATETIME DEFAULT CURRENT_TIMESTAMP);
|
||
```
|
||
|
||
在此代码示例中,我在语句的分句后按了回车键。以使其更易于阅读。除非以分号(`;`)终止,否则 SQLite 不会运行你的 SQL 语句。
|
||
|
||
你可以使用 SQLite 命令 `.tables` 验证表是否已创建:
|
||
|
||
```
|
||
sqlite> .tables
|
||
member
|
||
```
|
||
|
||
### 查看表中的所有列
|
||
|
||
你可以使用 `PRAGMA` 语句验证表包含哪些列和行:
|
||
|
||
```
|
||
sqlite> PRAGMA table_info(member);
|
||
0|name|TEXT|1||0
|
||
1|datestamp|DATETIME|0|CURRENT_TIMESTAMP|0
|
||
```
|
||
|
||
### 数据输入
|
||
|
||
你可以使用 `INSERT` 语句将一些示例数据填充到表中:
|
||
|
||
```
|
||
> INSERT INTO member (name) VALUES ('Alice');
|
||
> INSERT INTO member (name) VALUES ('Bob');
|
||
> INSERT INTO member (name) VALUES ('Carol');
|
||
> INSERT INTO member (name) VALUES ('David');
|
||
```
|
||
|
||
查看表中的数据:
|
||
|
||
```
|
||
> SELECT * FROM member;
|
||
Alice|2020-12-15 22:39:00
|
||
Bob|2020-12-15 22:39:02
|
||
Carol|2020-12-15 22:39:05
|
||
David|2020-12-15 22:39:07
|
||
```
|
||
|
||
#### 添加多行数据
|
||
|
||
现在创建第二个表:
|
||
|
||
```
|
||
> CREATE TABLE IF NOT EXISTS linux (
|
||
...> distro TEXT NOT NULL);
|
||
```
|
||
|
||
填充一些示例数据,这一次使用小的 `VALUES` 快捷方式,因此你可以在一个命令中添加多行。关键字 `VALUES` 期望以括号形式列出列表,而用多个逗号分隔多个列表:
|
||
|
||
```
|
||
> INSERT INTO linux (distro)
|
||
...> VALUES ('Slackware'), ('RHEL'),
|
||
...> ('Fedora'),('Debian');
|
||
```
|
||
|
||
### 修改表结构
|
||
|
||
你现在有两个表,但是到目前为止,两者之间没有任何关系。它们每个都包含独立的数据,但是可能你可能需要将第一个表的成员与第二个表中列出的特定项相关联。
|
||
|
||
为此,你可以为第一个表创建一个新列,该列对应于第二个表。由于两个表都设计有唯一标识符(这要归功于 SQLite 的自动创建),所以连接它们的最简单方法是将其中一个的 `rowid` 字段用作另一个的选择器。
|
||
|
||
在第一个表中创建一个新列,以存储第二个表中的值:
|
||
|
||
```
|
||
> ALTER TABLE member ADD os INT;
|
||
```
|
||
|
||
使用 `linux` 表中的唯一标识符作为 `member` 表中每一条记录中 `os` 字段的值。因为记录已经存在。因此你可以使用 `UPDATE` 语句而不是使用 `INSERT` 语句来更新数据。需要特别注意的是,你首先需要选中特定的一行来然后才能更新其中的某个字段。从句法上讲,这有点相反,更新首先发生,选择匹配最后发生:
|
||
|
||
```
|
||
> UPDATE member SET os=1 WHERE name='Alice';
|
||
```
|
||
|
||
对 `member` 表中的其他行重复相同的过程。更新 `os` 字段,为了数据多样性,在四行记录上分配三种不同的发行版(其中一种加倍)。
|
||
|
||
### 联接表
|
||
|
||
现在,这两个表相互关联,你可以使用 SQL 显示关联的数据。数据库中有多种 _联接方式_,但是一旦掌握了基础知识,就可以尝试所有的联接形式。这是一个基本联接,用于将 `member` 表的 `os` 字段中的值与 linux 表的 `rowid` 字段相关联:
|
||
|
||
```
|
||
> SELECT * FROM member INNER JOIN linux ON member.os=linux.rowid;
|
||
Alice|2020-12-15 22:39:00|1|Slackware
|
||
Bob|2020-12-15 22:39:02|3|Fedora
|
||
Carol|2020-12-15 22:39:05|3|Fedora
|
||
David|2020-12-15 22:39:07|4|Debian
|
||
```
|
||
|
||
`os` 和 `rowid` 字段形成了关联。
|
||
|
||
在一个图形应用程序中,你可以想象 `os` 字段是一个下拉选项菜单,其中的值是 `linux` 表中 `distro` 字段中的数据。将相关的数据集通过唯一的字段相关联,可以确保数据的一致性和有效性,并且借助 SQL,你可以在以后动态地关联它们。
|
||
|
||
### 了解更多
|
||
|
||
SQLite 是一个非常有用的自包含的、可移植的开源数据库。学习以交互方式使用它是迈向针对 Web 应用程序进行管理或通过编程语言库使用它的重要的第一步。
|
||
|
||
如果你喜欢 SQLite,也可以尝试由同一位作者 Richard Hipp 博士的 [Fossil][10]。
|
||
|
||
在学习和使用 SQLite 时,有一些常用命令可能会有所帮助,所以请立即下载我们的 [SQLite3 备忘单][3]!
|
||
|
||
--------------------------------------------------------------------------------
|
||
|
||
via: https://opensource.com/article/21/2/sqlite3-cheat-sheet
|
||
|
||
作者:[Klaatu][a]
|
||
选题:[lujun9972][b]
|
||
译者:[amwps290](https://github.com/amwps290)
|
||
校对:[wxy](https://github.com/wxy)
|
||
|
||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||
|
||
[a]: https://opensource.com/users/klaatu
|
||
[b]: https://github.com/lujun9972
|
||
[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/coverimage_cheat_sheet.png?itok=lYkNKieP "Cheat Sheet cover image"
|
||
[2]: https://www.redhat.com/sysadmin/yaml-beginners
|
||
[3]: https://opensource.com/downloads/sqlite-cheat-sheet
|
||
[4]: https://sqlite.org/copyright.html
|
||
[5]: https://directory.fsf.org/wiki/License:PublicDomain
|
||
[6]: https://www.sqlite.org/purchase/license?
|
||
[7]: https://www.sqlite.org/download.html
|
||
[8]: https://www.mysql.com/
|
||
[9]: https://mariadb.org/
|
||
[10]: https://opensource.com/article/20/11/fossil
|