hellodb.md 30.8 KB
Newer Older
雪洛's avatar
雪洛 已提交
1
## 基础概念@base
Q
qiang 已提交
2 3 4

`uniCloud`提供了一个 JSON 格式的文档型数据库。顾名思义,数据库中的每条记录都是一个 JSON 格式的文档。

5
它是 nosql 非关系型数据库,如果您之前熟悉 sql 关系型数据库,那么两者概念对应关系如下表:
Q
qiang 已提交
6

雪洛's avatar
雪洛 已提交
7 8 9 10 11
|关系型			|JSON 文档型												|
|:-				|:-															|
|数据库 database|数据库 database											|
|表 table		|集合 collection。但行业里也经常称之为“表”。无需特意区分	|
|行 row			|记录 record / doc											|
12 13
|字段 column / field	|字段 field													|
|使用sql语法操作|使用MongoDB语法或jql语法操作									|
Q
qiang 已提交
14

15 16 17 18
- 一个`uniCloud`服务空间,有且只有一个数据库;
- 一个数据库可以有多个表;
- 一个表可以有多个记录;
- 一个个记录可以有多个字段。
Q
qiang 已提交
19

20
例如,数据库中有一个表,名为user,存放用户信息。表user的数据内容如下:
Q
qiang 已提交
21 22 23 24 25 26 27 28

```json
{"name":"张三","tel":"13900000000"}
{"name":"李四","tel":"13911111111"}
```

上述数据中,每行数据表示一个用户的信息,被称之为“记录(record/doc)”。name和tel称之为“字段(field)”。而“13900000000”则是第一条记录的字段tel的值。

29
每行记录,都是一个完整的json文档,获取到记录后可以使用常规json方式操作。但表并非json文档,表是多个json文档的汇总,获取表需要使用专门的API。
Q
qiang 已提交
30 31 32

与关系型数据库的二维表格式不同,json文档数据库支持不同记录拥有不同的字段、支持多层嵌套数据。

33
仍然以user表举例,要在数据库中存储每个人的每次登录时间和登录ip,则变成如下:
Q
qiang 已提交
34 35 36

```json
{
雪洛's avatar
雪洛 已提交
37 38
	"name":"张三","tel":"13900000000",
	"login_log":[
Q
qiang 已提交
39 40 41 42 43 44 45
		{"login_date":1604186605445,"login_ip":"192.168.1.1"},
		{"login_date":1604186694137,"login_ip":"192.168.1.2"}
	]
}
{"name":"李四","tel":"13911111111"}
```

雪洛's avatar
雪洛 已提交
46
上述数据表示张三登录了2次,login_date里的值是时间戳(timestamp)格式,在数据库内timestamp就是一个数字类型的数据。而李四没有登录过。
Q
qiang 已提交
47 48 49

可以看出json文档数据库相对于关系型数据库的灵活,李四可以没有login_log字段,也可以有这个字段但登录次数记录与张三不同。

50
_此处仅为举例,实际业务中,登录日志单独存放在另一个表更合适_
Q
qiang 已提交
51

52
对于初学者,如果不了解数据库设计,可以参考[opendb](opendb.md),已经预置了大量常见的数据库设计。
Q
qiang 已提交
53 54 55 56 57 58 59

对于不熟悉传统数据库,但掌握json的js工程师而言,uniCloud的云数据库更亲切,没有传统数据库高昂的学习成本。

在uniCloud web控制台新建表时,在下面的模板中也可以选择各种`opendb`表模板,直接创建。

uniCloud同时支持阿里云和腾讯云,它们的数据库大体相同,有细微差异。阿里云的数据库是mongoDB4.0,腾讯云则使用自研的文档型数据库(兼容mongoDB 4.0版本)。uniCloud基本抹平了不同云厂商的差异,有差异的部分会在文档中单独标注。

60
## 创建第一个表
Q
qiang 已提交
61

