快速入门nodejs-前端必会

介绍

写后台管理程序,与之类似php .net java

目标

数据服务,文件服务,web服务

优势

性能高,方便、入门难度低、大公司都在用(BAT)

劣势

  • 服务器提供的相对较少
  • 相对其他语言,能用的上的学习资料少
  • 对程序员的要求高了

环境安装

官网:http://nodejs.cn/

测试环境: win+r->命令行(运行->cmd)->node -v

版本

Vx(主).x(子).x(修正)

主版本: 变化了,1/3的API发生巨变 , 使用方式变化了

子版本: API没有删减,使用方式没变化,内部实现发生了变化

修正版:什么都没变,处理一下bug

V6.8.0 稳定

V6.9.1 非稳定版

beta 测试

rc 、alpha测试稳定

node命令行

node 回车

运行

window
a. 找到目标目录-》地址栏输入cmd-》node 文件名.js | node 文件名

b. 当前目录->右键->git bash-> node 文件名
苹果
终端->cd 目录-> node 文件名.js | node 文件名
vscode
新建终端->cd 目录->node 文件名.js | node 文件名 
调试->运行
webstrom
terminal| run

开发注意

nodejs 使用的是ECMA语法,不可使用DOMBOM

web服务器

构成

  • 机器: 电脑
  • 数据库:mysql | sqlserver | mongoDB | oracle
    • 数据库存的是: 数字|字符
    • 磁盘(硬盘) 文件本身(图,视频,PDF) 文件服务器
  • 管理程序:nodejs(管理前后端工程文件)

前后端交互流程

大后端

​ 用户 - > 地址栏(http[s]请求) -> web服务器(收到) - > nodejs处理请求(返回静态、动态)->请求数据库服务(返回结果)->nodejs(接收)->node渲染页面->浏览器(接收页面,完成最终渲染)

大前端

​ 用户 - > http[s]请求 -> web服务器(收到) - > nodejs处理请求(返回静态、动态)->请求数据库服务(返回结果)->nodejs(接收)->返回给前端(渲染)->浏览器(接收页面,完成最终渲染)

实现

引入http模块

let http = require('http')

创建web服务 返回http对象

let app = http.createServer((req,res)=>{
    req 请求体  浏览器->服务器
    req.url  地址   提取地址栏数据
    req.on('data',()=>{}) 提取非地址栏数据 所有的http[s]都会触发end事件 这个可能一次抓不完,写一个变量累加,抓完后,要写end在end里面进行后续操作
    req.on('end',()=>{}) 
    
    res 响应  服务器->浏览器
    res.writeHead(200,{'Content-Type':'text/html;charset=utf-8'});响应头设置
    res.write(字符/数据<string><buffer>) 返回数据
    res.end() 结束响应 必须
})

监听服务器

app.listen(端口,[地址],[回调])

监听成功,回调一次

端口: 1-65535 1024以下系统占用

虚拟地址localhost 真实域名xx.duapp.com

更新后,需要每次服务器自动重启

推荐命令行工具:supervisor nodemon

安装方式: npm install supervisor -g

fs模块

磁盘操作,文件操作

读取

fs.readFile('文件路径',[编码方式],(err,data)=>{})

[^err ]: err 错误 ,null没有错误 //读取失败

编码方式可以设置成utf-8 如果不设置会自动成为buffer流

变量 = fs.readFileSync('文件路径') 

处理错误

try{要排错的代码}catch(e){}

try{

//要测试代码错误代码

let data = fs.readFileSync('./html/index123.html');

console.log('data',data)

}catch(e){//错误事件对象 error / ev / e

//处理错误,包装后续代码正常

console.log('e',e)

}

更名

fs.renameSync('改前','改后',回调可以捕获错误);
fs.rename('改前','改后',(err)=>{})

删除

fs.unlinkSync('文件路径')

静态资源托管

什么是静态资源

xx.css xx.html xx.js xx.图片 xx.json xx.字体 ...

前端资源请求

<a href=".."></a>
<img src="..."/>
location.href="..."
body{
    background:url(....)
}

后端资源读取

fs.readFile(文件名,[编码方式],回调(err,data));

接口实现

前端

表单:get/post/put/delete/...

js: ajax/jsonp

后端

处理方式:http[s]

​ address: req.url 抓取 get请求的数据 切字符 | url模块

​ !address: req.on('data',(chunk)=>{CHUNK==每次收到的数据buffer})

req.on('end',()=>{ 接收完毕 切字符 querystring })

postman 一个不用写前端,就可以发出各种请求的软件

url模块

作用

处理 url型的字符串

用法

url.parse(str,true)  返回 对象  true将query处理为对象

str -> obj 返回 对象 true
protocol: 'http:', 协议
slashes: true, 双斜杠
auth: null, 作者
host: 'localhost:8002', 主机
port: '8002', 端口
hostname: 'localhost', baidu
hash: '#title', 哈希(锚)
search: '?username=sdfsdf&content=234234', 查询字符串
query: 'username=sdfsdf&content=234234', 数据
pathname: '/aaa', 文件路径
path: '/aaa?username=sdfsdf&content=234234', 文件路径
href: 'http://localhost:8002/aaa?username=sdfsdf&content=234234#title'

url.format(obj) 返回字符

obj -> str 返回str

querystring 模块

作用

处理查询字符串 如:?key=value&key2=value2

用法

querystring.parse(str) 返回对象
querystring.stringify(obj) 返回字符串

模块化 commonJS

介绍

是主要为了JS在后端的表现制定,commonJS 是个规范 nodejs / webpack 是一个实现

ECMA 是个规范 js / as 实现了他

其他模块化规范:seajs.js / require.js CMD/AMD/UMD es5

作用

是变量具有文件作用域,不污染全局变量

系统模块

http fs querystring url

输入

require('模块名')
require('模块名').xx  按需引用

不指定路径:先找系统模块-> 再从项目环境找node_modules|bower_components (依赖模块)->not found

指定路径 : 找指定路径 -> not found

支持任何类型

输出

exports.自定义属性 = 值 | any

批量输出 都是属性

可输出多次

module.exports = 值 | any        

只能输出一次

注意

commonJS 是 nodejs 默认模块管理方式,不支持es6的模块化管理方式,但支持所有es6+语法

使用: commonjs 引入 let http = require(“http”)

​ 导出 exports.a = a 单个导出,但是可以输出多次

​ module.exports = a 让这个a可以作为对象数组什么的,输出全部,但是只能输出一次

​ es6模块管理方式:引入 import d from './mod/d'

​ 导出 export default d

​ 举例 export 可以导出的是一个对象中包含的多个 属性,方法。
​ export default 只能导出 一个 可以不具名的 对象。

