目录
云函数
分类
客户端和云函数的通信
clientDB方式
云对象方式
云对象
云对象的Api
内置特殊方法
预处理 _before
后处理 _after
定时运行 _timing
云函数
概念
云函数是运行在云端的javascript代码,是基于Node.js的扩展
在Node API基础上,云函数环境内置了uniCloud对象,这个对象内置了网络、数据库等各种API。
每个云函数是一个js包,在云函数被调用时,由serverless调度系统分配硬件资源启动一个node环境来运行这个云函数。
每个云函数是一个目录,其中普通云函数有index.js入口文件,云对象的入口文件则是index.obj.js
注意事项
- 云函数内使用commonJs规范,不可使用import、export
- 不同项目使用同一个服务空间时,不可使用同名云函数,同名云函数会相互覆盖
- 阿里云版服务器,暂不支持使用相对路径读取文件,比如fs.readFileSync('./info.txt'),可以使用绝对路径fs.readFileSync(path.resolve(__dirname, './info.txt'))
分类
云函数有若干子概念,包括 普通云函数、云对象、公共模块、clientDB的action云函数、uniCloud扩展库
- 普通云函数——通过传统json接口方式和客户端通信,客户端使用uniCloud.callfunction("")调用云函数
- 云对象——通过前端导入对象来操作的,客户端使用uniCloud.importObject("")导入云对象
- 公共模块:用于不同的云函数/云对象,抽取和共享相同代码
- action云函数——为了弥补clientDB客户端直接操作数据库的局限而设计的。
- uncCloud扩展库——为了裁剪和控制云函数体积而设计的,避免增大每个云函数的体积。
|——— cloudfunctions 云函数目录 | │───common 云函数公用模块目录 详情 | | └──hello-common 云函数公用模块 | | │──index.js 公用模块代码 | | └──package.json 公用模块package.json | │───uni-clientDB-actions | │ └──new_action.js clientDB action代码 详情 | │───function-name 云函数目录 | │ │──index.js 云函数代码 | │ └──package.json 包含云函数的配置信息,如url化、定时设置、可用内存等内容 详情 | └───object-name 云对象目录 | │──index.obj.js 云对象代码 | └──package.json 包含云对象的配置信息,可用内存等内容 详情
客户端和云函数的通信
云函数是uniCloud的基础,本质上clientDB和云对象都是建立在云函数上针对特定场景的优化。
- clientDB针对的场景是数据库操作,它优化了可以不写或少写服务器代码。
- 云对象针对的场景是非数据库操作或不宜前端暴露的数据库操作时,和uni-app客户端的通信方式。它优化了代码结构,更精简、简单
clientDB方式
clientDB分API方式和组件方式
//API方式 客户端js直接操作云数据库,查询list表的数据。无需服务器代码
const db = uniCloud.database() // 获取云数据库的引用
db.collection('list').get()
.then((res)=>{
// res 为数据库查询结果
}).catch((err)=>{
console.log(err);
})
clientDB适用情况:
如果客户端使用uni-app开发,且向uniCloud服务空间的请求主要是为了操作云数据库(增删改查),那么推荐clientDB方式,由uni-app客户端直接操作云数据库。
如果操作数据库的同时,还需要同时执行一些云函数,可以使用clientDB的action云函数。
clientDB不适用的情况:
- 请求不操作云数据库,比如向外部web系统发送请求,操作redis、删除云文件等
- 操作的云数据库请求不希望暴露在前端
- 数据库表和字段数量多而接口数量少。给每个数据配置权限的工作量超过了控制少数接口权限的工作量
- 权限体系较复杂,除了用户和管理员外还有较多其他权限条件或动态权限。此时在schema和action中编写代码复杂度超过了写接口。
云对象方式
云对象和clientDB最大的区别,是云对象把数据库操作(以及其他逻辑)封装在云对象的方法里面。仍需在客户端和云端分别写代码,但应用场景不受限制,clientDB不适应的它都适应。
// 云端云对象代码,云对象名称:testco,有一个sum方法
module.exports = {
sum(a, b) {
// 此处省略a和b的有效性校验
return a + b
}
}
//然后在客户端的js中,import这个testco对象,调用它的sum方法
const testco = uniCloud.importObject('testco') //第一步导入云对象
async function sum () { //注意方法或生命周期需使用async异步方式
try {
const res = await testco.sum(1,2) //导入云对象后就可以直接调用该对象的方法了,注意使用异步await
console.log(res) // 结果是3
} catch (e) {
console.log(e)
}
}
云对象
背景:20年前,restful接口开发流行,服务器编写接口,客户端调用接口,传输json
现在:云对象,服务器编写API,客户端调用API,不再开发传输Json的接口,思路更清晰,代码更精简。
比如:服务端编写一个云对象todo,该对象有add、get、remove、update等方法。客户端的js则可以直接import这个todo云对象,直接调用add等方法
// 云对象名:todo
module.exports = {
add(title, content) {
title = title.trim()
content = content.trim()
if(!title || !content) {
return {
errCode: 'INVALID_TODO',
errMsg: 'TODO标题或内容不可为空'
}
}
// ...其他逻辑
return {
errCode: 0,
errMsg: '创建成功'
}
}
}
// 然后在客户端的js中,import这个todo对象,调用它的add方法
const todo = uniCloud.importObject('todo') //第一步导入云对象
async function addTodo () {
try {
const res = await todo.add('title demo', 'content demo') //导入云对象后就可以直接调用该对象的方法了,注意使用异步await
uni.showToast({
title: '创建成功'
})
} catch (e) {
// 符合uniCloud响应体规范 https://uniapp.dcloud.net.cn/uniCloud/cf-functions?id=resformat,自动抛出此错误
uni.showModal({
title: '创建失败',
content: e.errMsg,
showCancel: false
})
}
}
云对象的Api
内置特殊方法
所有_
开头的方法都是私有方法,客户端不可访问。也就是客户端调用云对象时不能调用_开头的方法
- _before
- _after
- _timing
预处理 _before
用于调用常规方法之前进行预处理,一般用于拦截器,统一的身份验证,参数校验等。
//疑问: this.getUniIdToken() 为null 不显示错误提示
module.exports = {
_before: function(){
const methodName = this.getMethodName()
if(methodName === 'add' && !this.getUniIdToken()) {
throw new Error('token不存在')
}
},
add: function(title = '', content = '') {
return {
errCode: 0,
errMsg: '创建成功'
}
}
}
后处理 _after
云对象内可以创建一个特殊的方法_after
用来再加工处理本次调用方法的返回结果或者抛出的错误
// todo/index.obj.js 没看出效果 T.T
module.exports = {
_before: function(){
this.startTime = Date.now() // 在before内记录开始时间并在this上挂载,以供后续流程使用
},
add: function(title = '', content = '') {
if(title === 'abc') {
throw new Error('abc不是一个合法的todo标题')
}
return {
errCode: 0,
errMsg: '创建成功'
}
},
_after(error, result) {
if(error) {
throw error // 如果方法抛出错误,也直接抛出不处理
}
result.timeCost = Date.now() - this.startTime
return result
}
}