62
1. 打开 uniCloud web控制台 [https://unicloud.dcloud.net.cn/](https://unicloud.dcloud.net.cn/)
Q
qiang 已提交
63

64
2. 创建或进入一个已存在的服务空间,选择 云数据库->云数据库,创建一个新表
Q
qiang 已提交
65

66
比如我们创建一个简历表,名为 `resume`。点击上方右侧的 创建 按钮即可。
Q
qiang 已提交
67

68
![](https://vkceyugu.cdn.bspapp.com/VKCEYUGU-f184e7c3-1912-41b2-b81f-435d1b37c7b4/e8ca747e-d39f-4da2-8a82-733f9d486cb0.jpg)
Q
qiang 已提交
69

70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202
新建表时,支持选择现成的 [opendb](opendb.md) 表模板,选择一个或多个模板表,可以点击右下方按钮创建。

创建表一共有3种方式:
1. 在web控制台创建
2. 在HBuilderX中,项目根目录/uniCloud/database点右键新建schema,上传时创建
3. 在代码中也可以创建表,但不推荐使用,[见下](?id=createCollection)

## 数据表的3个组成部分

每个数据表,包含3个部分:
- data:数据内容
- index:索引
- schema:数据表格式定义

在uniCloud的web控制台可以看到一个数据表的3部分内容。

### 数据内容@dbdata

data,就是存放的数据记录(record)。里面是一条一条的json文档。

record可以增删改查、排序统计。后续有API介绍。

可以先在 web控制台 为之前的 `resume` 表创建一条记录。

输入一个json
```json
{
    "name": "张三",
    "birth_year": 2000,
    "tel": "13900000000",
    "email": "zhangsan@zhangsan.com",
    "intro": "擅于学习,做事严谨"
}
```

![](https://vkceyugu.cdn.bspapp.com/VKCEYUGU-f184e7c3-1912-41b2-b81f-435d1b37c7b4/cf0ffb14-c81d-486f-a687-55f84bc99969.jpg)

创建一条新记录,是不管在web控制台创建,还是通过API创建,每条记录都会自带一个`_id`字段用以作为该记录的唯一标志。

`_id`字段是每个数据表默认自带且不可删除的字段。同时,它也是数据表的索引。

![](https://vkceyugu.cdn.bspapp.com/VKCEYUGU-f184e7c3-1912-41b2-b81f-435d1b37c7b4/7ed5185c-cfcd-461b-a178-5516e9937798.jpg)

阿里云使用的是标准的mongoDB,`_id`是自增的,后创建的记录的`_id`总是大于先生成的`_id`。传统数据库的自然数自增字段在多物理机的大型数据库下很难保持同步,大型数据库均使用`_id`这种长度较长、不会重复且仍然保持自增规律的方式。

**腾讯云使用的是兼容mongoDB的自研数据库,`_id`并非自增**

插入/导入数据时也可以自行指定`_id`而不使用自动生成的`_id`,这样可以很方便的将其他数据库的数据迁移到uniCloud云数据库。

### 数据库索引@dbindex

所谓索引,是指在数据表的众多字段中挑选一个或多个字段,让数据库引擎优先处理这些字段。

设置为索引的字段,在通过该字段查询(where)或排序(orderBy)时可以获得更快的查询速度。

但设置过多索引也不合适,会造成数据新增和删除变慢。

新建的表,默认只有一个索引`_id`

一个数据表可以有多个字段被设为索引。

索引分唯一型和非唯一型。

唯一型索引要求整个数据表多个记录的该字段的值不能重复。比如`_id`就是唯一型索引。

假使有2个人都叫“张三”,那么他们在user数据表里的区分就是依靠不同的`_id`来区分。

如果我们要根据name字段来查询,为了提升查询速度,此时可以把name字段设为非唯一索引。

索引内容较多,还有“组合索引”、“稀疏索引”、“地理位置索引”、“TTL索引”等概念。有单独的文档详细讲述索引,另见:[数据库索引](uniCloud/db-index.md)


**在web控制台添加上述索引**

![](https://bjetxgzv.cdn.bspapp.com/VKCEYUGU-dc-site/fca53140-1d91-11eb-880a-0db19f4f74bb.jpg)

**注意**
- 如果记录中已经存在多个记录某字段相同的情况,那么将该字段设为唯一型索引会失败。
- 如果已经设置某字段为唯一索引,在新增和修改记录时如果该字段的值之前在其他记录已存在,会失败。
- 假如记录中不存在某个字段,则对索引字段来说其值默认为 null,如果该索引字段设为唯一型索引,则不允许存在两个或以上的该字段为null或不存在该字段的记录。此时需要设置稀疏索引来解决多个null重复的问题


### 数据表格式定义@dbschema

`DB Schema`是表结构描述。描述数据表有哪些字段、值域类型是什么、是否必填、数据操作权限等很多内容。

因为 MongoDB 的灵活性,理论上`DB Schema`不是必须的,使用传统 MongoDB API 操作数据库不需要`DB Schema`

但如果使用 JQL,那`DB Schema`就是必须的。

`DB Schema`涉及内容较多,另见文档:[https://uniapp.dcloud.io/uniCloud/schema](uniCloud/schema)


## API操作数据库的方式@db-operation-type

uniCloud 的云数据库有多种操作方式。
- 支持在云函数操作,也支持在客户端操作。
- 支持使用传统MongoDB语法操作,也支持JQL语法操作。

uniCloud 默认推荐使用 JQL 语法操作数据库,它是一种更简单易用、对js开发者更友好的、开发效率更高的数据库操作语法。[详见](jql.md)

不管在云函数中还是在uni-app客户端中,均支持JQL。

同时 uniCloud 保留了在云函数中使用传统 MongoDB 的 nodejs API 操作云数据库。

|					|运行在云端	|运行在客户端	|支持 DB Schema	|
|--					|--			|--				|--				|
|JQL云函数			|√			|√				|√				|
|传统MongoDB客户端	|√			|X				|X				|

关于几种操作数据库方式的选择:
1. 优先使用客户端操作数据库(称为[clientDB](uniCloud/clientdb.md)
	
	传统开发中,其实大多数服务器接口开发,就是检验下前端数据和身份的合法性,然后写个SQL操作下数据库,返回JSON给前端。其实很无聊。
	
	clientDB最大的好处就是不用写服务端代码,客户端直接操作数据库。因为uniCloud提供了[DB Schema](schema.md)[uni-id](uni-id-summary.md),可以直接控制数据库的内容和权限校验。
	
	clientDB同时支持`action云函数`作为补充,当发起一个客户端操作云数据库的请求时,可以同时触发一个`action云函数`,在云端对数据库操作进行前置或后置处理。
	
	如下场景不适用clientDB:
	- 相关数据库操作逻辑不适合暴露在前端,比如抽奖
	- 注册、修改密码等操作password类型数据(password类型不会传给前端)的情况。一般情况下开发者不涉及相关问题。因为注册、修改密码等账户管理相关,官方已提供了[uni-id-pages](uni-id-pages.md),里面有uni-id-co云对象。所以开发者无需自己编写相关逻辑,直接用这个插件就好了。
	- 三方服务器回调。在登录、支付等涉及隐私的地方较常见。比如从微信服务器获取用户手机号,只支持云端获取,获取后要入库保存
	
2. 其次使用云函数的JQL扩展库来操作数据库
	
	对于clientDB不适应的场景,推荐在云函数或云对象中使用JQL。
	
	目前云函数中JQL不适用的场景:使用 set 操作符动态修改字段名称(不是字段值)。这个场景常规业务不涉及,未来JQL可能会完善并支持这个场景。
	
3. 除非开发者原本就对 nodejs 操作 MongoDB 非常熟悉,且需要使用 set 操作符动态修改字段名等,否则不推荐使用传统MongoDB写法。

	MongoDB API操作数据库,不能在客户端操作、不支持 DB Schema,不支持HBuilderX的jql查询器,不能在客户端的代码提示中提示数据库的表名、字段。
Q
qiang 已提交
203 204 205

不管使用哪种方法,都有很多公共的概念或功能。本文档将讲述这些公共的内容。

雪洛's avatar
雪洛 已提交
206
同时左侧导航有三种方法的专项链接,描述它们各自特有的功能。
207 208 209
- [客户端操作数据库clientDB](clientdb.md)
- [云函数内使用JQL语法操作数据库](jql-cloud.md)
- [云函数使用传统MongoDB语法操作数据库](cf-database.md)
Q
qiang 已提交
210

211
## 获取数据库对象的API@database
Q
qiang 已提交
212 213 214 215 216 217 218 219 220

想要通过代码操作数据库,第一步要获取服务空间里的数据库对象。

```js
const db = uniCloud.database(); //代码块为cdb
```

js中敲下代码块`cdb`,即可快速输入上述代码。

221
其中,云函数内使用JQL扩展库时,还需要做一个工作,就是指定操作用户身份。[详见](jql-cloud.md?id=use-in-object)
Q
qiang 已提交
222

雪洛's avatar
雪洛 已提交
223
```js
224
// 云函数中JQL使用示例
雪洛's avatar
雪洛 已提交
225 226 227 228 229 230 231
'use strict';
exports.main = async (event, context) => {
	const dbJQL = uniCloud.databaseForJQL({ // 获取JQL database引用,此处需要传入云函数的event和context,必传
		event,
		context 
	})
	return {
232
		dbJQL.collection('book').get() // 直接执行数据库操作
雪洛's avatar
雪洛 已提交
233 234 235 236
	}
};
```

237
### 获取其他服务空间数据库实例@init-db
雪洛's avatar
雪洛 已提交
238

239
如果当前应用仅使用一个服务空间,在HBuilderX中做好服务空间关联即可。获取当前空间的数据库实例时无需传递配置,直接调用database方法默认就是操作关联服务空间的数据库。
雪洛's avatar
雪洛 已提交
240 241

如果应用有连接其他服务空间数据库的需求,可以在获取database实例时传递对应服务空间的配置
Q
qiang 已提交
242

雪洛's avatar
雪洛 已提交
243
> HBuilderX 3.2.11及更高版本支持客户端初始化其他服务空间database实例,此前仅腾讯云云函数环境支持。阿里云云函数环境不支持此用法。
雪洛's avatar
雪洛 已提交
244

雪洛's avatar
雪洛 已提交
245 246
调用`uniCloud.database()`时可以传入对应的服务空间信息(参数同uniCloud.init,参考:[uniCloud.init](uniCloud/init.md?id=init-unicloud))来获取指定服务空间的database实例。

雪洛's avatar
雪洛 已提交
247 248 249 250 251
**注意**

- 云函数环境(仅腾讯云支持)仅能通过init返回同账号下其他的腾讯云服务空间的数据库实例。
- 客户端环境(腾讯云阿里云均支持)可以通过init返回本账号下任意云厂商服务空间的数据库实例

雪洛's avatar
雪洛 已提交
252
**示例**
Q
qiang 已提交
253 254 255

```js
const db = uniCloud.database({
雪洛's avatar
雪洛 已提交
256 257 258 259 260
  provider: 'tencent',
  spaceId: 'xxx'
})

db.collection('uni-id-users').get()
Q
qiang 已提交
261 262
```

雪洛's avatar
雪洛 已提交
263 264 265 266 267 268 269
**参数说明**

|参数名			|类型	|必填	|默认值						|说明																					|
|:-:			|:-:	|:-:	|:-:						|:-:																					|
|provider		|String	|是		|-							|aliyun、tencent																		|
|spaceId		|String	|是		|-							|服务空间ID,**注意是服务空间ID,不是服务空间名称**										|
|clientSecret	|String	|是		|-							|仅阿里云支持,可以在[uniCloud控制台](https://unicloud.dcloud.net.cn)服务空间列表中查看	|
270
|endpoint		|String	|否		|`https://api.bspapp.com`	|服务空间地址,仅阿里云支持															|
Q
qiang 已提交
271
 
272
## 创建集合/表的API@createCollection
Q
qiang 已提交
273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292

- 阿里云

调用add方法,给某数据表新增数据记录时,如果该数据表不存在,会自动创建该数据表。如下代码给table1数据表新增了一条数据,如果table1不存在,会自动创建。

```js
const db = uniCloud.database();
db.collection("table1").add({name: 'Ben'})
```

- 腾讯云

腾讯云提供了专门的创建数据表的API,此API仅支持云函数内运行,不支持clientDB调用。

```js
const db = uniCloud.database();
db.createCollection("table1")
```

**注意**
雪洛's avatar
雪洛 已提交
293 294 295 296

- 如果数据表已存在,腾讯云调用createCollection方法会报错
- 腾讯云调用collection的add方法不会自动创建数据表,不存在的数据表会报错
- 阿里云没有createCollection方法
297
- **使用代码方式创建的表没有索引、schema,性能和功能都受影响,不建议使用这种方式**
雪洛's avatar
雪洛 已提交
298 299 300 301 302 303 304 305

## 获取集合/数据表对象@collection

创建好数据表后,可以通过API获取数据表对象。

```js
const db = uniCloud.database();
// 获取名为 `table1` 数据表的引用
306
const resume = db.collection('resume');
雪洛's avatar
雪洛 已提交
307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326
```

**集合/数据表 Collection 的方法**

通过 `db.collection(name)` 可以获取指定数据表的引用,在数据表上可以进行以下操作

| 类型		| 接口		| 说明																														|
| --------	| -------	| ----------------------------------------------------------------------------------										|
| 写		| add		| 新增记录(触发请求)																										|
| 计数		| count		| 获取符合条件的记录条数																									|
| 读		| get		| 获取数据表中的记录,如果有使用 where 语句定义查询条件,则会返回匹配结果集 (触发请求)										|
| 引用		| doc		| 获取对该数据表中指定 id 的记录的引用																						|
| 查询条件	| where		| 通过指定条件筛选出匹配的记录,可搭配查询指令(eq, gt, in, ...)使用														|
|			| skip		| 跳过指定数量的文档,常用于分页,传入 offset。clientDB组件有封装好的更易用的分页,[另见](uniCloud/uni-clientdb-component)	|
|			| orderBy	| 排序方式																													|
|			| limit		| 返回的结果集(文档数量)的限制,有默认值和上限值																			|
|			| field		| 指定需要返回的字段																										|

collection对象的方法可以增和查数据,删和改不能直接操作,需要collection对象通过doc或get得到指定的记录后再调用remove或update方法进行删改。

327 328 329
数据操作的API较多,请单独参考文档:
- [JQL语法操作数据库](jql.md)
- [MongoDB语法操作数据库](cf-database.md)
雪洛's avatar
雪洛 已提交
330

331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364
**示例**

这里我们使用clientDB,实现一个简单的在前端获取刚才输入 resume 表的记录的功能。
1. 创建一个uni-app项目,开通uniCloud环境,并在uniCloud初始化向导中关联 resume表所在的服务空间;或者使用老项目对uniCloud目录点右键关联服务空间。
2. 对项目下的 uniCloud/database 点右键,下载所有DB Schema。
3. 打开新下载的 resume.schema.json ,将其中的 "permission" 节点下的 "read" 从 false 改为 true。即设置该表允许任意用户读 (后续可以深入学习 DB Schema)
4. 在客户端 pages/index/index.vue 编写代码
```html
<template>
	<view class="content">
		<button @click="testclientdb()">请求数据库</button>
	</view>
</template>
<script>
	export default {
		data() {
			return {}
		},
		methods: {
			testclientdb() {
				const db = uniCloud.database();
				db.collection("resume").get().then((res) => {
					// res 为数据库查询结果
					console.log(res)
				}).catch((e) => {
					console.log(e)
				});
			}
		}
	}
</script>
```
5. 运行项目,点击按钮,即打印出查询到的数据库内容
	`res.result.data`下即为数据表 resume 中的数据。
雪洛's avatar
雪洛 已提交
365

366
## 数据类型@data-type
雪洛's avatar
雪洛 已提交
367

368 369 370 371 372 373 374 375 376 377 378 379 380
数据库内数据类型有以下几种:

* string:字符串
	+ password:[DB Schema]中扩展的特殊string。用于保存密码。这类字段不会通过clientDB传递给前端,所有用户都不能通过clientDB读写,即使是admin管理员
* number:数字
	+ 在[DB Schema]中细化为int和double
* bool:布尔值
* date:时间
* timestamp 时间戳
* object:对象
	+ file:[DB Schema]中扩展的特殊 object,用于云存储文件的信息体。不直接存储文件,而是一个json object,包括云存储文件的名称、路径、文件体积等信息。(HBuilderX 3.1.0+ )
* array:数组
* null:相当于一个占位符,表示一个字段存在但是值为空。
雪洛's avatar
雪洛 已提交
381 382 383 384 385 386 387
* GeoPoint:地理位置点
* GeoLineStringLine: 地理路径
* GeoPolygon: 地理多边形
* GeoMultiPoint: 多个地理位置点
* GeoMultiLineString: 多个地理路径
* GeoMultiPolygon: 多个地理多边形

388
DB Schema的数据类型有专门文档,[详见](schema.md?id=bsontype)
雪洛's avatar
雪洛 已提交
389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503

### Date类型

Date 类型用于表示时间,精确到毫秒,可以用 JavaScript 内置 Date 对象创建。需要特别注意的是,连接本地云函数时,用此方法创建的时间是客户端当前时间,不是服务端当前时间,只有连接云端云函数才是服务端当前时间。

另外,我们还单独提供了一个 API 来创建服务端当前时间,使用 serverDate 对象来创建一个服务端当前时间的标记,**该对象暂时只支持腾讯云空间**,当使用了 serverDate 对象的请求抵达服务端处理时,该字段会被转换成服务端当前的时间,更棒的是,我们在构造 serverDate 对象时还可通过传入一个有 offset 字段的对象来标记一个与当前服务端时间偏移 offset 毫秒的时间,这样我们就可以达到比如如下效果:指定一个字段为服务端时间往后一个小时。

```js
// 服务端当前时间
new db.serverDate()
// 在云函数内使用new Date()和new db.serverDate()效果一样
```

```js
//服务端当前时间加1S
new db.serverDate({
  offset: 1000
})
// 在云函数内使用new Date(Date.now() + 1000)和上面的用法效果一样
```

### 地理位置类型@geo-data-type

#### Point@geo-point

用于表示地理位置点,用经纬度唯一标记一个点,这是一个特殊的数据存储类型。

签名:`Point(longitude: number, latitude: number)`

示例:
```js
new db.Geo.Point(longitude, latitude)
```

#### LineString@geo-line-string

用于表示地理路径,是由两个或者更多的 `Point` 组成的线段。

签名:`LineString(points: Point[])`

示例:

```js
new db.Geo.LineString([
  new db.Geo.Point(lngA, latA),
  new db.Geo.Point(lngB, latB),
  // ...
])
```

#### Polygon@geo-polygon

用于表示地理上的一个多边形(有洞或无洞均可),它是由一个或多个**闭环** `LineString` 组成的几何图形。

由一个环组成的 `Polygon` 是没有洞的多边形,由多个环组成的是有洞的多边形。对由多个环(`LineString`)组成的多边形(`Polygon`),第一个环是外环,所有其他环是内环(洞)。

签名:`Polygon(lines: LineString[])`

示例:

```js
new db.Geo.Polygon([
  new db.Geo.LineString(...),
  new db.Geo.LineString(...),
  // ...
])
```

#### MultiPoint@geo-multi-point

用于表示多个点 `Point` 的集合。

签名:`MultiPoint(points: Point[])`

示例:

```js
new db.Geo.MultiPoint([
  new db.Geo.Point(lngA, latA),
  new db.Geo.Point(lngB, latB),
  // ...
])
```

#### MultiLineString@geo-multi-line-string

用于表示多个地理路径 `LineString` 的集合。

签名:`MultiLineString(lines: LineString[])`

示例:

```js
new db.Geo.MultiLineString([
  new db.Geo.LineString(...),
  new db.Geo.LineString(...),
  // ...
])
```

#### MultiPolygon@geo-multi-polygon

用于表示多个地理多边形 `Polygon` 的集合。

签名:`MultiPolygon(polygons: Polygon[])`

示例:

```js
new db.Geo.MultiPolygon([
  new db.Geo.Polygon(...),
  new db.Geo.Polygon(...),
  // ...
])
```
Q
qiang 已提交
504 505


雪洛's avatar
雪洛 已提交
506
## 与传统开发区别@difference
Q
qiang 已提交
507

508
不同于传统开发,连接数据库有单次操作时长限制,目前单次操作时间限制如下。超出此时间会报超时错误。一般情况下在设置了合适的索引时不会遇到超时错误,如何优化查询速度请参考:[数据库性能优化](uniCloud/db-performance.md)
Q
qiang 已提交
509

雪洛's avatar
雪洛 已提交
510
|腾讯云	|阿里云	|
DCloud_JSON's avatar
DCloud_JSON 已提交
511
|--	|--	|
512
|5秒	|3秒	|
Q
qiang 已提交
513

DCloud_JSON's avatar
DCloud_JSON 已提交
514 515 516
如果是大数据批处理,可以参考云函数递归调用,连续执行多个云函数处理一个任务[详情查看](https://uniapp.dcloud.net.cn/uniCloud/cf-functions.html#recurrence)


Q
qiang 已提交
517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600
## 数据导入导出和备份@dbmigration

uniCloud数据库提供了多种数据导入导出和备份方案。

- db\_init.json:常用于插件市场的插件做环境初始化。完整支持数据、索引、schema三部分。不适合处理大量数据,操作可能超时
- 数据库回档备份和恢复。仅腾讯云支持。支持数据和索引,不支持schema
- 数据库导入导出。仅阿里云支持,适用于大数据量操作。仅支持数据,不支持索引和schema

除上述三种方法外,开发者还可以编程处理数据的导入导出。如进行大量数据操作,建议在HBuilderX的本地运行云函数环境中操作,这样可以避免触发云端的云函数超时限制。

下面对三种方法的使用方式进行详细说明:

### `db_init.json`初始化数据库@db-init

`db_init.json`定义了一个json格式,里面包含了表名、表数据、表索引等表的相关数据。

在HBuilderX中,项目的cloudfunctions目录(HBuilderX 2.5.11 - 2.9.11版本) 或 uniCloud/database 目录(HBuilderX 3.0+版本),可以放置`db_init.json`文件,对该文件点右键,可以按`db_init.json`的描述,在云服务空间创建相应的表、初始化表中的数据、索引和schema。

这个功能尤其适合插件作者,可以快速初始化插件所需的数据库环境。

**`db_init.json`的数据格式**

`db_init.json`包含三部分:数据内容(data)、数据表索引(index)、数据表结构(schema),形式如下

**注意:HBuilderX 3.0.0以上版本schema不再放在db_init.json内,而是独立放在uniCloud/database/目录下。**

详细调整如下:

- db_init.json位置由`cloudfunctions/db_init.json`移至`uniCloud/database/db_init.json`
- schema不再放在db_init.json内,每个表都有一个单独的schema文件,比如news表对应的schema为`uniCloud/database/news.schema.json`
- schema可以在`uniCloud/database`目录上右键创建
- `db_init.json`文件右键初始化云数据库时依然会带上schema进行数据库的初始化,除schema外HBuilderX3.0.0以上版本使用db_init.json初始化数据库还会带上扩展校验函数,扩展校验函数位于`uniCloud/database/validateFunction`目录下,扩展校验函数文档详见:[validateFunction](https://uniapp.dcloud.net.cn/uniCloud/schema?id=validatefunction)

**HBuilderX 3.0.0版本之前的db_init.json示例**

```json
{
  "collection_test": { // 集合(表名)
    "data": [ // 数据
      {
        "_id": "da51bd8c5e37ac14099ea43a2505a1a5", // 一般不带_id字段,防止导入时数据冲突。
        "name": "tom"
      }
    ],
    "index": [{ // 索引
      "IndexName": "index_a", // 索引名称
      "MgoKeySchema": { // 索引规则
        "MgoIndexKeys": [{
          "Name": "index", // 索引字段
          "Direction": "1" // 索引方向,1:ASC-升序,-1:DESC-降序,2dsphere:地理位置
        }],
        "MgoIsUnique": false, // 索引是否唯一
        "MgoIsSparse": false // 是否为稀疏索引,请参考 https://uniapp.dcloud.net.cn/uniCloud/db-index.md?id=sparse
      }
    }],
    "schema": { // HBuilderX 3.0.0以上版本schema不在此处,而是放在database目录下单独的`表名.schema.json`文件内
      "bsonType": "object",
      "permission": {
        ".read": true,
        ".create": false,
        ".update": false,
        ".delete": false
      },
      "required": [
        "image_url"
      ],
      "properties": {
        "_id": {
          "description": "ID,系统自动生成"
        },
        "image_url": {
          "bsonType": "string",
          "description": "可以是在线地址,也支持本地地址",
          "label": "图片url"
        }
      }
    }
  }
}

```

在HBuilderX中对上述`db_init.json`点右键,可初始化数据库到云服务空间,创建`collection_test`表,并按上述json配置设置该表的index索引和schema,以及插入data下的数据。

601
[opendb](opendb.md)的表,在`db_init.json`中初始化时,不建议自定义index和schema。系统会自动从opendb规范中读取最新的index和schema。
Q
qiang 已提交
602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631

**使用`db_init.json`导入数据库**

在HBuilderX中,对项目下的cloudfunctions目录下的`db_init.json`点右键,即可选择`初始化云数据库`。将`db_init.json`里的内容导入云端。

注意事项:
- 目前`db_init.json`为同步导入形式,无法导入大量数据。导入大量数据请使用web控制台的数据库的导入功能。
- 如果`db_init.json`中的表名与opendb中任意表名相同,且`db_init.json`中该表名内没有编写schema和index,则在初始化时会自动拉取最新的opendb规范内对应表的schema和index。
- 如果`db_init.json`中的数据表在服务空间已存在,且`db_init.json`中该表含有schema和index,则在初始化时schema会被替换,新增索引会被添加,已存在索引不受影响。

**生成`db_init.json`的方式**

在uniCloud web控制台的数据库界面,左侧导航点击 生成`db_init.json`,会将选择的表的内容、索引、表结构导出为`db_init.json`文件。

注意事项:
- 如果表名与opendb中任意表名相同,web控制台导出时将不会带上schema和index。
- web控制台导出时默认不包括`_id`字段,在导入时,数据库插入新记录时会自动补`_id`字段。如果需要指定`_id`,需要手工补足数据。

在db_init.json内可以使用以下形式定义Date类型的数据:

```js
{
  "dateObj": { // dateObj字段就是日期类型的数据
    "$date": "2020-12-12T00:00:00.000Z" // ISO标准日期字符串
  }
}
```

### 数据库回档备份和恢复@backup

Q
qiang 已提交
632
uniCloud会在每天凌晨自动备份一次数据库,最多保留7天。这让开发者不再担心数据丢失。
Q
qiang 已提交
633 634 635 636 637 638

**操作说明**

1. 登录[uniCloud后台](https://unicloud-dev.dcloud.net.cn/)
2. 点击左侧菜单`云数据库 --> 数据库回档`,点击`新建回档`
3. 选择可回档时间
639
4. 选择需要回档的表(注意:回档后表不能与现有表重名,如需对表重命名可以在表列表处操作)
Q
qiang 已提交
640 641 642 643 644 645

![数据库回档](https://img.cdn.aliyun.dcloud.net.cn/uni-app/uniCloud/unicloud-db-backup.jpg)


### 数据导出为文件@export

646
此功能主要用于导出整个表的数据
Q
qiang 已提交
647 648 649 650

**用法**

1. 进入[uniCloud web控制台](https://unicloud.dcloud.net.cn/home),选择服务空间,或者直接在HBuilderX云函数目录`cloudfunctions`上右键打开uniCloud web控制台
651
2. 进入云数据库选择希望导入数据的表
Q
qiang 已提交
652 653 654 655 656 657 658
3. 点击导出按钮
4. 选择导出格式,如果选择csv格式还需要选择导出字段
5. 点击确定按钮等待下载开始即可

**注意**

- 导出的json文件并非一般情况下的json,而是每行一条json数据的文本文件
雪洛's avatar
雪洛 已提交
659
- 导出为csv格式时必须填写字段选项。字段之间使用英文逗号隔开。例如:`_id, name, age, gender`
雪洛's avatar
雪洛 已提交
660
- 导出为csv格式会丢失数据类型,如需迁移数据库请导出为json格式
Q
qiang 已提交
661 662 663 664 665 666 667 668 669
- 数据量较大时可能需要等待一段时间才可以开始下载

### 从文件导入数据@import

uniCloud提供的`db_init.json`主要是为了对数据库进行初始化,并不适合导入大量数据。与`db_init.json`不同,数据导入功能可以导入大量数据,目前支持导入 CSV、JSON 格式(关于json格式看下面注意事项)的文件数据。

**用法**

1. 进入[uniCloud web控制台](https://unicloud.dcloud.net.cn/home),选择服务空间,或者直接在HBuilderX云函数目录`cloudfunctions`上右键打开uniCloud web控制台
670
2. 进入云数据库选择希望导入数据的表
Q
qiang 已提交
671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686
3. 点击导入,选择json文件或csv文件
4. 选择处理冲突模式(关于处理冲突模式请看下方注意事项)
5. 点击确定按钮等待导入完成即可

**注意**

- 目前导入文件最大限制为50MB
- 导入导出文件无法保留索引和schema
- 导入导出csv时数据类型会丢失,即所有字段均会作为字符串导入
- 冲突处理模式为设定记录_id冲突时的处理方式,`insert`表示冲突时依旧导入记录但是是新插入一条,`upsert`表示冲突时更新已存在的记录
- 这里说的json文件并不是标准的json格式,而是形如下面这样每行一个json格式的数据库记录的文件
  ```json
  {"a":1}
  {"a":2}
  ```

687
> 如果是自己拼接的json格式数据请注意:如果存在表A关联表B的字段的场景需要保证关联字段在A、B内是一致的(特别需要注意的是各种与_id关联的字段)
Q
qiang 已提交
688 689 690 691 692 693 694

例:

**正确示例**

```js
// 这里为了方便看数据进行了格式化,实际导入所需的json文件是每行一条记录
695
// article表
Q
qiang 已提交
696 697 698 699 700
{
  "user_id": {
    $oid: "601cf1dbf194b200018ed8ec"
  }
}
701
// user表
Q
qiang 已提交
702 703 704 705 706 707 708 709 710 711 712
{
  "_id": {
    $oid: "601cf1dbf194b200018ed8ec"
  }
}
```

**错误示例**

```js
// 这里为了方便看数据进行了格式化,实际导入所需的json文件是每行一条记录
713
// article表
Q
qiang 已提交
714 715 716
{
  "user_id": "601cf1dbf194b200018ed8ec"
}
717
// user表
Q
qiang 已提交
718 719 720 721 722
{
  "_id": {
    $oid: "601cf1dbf194b200018ed8ec"
  }
}
Q
qiang 已提交
723 724 725 726
```

### 云厂商之间的迁移@cross-provider

雪洛's avatar
雪洛 已提交
727
文档移至:[在云厂商之间迁移数据库](uniCloud/price.md?id=cross-provider-db)