​ import {fn} from ‘./xxx/xxx’ ( export 导出方式的 引用方式 )
​ import fn from ‘./xxx/xxx1’ ( export default 导出方式的 引用方式

NPM

作用

帮助你安装模块(包),自动安装依赖,管理包(增,删,更新,项目所有包)

类似: bower yarn

安装到全局环境

  • 安装到电脑系统环境下
  • 使用时在任何位置都可以使用
  • 被全局安装的通常是:命令行工具,脚手架
npm i 包名 -g                             安装
npm uninstall 包名 -g             卸载

安装到项目环境

只能在当前目录使用,需要使用npm代运行

初始化项目环境

npm init

初始化npm管理文件package.json

package-lock.json 文件用来固化依赖

{
  "name": "npm",    //项目名称
  "version": "0.0.1",   //版本
  "description": "test and play",   //描述
  "main": "index.js", //入口文件
  "dependencies": {  //项目依赖  上线也要用
    "jquery": "^3.2.1"
  },
  "devDependencies": { //开发依赖 上线就不用
    "animate.css": "^3.5.2"
  },
  "scripts": {  //命令行
    "test": "命令行",
  },
  "repository": {   //仓库信息
    "type": "git",
    "url": "git+https://github.com/alexwa9.github.io/2017-8-28.git"
  },
  "keywords": [  //关键词
    "test",'xx','oo'
  ],
  "author": "wan9",
  "license": "ISC", //认证
  "bugs": {
    "url": "https://github.com/alexwa9.github.io/2017-8-28/issues"//问题提交
  },
  "homepage": "https://github.com/alexwa9.github.io/2017-8-28#readme"//首页
}

项目依赖

只能在当前项目下使用,上线了,也需要这个依赖 --save

//安装
npm i 包名 --save
npm install 包名 -S
npm install 包名@x.x.x -S

//卸载
npm uninstall 包名 --save
npm uninstall 包名 -S

开发依赖

只能在当前项目下使用 ,上线了,依赖不需要了 --save-dev

npm install 包名 --save-dev
npm install 包名 -D

查看包

npm list  列出所有已装包
npm outdated 版本对比(安装过得包)
npm info 包名 查看当前包概要信息 
npm view 包名 versions 查看包历史版本列表

安装所有依赖

npm install 

安装package.json里面指定的所有包

版本约束

^x.x.x   约束主版本,后续找最新
~x.x.x   保持前两位不变,后续找最新
*        装最新
x.x.x    定死了一个版本

选择源

npm install nrm -g     安装选择源的工具包
nrm ls 查看所有源
nrm test 测试所有源
nrm use 切换源名

安装卡顿时

ctrl + c -> npm uninstall 包名  -> npm cache 清除缓存 -> 换4g网络 -> npm install 包名

npm run XXX 会运行package下scripts下的命令

发布包

  • 官网 注册
  • 登录
    • npm login 登录
    • 输入 user/password/email
  • 创建包
    • npm init -y
    • 创建入口index.js
    • 编写,输出
  • 发布
    • npm publish
  • 迭代
    • 先修改package里面版本号
    • npm publish
  • 删除
    • npm unpublish

包的发布、迭代、删除,需要在包目录下进行

删除包,有时需要发送邮件

扩展

peerDependencies 发布依赖
optionalDependencies 可选依赖
bundledDependencies 捆绑依赖
contributors 为你的包装做出贡献的人。贡献者是一群人。
files 项目中包含的文件。您可以指定单个文件,整个目录或使用通配符来包含符合特定条件的文件

YARN

安装

注意:为省事,不要用npm i yarn -g,去安装yarn,而是去下载压缩包,保证注册表和环境变量的硬写入,后期通过yarn安装全局包时方便

使用

初始化一个新项目

yarn init

添加依赖包

yarn add [package]
yarn add [package]@[version]
yarn add [package]@[tag]

将依赖项添加到不同依赖项类别中

分别添加到 dependencies,devDependenciespeerDependenciesoptionalDependencies 类别中:

yarn add [package] --save   | -S 
yarn add [package] --dev    | -D 
yarn add [package] --peer
yarn add [package] --optional

升级依赖包

yarn upgrade [package]
yarn upgrade [package]@[version]
yarn upgrade [package]@[tag]

移除依赖包

yarn remove [package]

安装项目的全部依赖

yarn

或者

yarn install

安装到全局

yarn global add [package]               //global的位置测试不能变
yarn global remove [package]

BOWER

安装bower

npm install -g bower

安装包到全局环境

bower i 包名 -g       安装
bower uninstall 包名 -g    卸载

安装包到项目环境

初始化项目环境

bower init

bower.json 第三方包管理配置文件

项目依赖

只能在当前项目下使用,上线了,也需要这个依赖 --save

//安装
同npm
bower install 包名#x.x.x -S 指定版本使用#

//卸载
同npm

开发依赖

只能在当前项目下使用 ,上线了,依赖不需要了 --save-dev

同npm

EXPRESS

nodejs库,不用基础做起,工作简单化,点击进入官网,类似的还有 koa

特点

二次封装,非侵入式,增强形

搭建web服务

let express=require('express')
let server=express()
let server.listen(端口,地址,回调)

静态资源托管

server.use(express.static('./www'));

多资源托管

app.use(express.static(path.join(__dirname, 'public','template')));
app.use('/admin',express.static(path.join(__dirname, 'public','admin')));    //加别名。访问admin时会访问admin下的静态资源
app.use(express.static(path.join(__dirname, 'public')));     //使静态资源托管返回扩大到public下

接口响应

支持各种请求姿势:get、post、put、delete...

app.请求姿势API(接口名称,处理函数)
app.get(url,(req,res,next)=>{})
app.post(url,(req,res,next)=>{})
...

req 请求体

request 对象表示 HTTP 请求,包含了请求查询字符串,参数,内容,HTTP 头部等属性

req.query //获取地址栏的数据
req.body //获取非地址栏的数据  依赖中间件 body-parser  
        中间件使用:body-parser
        1. npm install body-parser 
        2. let bodyParser = require('body-parser') 
        3. app.use(bodyParser ())

req.params //获取动态接口名 返回一个对象{id:3}
        eg:app.get('/api/goods/:id',(req,res)=>{
                // console.log('详情',req.params)
        })
req.method //获取前端提交方式

req.body依赖中间件

中间件使用:body-parser

  1. npm install body-parser
  2. let bodyParser = require('body-parser')
  3. app.use(bodyParser ())

res 响应体

response 对象表示 HTTP 响应,即在接收到请求时向客户端发送的 HTTP 响应数据

res.send(any) //对等 res.write + res.end
res.end(string|buffer)
res.json(json) //返回json
res.status(404).send({error:1,msg:"Sorry can't find that!"}) //返回状态码返回一个404

res.jsonp(响应数据) //调用请求时的回调函数并传递响应数据
res.sendFile(path.resolve('public/error.html'))//渲染纯 HTML 文件

jsonp响应

server.set('jsonp callback name','cb')//默认callback
server.get('/jsonp接口',(req,res,next)=>res.jsonp(数据))        

处理一部分接口

共有业务逻辑,在一起给处理了

server.all('/admin/*',(req,res,next)=>{}))

