【Node.js工程师养成计划】之express框架

一、Express

官网:http://www.expressjs.com.cn

express 是一个基于内置核心 http 模块的,一个第三方的包,专注于 web 服务器的构建。

Express 是一个简洁而灵活的 node.js Web应用框架, 提供了一系列强大特性帮助你创建各种 Web 应用,和丰富的 HTTP 工具。

使用 Express 可以快速地搭建一个完整功能的网站。

Express 框架核心特性:

  • 可以设置中间件来响应 HTTP 请求。

  • 定义了路由表用于执行不同的 HTTP 请求动作。

  • 可以通过向模板传递参数来动态渲染 HTML 页面。

Express适合做什么

  • 传统web网站
  • API接口服务器
  • 服务端渲染中间层
  • 开发辅助工具
  • 自定义集成框架

二、Express项目构建

npm init -y
npm install express --save
npx express-generator

使用 npx express-generator 命令可以自动创建一个包含所有基本目录和文件的 Express.js 应用模板,这样开发者就可以直接开始编写业务代码,而不需要从零开始设置项目结构。
这个生成的项目通常包括以下部分:

  • bin 目录:存放可执行文件,包括启动应用的脚本。
  • public 目录:用于存放静态文件,如图片、JavaScript 和 CSS 文件。
  • routes 目录:存放路由文件,定义应用的路由逻辑。
  • views 目录:存放视图文件,通常是模板文件,用于生成 HTML 响应。
  • app.js:应用的主入口文件,配置中间件、路由等。
    在这里插入图片描述

三、express框架的基本使用