all匹配全路径 处理所有HTTP

需要next 延续后续

use

安装中间件、路由、接受一个函数,

server.use([地址],中间件|路由|函数体)

中间件

middleware, 处理自定义业务,只处理请求到结束响应的中间部分

举例

npm i body-parser -S //安装包
let bodyParser=require('body-parser')//引入中间件
server.use(bodyParser())//安装中间件

body-parser 使用方式,实时查询 npm,可获得最新

后端跳转

res.redirect(url)      指向一个接口
用新接口代替老接口,在老接口res.redirect(url) 转到新接口

use可以结果根请求 server.use("/"或者不写或者写“/*",(req,res,next)=>{

​ 需要next(),才能往下传

})

扩展

req

  • req.app:当callback为外部文件时,用req.app访问express的实例
  • req.baseUrl:获取路由当前安装的URL路径
  • req.cookies:Cookies
  • req.fresh / req.stale:判断请求是否还「新鲜」
  • req.hostname / req.ip:获取主机名和IP地址
  • req.originalUrl:获取原始请求URL
  • req.path:获取请求路径
  • req.protocol:获取协议类型
  • req.route:获取当前匹配的路由
  • req.subdomains:获取子域名
  • req.accepts():检查可接受的请求的文档类型
  • req.acceptsCharsets / req.acceptsEncodings / req.acceptsLanguages:返回指定字符集的第一个可接受字符编码
  • req.get():获取指定的HTTP请求头
  • req.is():判断请求头Content-Type的MIME类型

res

  • res.app:同req.app一样
  • res.append():追加指定HTTP头
  • res.set()在res.append()后将重置之前设置的头
  • res.cookie(name,value [,option]):设置Cookie
  • opition: domain / expires / httpOnly / maxAge / path / secure / signed
  • res.clearCookie():清除Cookie
  • res.download():传送指定路径的文件
  • res.get():返回指定的HTTP头
  • res.location():只设置响应的Location HTTP头,不设置状态码或者close response
  • res.render(view,[locals],callback):渲染一个view,同时向callback传递渲染后的字符串,如果在渲染过程中有错误发生next(err)将会被自动调用。callback将会被传入一个可能发生的错误以及渲染后的页面,这样就不会自动输出了。
  • res.sendFile(path [,options] [,fn]):传送指定路径的文件 -会自动根据文件extension设定Content-Type
  • res.set():设置HTTP头,传入object可以一次设置多个头
  • res.status():设置HTTP状态码
  • res.type():设置Content-Type的MIME类型

作业

把jquery项目利用express搭建的node服务器来管理,保证资源托管到位,部分接口实现

身份验证

HTTP 是一种没有状态的协议,也就是它并不知道是谁访问。客户端用户名密码通过了身份验证,不过下回这个客户端再发送请求时候,还得再验证

session

思想

1、客户端用户名跟密码请求登录
2、服务端收到请求,去库验证用户名与密码
3、验证成功后,服务端种一个cookie或发一个字符到客户端,同时服务器保留一份session
4、客户端收到 响应 以后可以把收到的字符存到cookie
5、客户端每次向服务端请求资源的cookie会自动携带
6、服务端收到请求,然后去验证cookie和session,如果验证成功,就向客户端返回请求的库数据

Session存储位置: 服务器内存,磁盘,或者数据库里

Session存储内容: id,存储时间,用户名等说明一下登录的用户是谁

客户端携带 : cookie自动带,localStorage手动带

如何保存信息给浏览器

前端种:

cookie/localstorage

后端种:

服务器给浏览器种cookie: cookie-parser

服务器给浏览器种cookie的同时在服务器上生成seesion: cookie-session

cookie-session

安装引入

let cookieSession = require('cookie-session')

配置中间件

app.use(cookieSession({
  name:'保存到服务器的session的名字',
  keys:[必传参数,代表加密层级],    //示例keys:['aa','bb','cc','dd'],
  maxAge:1000 //保留cookie的时间
}))

种cookie,备份session

req.session.key=value    //示例req.session['nz_1909']='userid';

读cookie对比session

req.session.key  取cookie对比服务器的session,会返回userid,可以从返回的这个值是否存在来进行自动登录

删除cokkie、session

delete req.session.key 
req.session.key = undefined

token

思想

在服务端不需要存储用户的登录记录,全部发给客户端有客户端自己存(cookie,local)

1、客户端使用用户名跟密码请求登录
2、服务端收到请求,去验证用户名与密码
3、验证成功后,服务端会签发一个 Token(加了密的字符串),再把这个 Token 发送给客户端
4、客户端收到 Token 以后可以把它存储起来,比如放在 Cookie 里或者 Local Storage 里
5、客户端每次向服务端请求资源的时候需要带着服务端签发的 Token
6、服务端收到请求,然后去验证客户端请求里面带着的 Token,如果验证成功,就向客户端返回请求的数据

实现

jsonwebtoken的安装引入

let jwt = require('jsonwebtoken')

生成签名

let token = jwt.sign(payload, secretOrPrivateKey, [options, callback])

校验token

jwt.verify(token, secretOrPublicKey, [options, callback])

token删除

有客户端,负责删除

session vs token

session token
服务端保存用户信息 ×
避免CSRF攻击 ×
安装性 一般
多服务器粘性问题 存在 不存在

多服务器粘性问题

当在应用中进行 session的读,写或者删除操作时,会有一个文件操作发生在操作系统的temp 文件夹下,至少在第一次时。假设有多台服务器并且 session 在第一台服务上创建。当你再次发送请求并且这个请求落在另一台服务器上,session 信息并不存在并且会获得一个“未认证”的响应。我知道,你可以通过一个粘性 session 解决这个问题。然而,在基于 token 的认证中,这个问题很自然就被解决了。没有粘性 session 的问题,因为在每个发送到服务器的请求中这个请求的 token 都会被拦截

文件上传

思想

前端表单->后端接收到文件本身->保存到服务器上->给数据库记录文件一些信息->库返回给nodejs相关信息->nodejs返回给前端

前端: <input type=file enctype="multipart/form-data" name="fieldname"

实现

multer->文件名会随机->fs模块改名->path系统模块解析磁盘路径

后端:multer 接受 form-data编码数据

path系统模块

操作系统磁盘路径

编码

windows: c:\\user\\admin\\a.jpg

mac: ~/desktop/1901

UI呈现

windows: c:\user\admin
mac: ~/desktop/1901

API

磁盘路径解析 parse

path.parse('c:\\wamp\\xx.png') // string -> object

//返回
{
   root: 'c:\\', 盘符
   dir: 'c:\\wamp', 目录
   base: 'xx.png',  文件名
   ext: '.png', 扩展名
   name: 'xx'   文件,不含扩展名
}

片段合并join

path.join('磁盘路径1','磁盘路径2','磁盘路径n')

__dirname 魔术变量 返回当前文件所在的磁盘路径

片段合并 resolve

path.resolve('磁盘路径1','磁盘路径n')

合并磁盘片段,右到左找根,左到右拼接,没有找到根,以当前文件路径为根,

对比join来说,不需要__dirname 这个魔术变量

multer中间件

multer 接受 form-data编码数据,所有要求前端携带时注意一下,如:

<input type=file enctype="multipart/form-data" name="fieldname",

使用

//1 引入
let multer  = require('multer');
//2 实例化  
let objMulter = multer({ dest: './upload' }); //dest: 指定 保存位置(存到服务器)
//安装中间件, 
app.use(objMulter.any());  //允许上传什么类型文件,any 代表任何类型 

中间件扩展了req请求体 req.files

app.get('/reg',(req,res)=>{
  req.files
})

​ fieldname: 表单name名
​ originalname: 上传的文件名
​ encoding: 编码方式
​ mimetype: 文件类型
​ buffer: 文件本身
​ size:尺寸
​ destination: 保存路径
​ filename: 保存后的文件名 不含后缀
​ path: 保存磁盘路径+保存后的文件名 不含后缀

后端渲染

通常根据后端返回的json数据,然后来生成html被称为前端渲染,而后端渲染是后端把json与html结合渲染好后返回到浏览器,没前端什么事了

模板引擎

无论前后谁来渲染页面,都会用到模板引擎,前端渲染页面实际上是操作dom,后端渲染页面是把数据和html字符拼接后丢给浏览器

引擎 前端 后端
angularJs ×
vue/mustach
react
angularTs/mustach
jade/pug ×
ejs ×
jquery + art-template ×
handlerbars ×

jade

原理:fs抓取前端静态页面 + jade + 数据 -> 返回send(data) -> 浏览器

特点:侵入式,强依赖

使用

let jade = require('jade')
let html = jade.renderFile('jade模板文件',{数据},{pretty:true});  //返回字符

jade模板文件语法

父子要缩进
属性: 标签(key=value,key2=value)
内容: 标签 内容

其他扩展

ejs

原理:fs抓取前端静态页面 + ejs + 数据 -> 返回send(data) -> 浏览器

特点:非侵入式,温和,弱依赖

使用

let ejs = require('ejs')
ejs.renderFile('ejs模板文件',{要合并到html数据},回调(err,data))

err:错误,null代表没有错误

data: 渲染后的字符|流

ejs模板 : 后缀名为ejs的html文件

ejs模板文件语法

  • ejs 结构就是html
  • 输出: <%= 数据名|属性名|变量名 + 表达式 %>
  • 语句: <% 语句 %> 需要被<% %> 包裹
  • 非转义输出: <%- 数据名|变量名 + 表达式 %> 就是标签不会被转成符号gt等
  • 载入公共:<%- include('./hd.ejs',{数据}) %> //注意有个-

其他扩展

eg:条件渲染

 <% if(bl){ %>
    <nav>菜单1</nav>
<% }else{ %>
    <nav>菜单2</nav>
<% } %>
列表渲染  
     <ul>
        <% for(var i=0;i<list.length;i++){ %>
            <li>
                <a href="<%= list[i].href %>">
                    <%= list[i].des %>
                </a>
            </li>
        <% } %>
    </ul>

多引擎管理

把多个模板引擎用在一个后端应用中,统一他们的用法,绑定到res、req身上

安装+配置

npm i consolidate ejs jade -S

注意: ejs jade 等多个引擎需要安装,但无需引入

app.js

//中间件配置
app.set('view.engine','html');  //模板最终  输出类型设置
app.set('views','./views');     //引擎模板目录设置

app.engine('html',consolidate.ejs); //输出与引擎匹配
app.engine('css',consolidate.jade); //输出与引擎匹配

//渲染
app.get('xxx',(req,res)=>{
  res.render('模板文件名要加后缀',{要渲染的数据}) //整合页面和数据,完成渲染,发往浏览器,并结束响应
})

路由

告诉你去哪,对于前端,主要是导向告诉浏览器应该去哪,对于后端,可以理解为一个子服务,一个路由就是一个小的服务(server/app),处理一个接口

配置和使用

/routes/xx.js

// 1. 创建路由
let router = express.Router(); 

//2 路由处理响应
router.响应API(地址, 处理函数)

//3. 导出路由
module.exports = router;

/app.js主服务

//安装路由
app.use('地址',router); 

当需要嵌套使用时 /routes/xx.js

//字路由里安装路由 嵌套
router.use('地址',子router) 

//截获当前路由下的部分公共业务
router.all('*',当前router路由下以及所有子路由的验证工作) //需要next 延续
all出现 的位置很重要,all的上方的路由截获不到,只能截获all一下的路由

主路由的地址对应子路由的根

如:app.js :/api/user ~~ user.js: /

如: app.js: /api/user/add ~~ user.js: /add

作业

实现jquery项目当中,自动登录,注册时头像的上传,接口逻辑利用路由实现

数据库

mysql

关系数据库,二维表,不存在子表

sql语句

建库

CREATE DATABASE  `2017-12-6` DEFAULT CHARACTER SET armscii8 COLLATE armscii8_general_ci;

建表

CREATE TABLE  `2020-12-6`.`user` (
                    `name` VARCHAR( 32 ) NOT NULL ,
                    `age` INT( 3 ) NOT NULL ,
                    `address` VARCHAR( 128 ) NOT NULL
                    ) ENGINE = INNODB

INSERT INTO 表 (字段列表) VALUES(值列表)
INSERT INTO user (name,age,address) VALUES('苏菲',38,'外滩18号')

DELETE FROM 表 WHERE 字段名=值
DELETE FROM user WHERE name='alex'

UPDATE 表 SET 字段名=值 WHERE 字段名=值
UPDATE user set name='sufei' WHERE name='苏菲'

SELECT ? FROM 表
SELECT * FROM user  查所有

node + mysql客户端

安装+引入

npm install mysql -S
var mysql = require('mysql');

创建库链接

var connection = mysql.createConnection({
  host     : 'localhost',//主机名
  user     : 'me',
  password : 'secret',
  database : 'my_db'//库名
});
 
connection.connect();//连接数据库

表操作

connection.query('SQL语句', function (error//错误, results//结果, fields//成功后的描述) {
  if (error) throw error;
  console.log('The solution is: ', results==  查询array||  增删改object);
});

关闭库

connection.end();

mongodb

非关系型数据库,又叫nosql,缓存型,使用场景多是解决大规模数据集合多重数据种类

  1. 下载 安装帮助

  2. 配置数据文件存储位置:

找到安装目录C:\Program Files\MongoDB\Server\4.0\bin -> cmd回车-> mongod --dbpath c:\data\db

data和db目录要手动创建

  1. 服务端启动: 可选

找到安装目录C:\Program Files\MongoDB\Server\4.0\bin -> cmd回车-> mongod 回车

一般开启会默认启动

  1. 客户端启动:

找到安装目录C:\Program Files\MongoDB\Server\4.0\bin -> cmd回车-> mongo 回车

  1. 环境变量 可选

为了在任意盘符下去都可以启动 mongod服务端|mongo客户端,把安装目录添加到环境变量

mysql vs mongodb

mysql mongoDb
database(库) database(库)
table(表) collection(集合)
row(一条数据) document(文档)
column(字段) field(区域)
二维表,每次存到磁盘 json,存在缓存,关闭时存到磁盘 存储方式

mongodb命令行操作 声明式 | obj.api()

库操作

查: show dbs
    db 查看当前库
建:  use 库名     没有建,有就切换

集合(表)操作

建:db.createCollection('表名',{配置})
  //配置:{size:文件大小,capped:true,max:条数|文档数} capped定量
  //db.表(集合).isCapped() 返回 true/false 是否是定量
查:show collections / db.getCollectionNames()
删:db.表|集合.drop()

文档(row)操作

db.集合.save({}) //添加一条
db.集合.insert({})  //添加一条
db.集合.insertOne({}) //添加一条

db.集合.save([{},{}]) //多条
db.集合.insert([{},{}]) //多条
//insert  不会替换相同ID,会操作失败
//save会对相同id的数据进行替换

db.集合.deleteOne({要删数据条件描述}) //一条
db.集合.remove({},true)  //一条

db.集合.remove({要删数据条件描述}) //多条
db.集合.remove({}) //清空表

db.集合.update({查询条件},{替换条件},[插入false],[全替换false])

查询条件

{age:22} age == 22
{age:{gt:22}} age > 22 {age:{lt:22}} age < 22
{age:{gte:22}} age>=22 {age:{lte:22}} age<=22
{age:{lte:122,gte:22}} age<=122 && age>=22
{$or:[{age:22},{age:122}]} 22 or 122
{key:value,key2:value2} value && value2
{name:/正则/}

替换条件

{set:{数据},inc:{age:-1}}

$inc 年龄递减1

所有:db.集合.find(条件)
条数: db.集合.find().count()

db.集合.find({条件},{指定要显示列区域})

指定要显示列区域

username:1 显示这个区域,其他不显示

username:0 不显示这个区域,其他显示

_id 是默认显示

db.集合.find().sort({key:1,key2:-1}) //升
db.集合.find().sort({key:-1}) //降

限定

db.集合.find().limit(number)  //限定
db.集合.find().skip(number)   //跳过
db.集合.findOne()//找第一个
db.集合.find().limit(1)  //查询第一条

node + mongodb客户端

安装+引入

npm install mongodb -S
var mysql = require('mongodb');

实例化并连接

let mongoCt = mongodb.MongoClient;
mongoCt.connect('协议://地址:端口',{ useUnifiedTopology: true },回调(err,client)) 
//err 错误 client链接后的客户端
//没改的话,默认:mongodb://127.0.0.1:27017

链接库和集合

let db = client.db('库名')
let user = db.collection('集合名');

集合操作

//user.API()  集合操作  返回 对象

//增
    insertOne(对象数据,(err,res)=>{})  //res = 对象  成功后返回的数据 
    insertMany(arr数据,(err,res)=>{}) //res = 对象  成功后返回的数据 
    //res.result.n 结果  ok 状态
    //res.ops内容  数组
    //result.insertedId 插入后的id

//删:
  deleteOne({条件},(err,result)=>{})

//改:
  updateOne({条件},{更新后},(err,res)=>{})
  updateMany({条件},{更新后},(err,res)=>{})
  updateMany({条件},{更新后},{配置},(err,res)=>{})
    //配置: upsert:true 未找到的时候是否插入    projection:true 是否进行全局替换
    //更新后:替换条件{$set:{数据},$inc:{age:-1}} 比如说
    eg:     user.updateMany({
                age:{$gt:16}
            },{
                 $set:{address:'人民公园',fans:111}
            },{
            upsert:false,//没找到,是否把address插入成新数据
             projection:true//让什么显示,true全部显示,projection:{key:1}让key显示,0为不显示
             },(err,result)=>{})

//查:
  user.find({条件},{skip:1,limit:1,projection:{key:1}},(err,result)=>{result=对象})
  user.find({条件},{projection:{key:0}}).toArray((err,result)=>{reulst==arr})
  user.countDocuments((err,num)=>{num返回数量})
    skip 跳过第一个      projection:{key:1} 使key这个键显示,当设置为0的时候让他不显示

//排
  user.find(..).sort({key:-1}).toArray..
  user.find({},{projection:{},sort:{key:-1}}).toArray..

关闭库

client.close()

node + mongoose

一款mongodb客户端 官网 中文

可视化客户端

Express生成器

应用程序生成器、脚手架 、命令行工具、自动搭建项目环境的,无需手动

安装

npm install express-generator -g   

验证

express -h

生成环境

express -e 目录 | . 
    // . 当前目录创建 
    //-e 需要ejs模板引擎
    //express -f  强制在非空目录下创建
cd 目录
npm install         //安装依赖包
npm start    
node ./bin/www

扩展

项目

定义数据字典

也就是数据库设计,有了数据结构,后端才知道如何存储,前端才知道如何渲染,有条不成文的规定,数据结构有前端说了算,请求姿势(api分格)后端说了算,最终老大说了算

数据结构

q关键字筛选

banner: [
    { 
      "_id" : xx, 
      "title" : "1", 
      "sub_title" : "1", 
      "banner" : "xxxx", 
      "time":234234,
      "detail" : { 
        "auth" : "", 
        "content" : "<p>xxx<p>", 
        "icon" : "/upload/banner/9d4083b4f1d28a6c0fb4c463526790eb.jpg" 
      },
    }
  ]
product: 
  { 
    "_id" : xx, 
    "title" : "1_", 
    "des" : "2", 
    "time":234234,
    "detail" : { 
      "auth" : "4", 
      "content" :"<p>3</p>", 
      "auth_icon" : "/upload/user/xxx.jpg" 
    } 
  }
user:   
  { 
    "_id" : xx, 
    "username" : "alex", 
    "password" : "alex123", 
    "follow" : "100", 
    "fans" : "200", 
    "nikename" : "九叔_", 
    "icon" : "/upload/968a3b7218ee744931276a64c9b7ea01.png", 
    "time" : 1551620448550 
  }
super:
  { 
    "_id" : xx, 
    "username" : "admin", 
    "password" : "admin123", 
    "icon" : "/img/avatar-5.jpg" 
  }

请求方式 RESTful API

增 POST /user  body中包含数据
删 DELETE /user/1 | user?id=1 根据ID删除用户信息
改 PUT|PATCH /user body中包含数据 PUT覆盖修改 PATCH局部修改
查 GET /user/1 | user?id=1 GET  根据用户id查询用户数据 没有id查询所有 /1 返对象 id=1 返回数组>对象
分页  _page 第几页, _limit一页多少条
  GET /user?_page=7  不传递默认0条
  GET /user?_page=7&_limit=20 不传递默认10条
排序 _sort设定排序的字段 _order设定排序的方式(默认升序)
  GET /user?_sort=views&_order=asc
  GET /user/1/comments?_sort=votes&_order=asc
  GET /user?_sort=title,views&_order=desc,asc   多个字段排序
任意切片数据 _start 开始不包含  _end 结束包含
  GET /users?_start=20&_end=30
  GET /user/1/comments?_start=20&_end=30
  GET /user/1/comments?_start=20&_limit=10
全文检索    GET /user?q=九哥

搭建开发环境

引入各种包、各种中间件、做好目录规划

bin  |-
     www 启动文件
utils|- 全局公共
  |- douban|mgd|mysql
config 全局配置
  |- global (_page,_limit,q,upload...)
  |- server (local,http,https)
public 资源托管
  |-admin 管理端
  |-template 用户端
  |-upload
    |- banner|product|user
    |- product
      |- home|follow|column
routes 子服务,路由
  admin 管理端
    |- feedback
      |- success|error
    |- product
      |- add|del|check
    |- banner
      |- add|del|check
    |- user
      |- add|del|check
    |- home| product|banner|user
    |- islogin | login | reg | logout
  api 用户端
    |- product (home/follow/column)
    |- banner
    |- user 
    |- login
    |- reg
    |- logout
  proxy 代理
    |- douban
    |- ....
views 管理端模板 ejs 
  |- feedback
      |- success|error|app_error
  |- ... 结构同 admin 管理端
  |- common
    |- header|footer|slider|crumb|toolbar|paging

用户端API

用户密码入库时加密

let bcrypt = require('bcrypt')

加密: var hash = bcrypt.hashSync(用户传过来的明文密码, 加盐数); 

校验:  bcrypt.compareSync(用户传过来的明文密码, hash); // true|false

短信验证

  1. 开通短信服务

    1. 登录阿里云账号->进入
  2. 设置签名管理

    1. 添加 签名、模板 进入
    2. 冲1块钱,签名是收费的
  3. 申成短信服务器接口的node代码

    1. 进入

    2.               PhoneNumbers: 电话
                        SignName: 签名
                        TemplateCode: 模板id
                        accessKeyId: 阿里云账号->accessKey管理
                        accessKeySecret: 阿里云账号->accessKey管理
      

管理端API

登录注销

登录接口:/admin/login/submit

实现
var express = require('express');
var router = express.Router();
var mgdb = require('../../common/mgdb')

router.get('/', function(req, res, next) {
  res.render('login',{});
});
router.post('/submit', function(req, res, next) {
  let {username,password} = req.body;

  mgdb(
    {collection:'admin'},
    ({collection,client})=>{
      collection.find(
        {username,password},
        {
          projection:{_id:0}
        }
      ).toArray((err,result)=>{
        if(!err && result.length>0){
          //种cookie , 留session
          req.session['username']=result[0].username;
          req.session['icon']=result[0].icon;

          res.redirect('/admin/home');
        }else{
          // res.redirect(跳转地址==string)
          res.redirect('/admin/error?msg=登录失败,用户或者密码有误')
        }
      })
    }
  )


});
module.exports = router;

//======================================

注销接口:/admin/reg

实现
var express = require('express');
var router = express.Router();

router.get('/', function(req, res, next) {
  // req.session=null;
  // req.session.username=undefined
  delete req.session.username;//删除session 混淆 cookie
  delete req.session.icon;

  res.redirect('/admin/login');
});

module.exports = router;

===============================
  
自动登录
app.all('/admin/*',require('./routes/admin/islogin'))

islogin

module.exports=(req,res,next)=>{
  if(!req.session['username']){
    res.redirect('/admin/login')
  }else{
    //处理公共参数
    let start = req.query.start ? req.query.start - 1 : require('../../config/global').page_start - 1;
    let count = req.query.count ? req.query.count - 0 : require('../../config/global').page_num - 0;
    let q = req.query.q ? req.query.q : require('../../config/global').q;
    let rule = req.query.rule ? req.query.rule : require('../../config/global').rule;
    let _id = req.query._id;
    let dataName = req.query.dataName;
    let page_header = dataName;
    let active = dataName;

    res.params = {start,count,q,rule,dataName,page_header,active,_id}
    res.user_session={username:req.session.username,icon:req.session.icon}
    next();//交给app.use后续响应处理
  }
};

添加

接口: /admin/product/add?dataName=xx

实现

var express = require('express');
var router = express.Router();
var pathLib = require('path')
var uploadUrl = require('../../../config/global').upload.product
var fs = require('fs');
var mgdb = require('../../../common/mgdb')

router.get('/', function(req, res, next) {

  //1.必传参数
  let dataName = req.query.dataName;
  if(!dataName){
    res.redirect('/admin/error?msg=dataName为必传参数')
    return;
  }

  //公共数据 start=1|q=''|rule=''|page_header|dataName|user_session
  let common_data={
    ...res.user_session,
    ...res.params,
    page_header:dataName+'添加',
  }

  res.render('product/add',common_data);
});

router.post('/submit', function(req, res, next) {

  //1.必传参数
  let dataName = req.body.dataName;
  if(!dataName){
    console.log(1)
    res.send('/admin/error?msg=dataName为必传参数')
    return;
  }

  //2.整理公共数据|库数据
  let {title,content,des,auth} = req.body;
  let time = Date.now();//添加时间
  
  //multer拆出上传图片,需要解决没有上传头像
  let auth_icon = req.files.length ? uploadUrl + req.files[0].filename + pathLib.parse(req.files[0].originalname).ext : '';
  
  if(auth_icon){
    fs.renameSync(
      req.files[0].path,
      req.files[0].path+pathLib.parse(req.files[0].originalname).ext
    )
  }else{
    auth_icon = '/upload/noimage.png';
  }


  //3.写库 + 跳转

  mgdb({
    collection:dataName
  },({collection,client})=>{
    collection.insertOne({
      title,des,time,detail:{auth,content,auth_icon}
    },(err,result)=>{
      if(!err && result.result.n){
        let io=require('../../../bin/www');
        io.emit('update_product', {data:result.ops[0]})

        res.send('/admin/product?dataName='+dataName+'&start=1')
      }else{
        res.send('/admin/error?msg=集合操作错误')
      }
      client.close();
    })
  })

});

module.exports = router;

富文本框使用

注意jq库冲突: 要使用富文本框提供的jquery-3.2.1.slim.min
问题: slim.min没有$.ajax
解决: 使用jquery,禁用slim.min

图片上传FormData混合提交 流文件与普通表单混合

form_data = new FormData() | new FormData(表单本身)
form_data.append(key,value) 通过req.body获取
value 可以是file:
<input type="file" name="file2" id="file2" />
formData.append("file2", $('#file2')[0].files[0]);
通过multer的req.files获取

$.ajax({
  contentType: false,//不设置编码类型,在进行文件流与普通字符串混合上传的时候,需要设置为false
    processData: false,//不进行数据处理
})

后端需要处理未传图(req.files空)

js抓取ejs变量
form_data.append('dataName',"<%=dataName%>");
ajax提交后,nodejs需返回跳转地址,由前端跳转
子节点排序
.sort({'detail.time':-1,xx:oo})

删除

接口: /admin/product/del?dataName=xx&_id=xx&start=2&count=2&q=b&rule=_id

实现

var express = require('express');
var router = express.Router();
var mgdb = require('../../../common/mgdb');

router.get('/', function(req, res, next) {
  //1.必传参数
  let {dataName,_id,start,count,q,rule} = res.params;
  if(!dataName || !_id){
    res.redirect('/admin/error?msg=dataName和_id为必传参数')
    return;
  }
  
  //3. 写库
  mgdb({
    collection:dataName
  },({collection,client,ObjectID})=>{
    collection.deleteOne({
      _id:ObjectID(_id)
    },(err,result)=>{
      //4. 渲染页面|跳转页面
      if(!err && result.result.n){
        res.redirect('/admin/product?dataName='+dataName+'&start='+(start+1)+'&count='+count+'&q='+q+'&rule='+rule)
      }else{
        res.redirect('/admin/error?msg='+dataName+'操作错误')
      }
      client.close();
    })
  })
  
});

module.exports = router;

ID操作注意
var ObjectId = require('mongodb').ObjectId;
id = ObjectId(req.query.id); 此时的id才是ajax传过来的id,才能与数据库对照

修改

接口: /admin/product/check?dataName=xx&_id=xx&start=2&count=2&q=b&rule=_id

实现

var express = require('express');
var router = express.Router();
var mgdb = require('../../../common/mgdb')
let pathLib = require('path');
let fs = require('fs');
let uploadUrl = require('../../../config/global').upload.product;

router.get('/', function (req, res, next) {
  //1.必传参数
  let {dataName,_id,start} = res.params;
  if (!dataName || !_id) {
    res.redirect('/admin/error?msg=dataName和_id为必传参数')
    return;
  }

  //公共数据 
  let common_data = {
    ...res.user_session,
    ...res.params,
    page_header: dataName + '修改',
    start:start+1
  }

  //找到这条数据
  mgdb({
    collection: dataName
  }, ({ collection, client, ObjectID }) => {
    collection.find({
      _id: ObjectID(_id)
    }).toArray((err, result) => {
      
      if (!err && result.length>0) {
        let data = {
          ...common_data,
          page_data: result[0]
        }
        console.log(data)
        res.render('product/check', data);
      } else {
        res.redirect('/admin/error?msg=' + dataName + '操作错误')
      }
      client.close();
    })
  })


});

router.post('/submit', function (req, res, next) {
  //1.必传参数
  let dataName = req.body.dataName;
  let _id = req.body._id;
  if (!dataName || !_id) {
    res.redirect('/admin/error?msg=dataName和_id为必传参数')
    return;
  }

  //可选参数
  let start = req.body.start ? req.body.start - 0 : require('../../../config/global').page_start
  let count = req.body.count ? req.body.count - 0 : require('../../../config/global').page_num
  let q = req.body.q ? req.body.q : require('../../../config/global').q;
  let rule = req.body.rule ? req.body.rule : require('../../../config/global').rule;

  //2.整理公共数据|库数据
  let {title,content,des,auth,old_auth_icon} = req.body;
  //old_auth_icon 添加是保存的图
  
  //multer拆出上传图片,需要解决没有上传头像
  let auth_icon = req.files.length ? uploadUrl + req.files[0].filename + pathLib.parse(req.files[0].originalname).ext : '';
  
  if(auth_icon){
    fs.renameSync(
      req.files[0].path,
      req.files[0].path+pathLib.parse(req.files[0].originalname).ext
    )
  }else{
    auth_icon = old_auth_icon;
  }


  //3.写库 + 跳转

  mgdb({
    collection:dataName
  },({collection,client,ObjectID})=>{
    collection.updateOne({
      _id:ObjectID(_id)
    },{
      $set:{title,des,detail:{auth,content,auth_icon}}
    },(err,result)=>{
      if(!err && result.result.n){
        res.send('/admin/product?dataName='+dataName+'&start='+start+'&count='+count+'&q='+q+'&rule='+rule)
      }else{
        res.send('/admin/error?msg=集合操作错误')
      }
      client.close();
    })
  })
});

module.exports = router;

后端需要处理未传图(req.files空)
前端修改时抓取库图地址(渲染用),提交时传递接收到的库图和本地图,服务器优先抓取本地图
修改时删除之前的图片fs.unlink
ajax提交后,nodejs需返回跳转地址,由前端跳转

检索|排序

接口: /admin/product?dataName=home&start=2&count=2&q=b&rule=detail.time

实现
var express = require('express');
var router = express.Router();
var mgdb = require('../../common/mgdb')

router.get('/', function (req, res, next) {

  let {dataName,q,rule,start,count} = res.params;
  if (!dataName) {
    res.redirect('/admin/error?msg=dataName为必传参数')
    return;
  }


  let common_data = {
    ...res.user_session, 
    ...res.params, 
    page_header: dataName + '列表',
    start: start + 1,
    api_name:'product'
  }

  mgdb({
    collection: dataName
  }, ({ collection, client }) => {
    collection.find(
      q ? { title: eval('/' + q + '/g') } : {},
      {
        projection: {
          _id: 1, title: 1
        },
        // sort: rule? {[rule]:-1} : {'detail.auth':-1}
        sort: rule ? { [rule]: -1 } : { 'time': -1 } //排序条件默认按时间排序
      }
    ).toArray((err, result) => {
      let checkResult = result.slice(start * count, start * count + count)//提取要分页的数据
      let data = {
        ...common_data,
        page_data: checkResult,
        page_count: Math.ceil(result.length / count)//计算总页数
      }
      res.render('product', data);
      client.close();
    })
  })

});

router.use('/add', require('./product/add'));
router.use('/del', require('./product/del'));
router.use('/check', require('./product/check'));

module.exports = router;

查询 eval('/'+ q +'/g')
排序 sort:rule ? {[rule]:-1} : {'detail.time':-1}
排序关键字: (标题title|时间:detail.title)
分页: 取所有,挑出对应页数,返回给浏览器

接口文档编写

为了以后前端使用方便,把api的使用方式,或者说请求方式做成文档,文档可以是,word,pdf

扩展

代理

前端通过ajax访问第三方接口时,会出现浏览器的跨域行为,可以通过后端代理绕过,前端只需要访问我方后端接口,我方后端去请求第三方接口,浏览器和我方后端同域,这样绕开浏览器的跨域限定

豆瓣的请求姿势

hostname:'douban.uieee.com',//主机名
port: 443,//端口
path:'/v2/movie/top250?start=3&count=1',
method:'get'

聚合的请求参数

hostname:'v.juhe.cn',
// port:80,
path:'/toutiao/index?type=&key=55f8053eba54dab5a301a00f45523164',
method:'GET'

我方后端请求

let http[s]=require('http[s]')

options={
  hostname:'api.douban.com',
  port:443,
  path:'/v2/movie/top250?count='+req.query.count,
  method:'GET'
};

//发送http[s]请求
let reqHttp = http[s].request(配置项,回调(响应对象resHttp)){ //返回请求对象reqHttp
  resHttp 响应对象
  resHttp.statusCode 状态码  200 OK
  resHttp.headers 获取响应头信息
  resHttp.setEncoding('utf-8') 设置编码方式
  resHttp.on('data/end',fn)  ->send给前端
});

reqHttp //请求对象
reqHttp.on('error',(err)=>{console.log(err)});  //监听请求失败信息
reqHttp.end();//请求结束

正向代理 VS 反向代理

有一台服务器出现在客户端和真实服务器之间,这台服务器叫代理服务器,他可能两端都有可能出现

正向代理(客户端代理) 反向代理(服务端代理)
目标 帮客户端访问其无法访问的服务器 帮服务器做负载均衡,安全防护等
位置 客户端架设 服务器架设
知晓 真实服务器不知道客户端是谁 客户端不知道真实服务器是谁
场景 vue、react的开发环境中架设,浏览器端安装代理软件 机器集群中部署一个反向代理服务器ngnix

socket.io

介绍

Web领域的实时推送技术,也被称作Realtime技术。这种技术要达到的目的是让用户不需要刷新浏览器就可以获得实时更新。它有着广泛的应用场景,比如在线聊天室、在线客服系统、评论系统、WebIM等。

原理

双向通信,前端H5api (WebSocket) + 后端net模块

服务端配置

修改www

const SOCKETIO = require('socket.io');//创建socket实例
const io = SOCKETIO.listen(server);//监听http实例,未来3000端口下的http请求,会触发socket
module.export = io;//写在最后

注意: www 不热重启,不检查

客户端配置

html注入客户端库

<script src="/socket.io/socket.io.js"></script>
<script>
    连接服务器:socket = io('http://localhost:3000');
</script>

服务端推送

//完成一次添加工作后↓
let io = require('../../../bin/www'); //require要在需要时再引入
io.emit('mess_type',{data:'服务端的推送数据')//推送

客户端接收

<script src="/socket.io/socket.io.js"></script>
<script>
  //连接服务器
    var socket = io('http://localhost:3000');
  socket.on('new_movie',(data)=>{
    console.log('首页_客户端收到',data)
  })
</script>

客户端推送到客户端

客户端(未指定消息|指定的消息)->服务器(广播|私信给指定)->客户端

聊天室思想

客户端(未指定消息|指定的消息)->服务器(广播|私信给指定)->客户端

服务端API

检测客户端连接:io.on('connection', (socket) =>{}) 回调函数接收客户端socket
接受:socket.on('消息名称',(data)=>{}) data=接受到的消息
广播: io.emit('消息名称', {数据});

检测客户端下线:    socket.on('disconnect',(data)=>{})

接受私信:
socket.on('消息名称',(toUserName,data,callback)=>{})
                    toUserName==目标用户 callback==给发送端的回调
发私信:    接受消息的socket.emit('消息名称',{数据})
                    发私信 ->      socket   == onlineUsers[toUserName]
注意,data数据 里面不可以包含socket对象,发往客户端,量太大

客户端API

发送未指定消息:    socket.emit('消息名称',{到服务器的数据})
发送私息:   socket.emit('消息名称',toUserName,{到服务器的数据},(由服务器返回的数据)=>{})
接受消息:   socket.on('消息名称',(data)=>{})

跨域

有时,前端和后端的工程文件不在同一个域,也会出现跨域,一下是解决方案

后端解决

部分接口允许

要允许的接口内部
res.setHeader('Access-Control-Allow-Origin', req.headers.origin)

所有接口允许

let core = require('core');

app.use(cors({
  //允许所有前端域名
  "origin": ["http://localhost:8001","http://localhost:5000","http://localhost:8080"],  
  "credentials":true,//允许携带凭证
  "methods": "GET,HEAD,PUT,PATCH,POST,DELETE", //被允许的提交方式
  "allowedHeaders":['Content-Type','Authorization']//被允许的post方式的请求头
}));

前端解决

jsonp

浏览器装插件

环境做代理(webpack)

异步流程控制

有时前端的请求在后端处理时,后端要多次请求库时,用的上

安装:npm i async -S

串行无关联

多个异步依次请求,请求之间不依赖

async.series([fn1(callback),fn2(callback)],处理函数(err,result))
            //callback(err,数据)->callback(null,'one')
            //花费时间是:fn1+fn2

async.series({xx:fn(callback),xx:fn(callback)},处理函数(err,result))
            //花费时间是:fn1+fn2

并行无关联

多个异步同时请求,请求之间不依赖

async.parallel(数组|对象,回调(err,result))  √

async.parallel([fn1(callback),fn2(callback)],处理函数(err,result))
                callback(err,数据)->callback(null,'one')
async.parallel({xx:fn(callback),xx:fn(callback)},处理函数(err,result))
            //花费时间是:用时最多的那个fn

串行有关联

多个异步依次请求,请求之间依赖

async.waterfall(数组|对象,回调(err,result)) √
async.waterfall(
  [fn1(callback){callback(null,data)},fn2(data,callback)],
  处理函数(err,result)
)
        //result 接受最后一个函数传递过来的一个参数

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,817评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,329评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,354评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,498评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,600评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,829评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,979评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,722评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,189评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,519评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,654评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,329评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,940评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,762评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,993评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,382评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,543评论 2 349