// 引入express
const express = require('express')
// 创建应用对象
const app = express();app.get('/', function(req, res){// 响应消息内容res.send('全体起立!')
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

在这里插入图片描述
request 和 response 对象的具体介绍:

Request 对象 - request 对象表示 HTTP 请求,包含了请求查询字符串,参数,内容,HTTP 头部等属性。常见属性有:

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

Response 对象 - response 对象表示 HTTP 响应,即在接收到请求时向客户端发送的 HTTP 响应数据。常见属性有:

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.json():传送JSON响应
res.jsonp():传送JSONP响应
res.location():只设置响应的Location HTTP头,不设置状态码或者close response
res.redirect():设置响应的Location HTTP头,并且设置状态码302
res.render(view,[locals],callback):渲染一个view,同时向callback传递渲染后的字符串,如果在渲染过程中有错误发生next(err)将会被自动调用。callback将会被传入一个可能发生的错误以及渲染后的页面,这样就不会自动输出了。
res.send():传送HTTP响应
res.sendFile(path [,options] [,fn]):传送指定路径的文件 -会自动根据文件extension设定Content-Type
res.set():设置HTTP头,传入object可以一次设置多个头
res.status():设置HTTP状态码
res.type():设置Content-Type的MIME类型

四、Express管理用户数据信息

项目目录:
在这里插入图片描述
db.json

{"users": [{"id": 1,"username": "Monica","age": "22"},{"id": 2,"username": "Alen","age": "26"}],"video": []
}
// 引入express
const express = require('express')
const fs = require('fs');
const { json } = require('stream/consumers');
// 创建应用对象
const app = express();app.get('/', function(req, res){fs.readFile('./db.json', 'utf-8',(err, data) => {if (!err) {const back = JSON.parse(data)res.send(back.users)} else {res.status(500).json({err})}})
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

在这里插入图片描述
在这里插入图片描述
可以使用apifox去管理接口
在这里插入图片描述
解决回调地狱问题:

// 引入express
const express = require('express')
const fs = require('fs');
const { promisify } = require('util')
const readFile = promisify(fs.readFile)// 创建应用对象
const app = express();app.get('/', async function(req, res){try {let back = await readFile('./db.json', 'utf-8')res.send(JSON.parse(back))} catch (err) {res.status(500).json({err})}
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

五、处理客户端Post请求数据

// 引入express
const express = require('express')
const fs = require('fs');
const { promisify } = require('util')
const readFile = promisify(fs.readFile)// 创建应用对象
const app = express();
// app.use(express.urlencoded())
app.use(express.json())app.get('/', async function(req, res){try {let back = await readFile('./db.json', 'utf-8')res.send(JSON.parse(back))} catch (err) {res.status(500).json({err})}
})app.post('/', async (req, res) => {console.log(req.headers)console.log(req.body)
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

在这里插入图片描述
app.use(express.urlencoded()) 是 Express 框架中的一个中间件,用于解析来自客户端的 URL 编码的请求体数据,并将其转换为 JavaScript 对象。这通常用于处理 POST 请求中的表单数据。使用这个中间件后,你可以在路由处理程序中通过 req.body 来访问表单数据。
app.use(express.json()) 来解析 JSON 格式的请求体数据

六、添加用户信息到db文件

拿到提交的表单值:

// 引入express
const express = require('express')
const fs = require('fs');
const { promisify } = require('util')
const readFile = promisify(fs.readFile)// 创建应用对象
const app = express();
// app.use(express.urlencoded())
app.use(express.json())app.get('/', async function(req, res){try {let back = await readFile('./db.json', 'utf-8')res.send(JSON.parse(back))} catch (err) {res.status(500).json({err})}
})app.post('/', async (req, res) => {console.log(req.headers)// console.log(req.body)let body = req.bodyif (!body) {res.status(403).json({err: '缺少用户信息'})}let back = await readFile('./db.json', 'utf-8')const jsonObj = JSON.parse(back)// 给输入的用户信息添加个idbody.id = jsonObj.users[jsonObj.users.length-1].id + 1jsonObj.users.push(body)console.log(body);res.send(body)
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

文件写入:

// 引入express
const express = require('express')
const fs = require('fs');
const { promisify } = require('util')
const readFile = promisify(fs.readFile)
const writeFile = promisify(fs.writeFile)// 创建应用对象
const app = express();
// app.use(express.urlencoded())
app.use(express.json())app.get('/', async function(req, res){try {let back = await readFile('./db.json', 'utf-8')res.send(JSON.parse(back))} catch (err) {res.status(500).json({err})}
})app.post('/', async (req, res) => {console.log(req.headers)// console.log(req.body)let body = req.bodyif (!body) {res.status(403).json({err: '缺少用户信息'})}let back = await readFile('./db.json', 'utf-8')const jsonObj = JSON.parse(back)// 给输入的用户信息添加个idbody.id = jsonObj.users[jsonObj.users.length-1].id + 1jsonObj.users.push(body)// 写入文件try{let w = await writeFile('./db.json', JSON.stringify(jsonObj))if(!w){res.status(200).send({msg: '添加成功'})}} catch(err){res.status(500).json({err})}
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

在这里插入图片描述
逻辑抽离:
db.js

const { promisify } = require('util')
const fs = require('fs')
const readFile = promisify(fs.readFile)
const writeFile = promisify(fs.writeFile)exports.getDb = async () => {let data = await readFile('./db.json', 'utf-8')return JSON.parse(data)
}exports.serveDb = async (data) => {return await writeFile('./db.json', JSON.stringify(data))
}

app.js

// 引入express
const express = require('express')
const db = require('./db')// 创建应用对象
const app = express();
// app.use(express.urlencoded())
app.use(express.json())app.get('/', async function(req, res){try {let back = await db.getDb()res.send(back)} catch (err) {res.status(500).json({err})}
})app.post('/', async (req, res) => {// console.log(req.headers)// console.log(req.body)let body = req.bodyif (!body) {res.status(403).json({err: '缺少用户信息'})}let back = await db.getDb()const jsonObj = back// 给输入的用户信息添加个idbody.id = jsonObj.users[jsonObj.users.length-1].id + 1jsonObj.users.push(body)// 写入文件try{let w = await db.serveDb(jsonObj)console.log(111, w)if(!w){res.status(200).send({msg: '添加成功'})}} catch(err){res.status(500).json({err})}
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

七、修改用户信息

// 引入express
const express = require('express')
const db = require('./db')// 创建应用对象
const app = express();
// app.use(express.urlencoded())
app.use(express.json())app.get('/', async function(req, res){try {let back = await db.getDb()res.send(back)} catch (err) {res.status(500).json({err})}
})app.post('/', async (req, res) => {// console.log(req.headers)// console.log(req.body)let body = req.bodyif (!body) {res.status(403).json({err: '缺少用户信息'})}let back = await db.getDb()const jsonObj = back// 给输入的用户信息添加个idbody.id = jsonObj.users[jsonObj.users.length-1].id + 1jsonObj.users.push(body)// 写入文件try{let w = await db.serveDb(jsonObj)if(!w){res.status(200).send({msg: '添加成功'})}} catch(err){res.status(500).json({err})}
})app.put('/:id', async (req, res) => {// console.log(req.params.id) // 传进来的数据// console.log(req.body)try{let userInfo = await db.getDb()let userId = Number(req.params.id)let user = userInfo.users.find(item => item.id === userId)if (!user){res.status(403).send({msg: '用户不存在'})}const body = req.bodyuser.username = body.username ? body.username : user.usernameuser.age = body.age ? body.age : user.ageuserInfo.users[userId - 1] = userlet w = await db.serveDb(userInfo)if (!w) {res.status(201).send({msg: '修改成功'})}} catch(err){res.status(500).json({err})}
})// 监听服务
app.listen(4000, () => {console.log('run http://127.0.0.1:4000');
})

在这里插入图片描述
在这里插入图片描述

八、学到这里,感觉nodeJS还挺有意思的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/5027.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用LocalGPT+cpolar打造可远程访问的本地私有类chatgpt服务

文章目录 前言环境准备1. localGPT部署2. 启动和使用3. 安装cpolar 内网穿透4. 创建公网地址5. 公网地址访问6. 固定公网地址 前言 本文主要介绍如何本地部署LocalGPT并实现远程访问,由于localGPT只能通过本地局域网IP地址端口号的形式访问,实现远程访问…

iOS 实现视图遮罩效果

有时候,我们会遇到这种需求,只讲视图的某个部分展示出来 这时候,我们可以通过设置该视图layer.mask layerb来实现,需要注意的是,这里的layerb必须要设置backgroundColor,渐变layer有colors,否则达不到效果…

Java学习3:程序流程控制

Java程序流程控制 1.执行顺序 顺序结构分支顺序 if,switch 循环结构 for ,while ,do-while 2.if分支 三种形式 if(条件表达式){} else if(){} else{}3.switch分支 string week "周一"; switch(week){case "周一":stem.out.println("周一&qu…

UG NX二次开发(C++)-获取模型中所有的拉伸(Extrude)特征

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 1、前言2、在UG 12中创建几个拉伸特征3、UFun中获取对象类型4、通过NXOpen过渡5.测试结果1、前言 在采用UG NX二次开发时,有时需要在模型中获取特定的对象,比如拉伸特征、关联特征等等。但是通过…

vue2 实现echarts图表进入可视区域后再加载动画,以及 使用了resize之后,动画失效问题解决

Intersection Observer API 是一个现代的浏览器 API,用于监测一个或多个目标元素与其祖先元素或视窗(viewport)之间的交叉状态(intersection)的变化。它可以有效地监听元素是否进入或离开可视区域,从而实现…

【氮化镓】AlGaN/GaN HEMTs沟道温度测量

文章是关于AlGaN/GaN HEMTs(高电子迁移率晶体管)在不同基底(如蓝宝石和硅)上生长时,通过直流(DC)特性方法确定沟道温度的研究。文章由J. Kuzmk, P. Javorka, A. Alam, M. Marso, M. Heuken, 和 …

UWB人员实时定位系统,Spring boot +Vue框架开发的UWB源码

UWB定位技术最核心的优势就是定位精度,可达厘米级,是其它定位技术的成百上千倍,在此精度下,可以满足绝大多数行业精细化管理的需求。 有了精准的位置数据,就可以把人员和物资的轨迹进行数据化还原,通过人工…

爬虫自动调用shell通过脚本运行scrapy爬虫(crawler API)

一、爬虫时如何同时调用shell 1)终端cd项目>>scrapy crawl example 2)打开example.py import scrapy from scrapy.shell import inspect_response#引入shellclass ExampleSpider(scrapy.Spider):name "example"allowed_domains ["example.com"]…

从0开始学习制作一个微信小程序 学习部分(2)json文件的说明与app.json文件的操作

系列文章目录 本文是小程序制作系列的学习篇的第二篇 学习篇第一篇我们讲了编译器下载,项目、环境建立、文件说明与简单操作:第一篇链接 本篇将继续讲解一些基础的编码,分析json文件的作用,着重讲解app.json里可以对小程序进行的切换页面&am…

Blender常见操作

1.局部视图:Local View,也可称作Solo模式,按快捷键 “/”进入,在按退出,只显示选中的物体(可多选),方便编辑 2.物体合并:Ctrl J 其中,当选中多个物体时&am…

二、VLAN原理和配置

vlan不是协议,是一个技术,虚拟局域网技术,基于802.1q协议。 vlan(虚拟局域网),将一个物理的局域网在逻辑上划分成多个广播域的技术。 目录 1.冲突域和广播域 概念 范围 2.以太网帧格式 3.以太网帧封装…

字节5面挂,恶心到了。。。

字节五面 今天脉脉看到一篇帖子: 楼主是 tx 的前员工,在字节五面(加轮)被挂后,认定(或许私下做了一些调查)是字节 HR 向 tx 背调,然后被前同事捏造虚假信息,导致的面试失…

分类规则挖掘(一)

目录 一、分类问题概述(一)分类规则挖掘(二)分类规则评估(三)分类规则应用 二、k-最近邻分类法 一、分类问题概述 动物分类:设有动物学家陪小朋友林中散步,若有动物突然从小朋友身边…

【Linux】yum、vim

🌈个人主页:秦jh__https://blog.csdn.net/qinjh_?spm1010.2135.3001.5343🔥 系列专栏:https://blog.csdn.net/qinjh_/category_12625432.html 目录 Linux 软件包管理器 yum 什么是软件包 查看软件包 如何安装软件 如何卸载软…

机器学习:基于Sklearn、XGBoost框架,使用逻辑回归、支持向量机和XGBClassifier来诊断并预测一个人是否患有自闭症

前言 系列专栏:机器学习:高级应用与实践【项目实战100】【2024】✨︎ 在本专栏中不仅包含一些适合初学者的最新机器学习项目,每个项目都处理一组不同的问题,包括监督和无监督学习、分类、回归和聚类,而且涉及创建深度学…

JAVA系列 小白入门参考资料 继承

目录 1. 为什么需要继承 2. 继承的概念 3. 继承的语法 4. 父类成员访问 4.1 子类中访问父类的成员变量 1. 子类和父类不存在同名成员变量 2. 子类和父类成员变量同名 4.2 子类中访问父类的成员方法 1. 成员方法名字不同 2. 成员方法名字相同 ​5. super关键字 …

Kubernetes学习笔记06

第十六章、Kubernetes容器交付介绍 如何在k8s集群中部署Java项目 容器交付流程 开发代码阶段 编写代码编写Dockerfile【打镜像做准备】持续交付/集成 代码编译打包制作镜像上传镜像仓库应用部署 环境准备PodServiceIngress运维 监控故障排查应用升级 k8s部署Java项目流程 …

使用Github+Picgo+npm实现免费图床

本文参考自 Akilar,原文地址:https://akilar.top/posts/3e956346/ Picgo的配置 Github图床仓库内容不能超过1GB,因为Github原则上是反对仓库图床化的,超过1GB之后会由人工审核仓库内容,如果仓库被发现用来做图床&…

H266 编码标准开源编码器 VVenC 介绍

H266 H.266,也称为VVC(Versatile Video Coding),是最新的国际视频编码标准,由MPEG(Moving Picture Experts Group)和ITU(International Telecommunication Union)联合开…

MySQL CRUD操作

前言👀~ 上一章我们介绍了数据库的一些基础操作,关于如何去创建一个数据库,还有使用数据库,删 除数据库以及对表进行的一些基础操作,今天我们学习CRUD操作 俗称(增删改查) 如果各位对文章的内…