express框架学习笔记

news2025/1/10 9:13:20

express简介

express是一个基于Node.js平台的极简的、灵活的WEB应用开发框架。express是一个封装好的工具包,封装了很多功能,便于我们开发WEB应用(HTTP服务)

express使用

新建express文件夹新建文件test01.js,代码如下

// 导入express
const express = require('express');

// 创建应用对象
const app = express();

// 创建路由
app.get('./home', (req, res) => {
    res.end('hello');
});

// 监听端口,启动服务
app.listen(3000, () => {
    console.log('服务已经启动,端口3000监听中')
})

npm初始化,然后新建express-learn,接着运行node test01.js,报错
在这里插入图片描述

在目录下安装express并将其保存到依赖列表

npm install express --save

运行成功。
只有请求方法是get请求路径是/home的时候回调函数才会实现,请求的url路径是/,所以响应404
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

express路由

路由确定了应用程序如何响应客户端对特定端点的请求

express路由的使用

一个路由由请求方法,路径和回调函数组成

app.<method>(path.callback)

GET请求

app.get('/', (req, res) => {
    res.end('home');
});

POST请求

app.post('/login', (req, res) => {
    res.end('i m login')
})

使用表单发送post请求

   <form method="post" action="http://127.0.0.1:3000">
        <button>登录</button>
    </form>

其他的一些具体案例

//post
app.post('/login', (req, res) => {
    res.end('i m login')
});
// 匹配所有的方法
app.all('/test', (req, res) => {
    res.end('test')
})
// 404响应
app.all('*', (req, res) => {
    res.end('404 not found')
})

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

注意路径名不要添加点,不然可能会出现错误

获取报文数据

express框架封装了一些API来方便请求报文中的数据,并且兼容原生HTTP模块的获取方式。

核心代码

 // 获取查询字符串
    console.log(req.query);
    // 获取指定的请求头
    console.log(req.get('host'));

    res.send('请求报文的获取');

在这里插入图片描述

获取路由参数

商城中的商品的id不同,对应的商品详情页面也不相同,所以使用下面的代码匹配

app.get('/:id.html', (req, res) => {
    // 获取URL路由参数
    console.log(req.params.id);
    res.getHeader('content-type', 'text/html;charset=utf-8');
    res.end('商品详情');
});

两个id名称保持一致。
在这里插入图片描述
在这里插入图片描述

express响应设置

express框架封装了一些API方便客户端响应数据,并且兼容原生HTTP模块的获取方式。
原生方式

app.get("/response", (req, res) => {
    // express中设置响应的方式兼容HTTP模块的方式
    res.statusCode = 404;
    res.statusMessage = 'xxx';
    res.setHeader('abc', 'xyz');
    res.write('响应体');
    res.end('xxx');
    })

express的响应方法

    res.status(500);
    res.set('xxx', 'yyy');
    res.send('中文响应不乱码');
    // 连贯操作
    res.status(404).set('xxx', 'yyy').send('你好朋友')
    // 3其他响应
    res.redirect('http://xxx.com')//重定向
    res.download('./package.json');//下载响应
    res.json();//响应json
    res.sendFile(__dirname + '/home.html')//响应文件内容

json响应可以是下面这样

  res.json({
        name: '隐藏用户',
        rank: 'top1'
    });

将html文件内容响应给网页可以使用

res.sendFile(_dirname+'/test.html')

或者是path.resolve()

中间件

Middleware本质是一个回调函数,中间件函数可以像路由回调一样访问请求对象(request),响应对象response

中间件的作用

使用函数封装公共操作,简化代码

中间件的类型

  • 全局中间件
  • 路由中间件

定义全局中间件

每一个请求到达服务器之后都会执行全局中间件函数

执行下面的函数

app.get('/home', (req, res) => {
    // 获取url和ip
    let { url, ip } = req;
    // 将信息保存在文件中access.log
    fs.appendFileSync(path.resolve(__dirname, './access.log'), `${url}  ${ip}\r\n`);///home 127.0.0.1
    res.send('前台首页');
});

推荐插件Template String Converter,可以在${}输入字符串时候自动生成反引号
在这里插入图片描述
在这里插入图片描述

每个路由规则都要写app.get里面的两行代码,先上车后检查的类似操作后续维护不便,这是我们可以考虑中间件操作

function recordMiddleware(req, res, next) {
    // 获取url和ip
    let { url, ip } = req;
    // 将信息保存在文件中access.log
    fs.appendFileSync(path.resolve(__dirname, './access.log'), `${url}  ${ip}\r\n`);///home 127.0.0.1
    // 调用next
    next();
}
// 使用中间件函数
app.use(recordMiddleware);

req是接收请求报文的对象,res是接收响应报文的对象,next是内部函数,执行后指向路由回调或者中间件回调.

路由中间件实践

需求:针对/admin /setting的请求,要求URL携带code=521的参数,如未携带提示【暗号错误】;
声明中间件函数,并设置在受约束的路由规则当中

// 声明中间件
let checkCodeMiddleware = (req, res, next) => {
    // 判断URL中的是否code参数等于521
    if (req.query.code === '521') {
        // res.send('登录页面');不能每个请求都响应登录页面,这样其他页面都失效了
        next();//满足条件就可以执行后续代码
    } else {
        res.send('暗号错误');
    }
}
app.get('/admin', checkCodeMiddleware, (req, res) => {
    res.send('后台首页');
});
// 后台设置
app.get('/setting', checkCodeMiddleware, (req, res) => {
    res.send('设置页面');
});

在这里插入图片描述

先执行中间件里面的代码,执行next,执行路由回调。路由中间件就可以封装代码。项目中一般使用中间件校验用户身份或者权限。

静态资源中间件

静态资源中间件的设置

// 静态资源中间件设置
app.use(express.static(__dirname + '/public'));

express.static()返回结果是一个中间件函数,参数是静态资源文件夹的路径,服务端到文件夹寻找对应文件,读取文件,响应内容。
在这里插入图片描述

注意事项

  1. express.html文件为默认打开的资源
    在这里插入图片描述

  2. 如果静态资源和路由规则同时匹配,谁先匹配谁就响应

   app.use(express.static(__dirname + '/public'));


   app.get('/', (req, res) => {
       res.send('前台首页');
   });

两个同时匹配上,先匹配前面的就先响应前面的
3. 路由响应动态资源,静态资源中间件响应静态资源

获取请求体数据

核心步骤

  1. 安装
  2. 导入包
  3. 获取中间件函数
  4. 设置路由中间件,然后使用request.body来获取请求体数据

需求

按照要求搭建HTTP服务
get /login 显示表单网页
post /login 获取表单中的用户名和密码

const express = require('express');
const bodyParser = require('body-parser');

const app = express();

app.get('/login', (req, res) => {
    res.send('表单页面')
});

app.post('/login', (req, res) => {
    res.send('获取用户的数据')
});
app.listen(3000, () => {
    console.log('服务已启动')
})

在这里插入图片描述

post请求需要用表单来发起,新建form表单,修改路由规则里面响应html内容

app.get('/login', (req, res) => {
    // res.send('表单页面')
    res.sendFile(__dirname + '/_form.html')
});

在这里插入图片描述

因为响应的表单数据是querystring格式的,所以使用第二个

//解析json格式的请求体的中间件
var jsonParser = bodyParser.json()

// 解析querystrinng格式请求体的中间件
var urlencodedParser = bodyParser.urlencoded({ extended: false })

app.post('/login', urlencodedParser, (req, res) => {
    console.log(req.body);
    res.send('获取用户的数据');
});

防盗链

在网页中选择图片,右键复制图片链接,将其引入html的img里面,打开html,有的显示有的不显示,不显示的就是开启了防盗链。防止外部网站盗用本网站资源。

设置一张图片可以127.0.0.1访问禁止localhost访问

// 声明中间件
app.use((req, res, next) => {
    // 检测请求头中的referer是否为127.0.0.1
    // 获取referer
    let referer = req.get('referer');
    if (referer) {//有refer进入函数内,没有直接下一步
        // 实例化
        let url = new URL(referer);
        // 获取hostname
        let hostname = url.hostname;
        console.log(hostname)
        if (hostname !== '127.0.0.1') {
            res.status(404).send('<h1>404</h1>')
        }
    }
    console.log(referer);
    next();
})
app.use(express.static(__dirname + '/public'));

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

路由模块化

将路由的代码进行模块化处理,将13文件中的前台的路由移动到homeRouter.js中,然后在13文件中导入homeRouter,设置app.use(homeRouter)

主要步骤

  1. 导入express
  2. 创建路由对象
  3. 创建路由规则
  4. 暴露router
    在文件13中引入
const homeRouter = require('./homeRouter')
app.use(homeRouter);

homeRouter的代码

const express = require('express');
const router = express.Router();
router.get('/home', (req, res) => {
    res.send('前台首页');
});
router.get('/search', (req, res) => {
    res.send('内容搜索');
});
module.exports = router;

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

后台也可以路由模块化处理。

EJS模板引擎

模板引擎是分离用户界面和业务数据的一种技术。

EJS是一个高效的JavaScript的模板引擎。

使用方法

下载安装

npm i ejs --save

使用步骤

  1. 下载安装
  2. 引入ejs
  3. 定义数据
  4. ejs解析模板返回结构

ejs列表渲染

const ejs = require('ejs');
let person = ['张三', '李四', '二狗'];
// js
let str = '<ul>';
person.forEach(item => {
    str += `<li>${item}</li>`;
})

str += '</ul>';
console.log(str);
// ejs
let result = ejs.render(`<ul>
<% person.forEach(item =>{ %>
    <li><%= item %></li>
    <% }) %>
    </ul>`, { person: person });

console.log(result)

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

<ul>
        <% person.forEach(item=>{ %>
            <li>
                <%= item %>
            </li>
            <% }) %>
    </ul>

写入html文件,然后编写EJS

const fs = require('fs');
let html = fs.readFileSync('./02ejs.html').toString();
let result = ejs.render(html, { person: person });

console.log(result)

在这里插入图片描述

EJS条件渲染

需求:通过isLogin决定最终的输出内容
true 输出 欢迎回来
false输出登录 注册

// 变量
let isLogin = true;

// 原生js
if (isLogin) {
    console.log('<span>欢迎回来</span>')
} else {
    console.log('<button>登录</button> <button>注册</button>')
}

// ejs
let result = ejs.render(`
    <% if(isLogin){%>
<span>欢迎回来</span>
   <% }else{%>
   <button>登录</button> <button>注册</button>
    <% }%>`,
    { isLogin: isLogin }
);

左边的isLogin和if里面的保持一致,,右边的和变量名称保持一致.
进一步操作,将ejs代码剪切到html中

   <% if(isLogin){%>
        <span>欢迎回来</span>
        <% }else{%>
            <button>登录</button> <button>注册</button>
            <% }%>

在js文件中读取html文件

let html=fs.readFileSync('./_home.html').toString();

运行报错,原因忘记引入fs模块和ejs模块,引入后运行截图
在这里插入图片描述

express中使用ejs

模板文件:具有模板语法的文件,比如之前具有模板语法的html文件

res.render(‘模板的文件名’,‘数据’)

// 导入express
const express = require('express');
// 创建应用对象
const app = express();
//1. 设置模板引擎
app.set('view engine', 'ejs')
// 2.设置模板文件存放位置
app.set('views', path.resolve(__dirname, './views'));


// 创建路由
app.get('/home', (req, res) => {
    //    3.render响应
    //声明变量
    let title = '隐藏用户top1';
    res.render('home', { title });
    // 4.模板文件

});

// 监听端口启动服务
app.listen(3000, () => {
    console.log('服务启动,端口监听中')
})

express-generator

通过应用生成器工具 express-generator 可以快速创建一个应用的骨架。

安装使用

npm install -g express-generator

安装完成后使用express -h可以查看帮助
在这里插入图片描述

将代码下载到一个新的文件夹下面

express -e  文件夹名称

新建的文件夹为generator,在generator里面安装依赖npm i
之后可以使用npm start运行项目
404的两种写法

 app.use(function(req, res, next) {
  next(createError(404));
});
app.all('*', (req, res) => {
    res.end('404 not found')
})

查看文件上传报文

multipart/form-data是文件上传的必需属性。

在这里插入图片描述

npm start运行之后报错,
在这里插入图片描述

原因在于index.js里面没有写post
在这里插入图片描述

机械的敲代码,看走眼了。。。

修改之后页面功能全部正常

处理文件上传

首先安装一个包formidable

npm i formidable

引入formidable然后写入下面的代码

router.post('/portrait', (req, res) => {
  // 创建form对象
  const form = formidable({
    multiples: true,
    // 设置上传文件的保存目录
    uploadDir: __dirname + '/../public/images',
    // 保存文件后缀
    keepExtensions: true

  });

  form.parse(req, (err, fields, files) => {
    if (err) {
      next(err);
      return;
    }
    // console.log(fields);
    // console.log(files);
    // 服务器保存该图片的访问url
    let url = '/images/' + files.portrait.newFilename;//将来将此数据保存在数据库中
    res.send('ok')
    // res.json({ fields, files });
  });
', (req, res) => {
  // 创建form对象
  const form = formidable({
    multiples: true,
    // 设置上传文件的保存目录
    uploadDir: __dirname + '/../public/images',
    // 保存文件后缀
    keepExtensions: true

  });
  form.parse(req, (err, fields, files) => {
    if (err) {
      next(err);
      return;
    }
    // console.log(fields);
    // console.log(files);
    // 服务器保存该图片的访问url
    let url = '/images/' + files.portrait.newFilename;//将来将此数据保存在数据库中
    res.send('ok')
    // res.json({ fields, files });
  });

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

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

相关文章

特征点Features2D类介绍

文章目录 Features2D类介绍1. cv::AgastFeatureDetector2. cv::AKAZE3. cv::BRISK4. cv::FastFeatureDetector5. cv::GFTTDetector6. cv::KAZE7. cv::MSER8. cv::SimpleBlobDetector9. cv::StarDetector10. cv::SIFT11. cv::SURF12. cv::FastFeatureDetector13. cv::AgastFeatu…

leetcode2385. 感染二叉树需要的总时间(java)

感染二叉树需要的总时间 感染二叉树需要的总时间递归 dfs代码演示 二叉树专题 感染二叉树需要的总时间 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/amount-of-time-for-binary-tree-to-be-infected 给你一棵二叉树的…

每日一练 | 华为认证真题练习Day60

1、启用GRE的keepalive功能后&#xff0c;GRE隧道的本端会周期性的每10s向对端发送一次keepalive报文。 A. 对 B. 错 2、AAA协议是RADIUS协议。 A. 对 B. 错 3、路由器Radius信息配置如下&#xff1a;下列说法正确的有&#xff1f;&#xff08;多选&#xff09; A. 计费服务…

[Spring Cloud]:学习笔记·(一)

文章目录 摘要1 认识微服务1.1 单体架构与分布式架构1.2 分布式架构与微服务1.3 微服务架构 摘要 摘要&#xff1a;分布式&#xff1b;微服务&#xff1b;springcloud 1 认识微服务 1.1 单体架构与分布式架构 架构方式解释优点缺点单体架构将业务所有功能集中在一个项目中开…

java从入门到起飞(二)——运算符

目录 前提——运算符概念算数运算符注意事项&#xff1a;字符的“”操作字符串的“”操作 赋值运算符注意事项&#xff1a; 自增自减运算符注意事项&#xff1a; 关系运算符注意事项&#xff1a; 逻辑运算符短路逻辑运算符注意事项&#xff1a; 三元运算符计算规则&#xff1a;…

OpenCV Mat类

文章目录 Mat类Mat类数据类型读取Mat类支持的运算OpenCV Mat数据类型指针ptr的使用多维矩阵创建 Mat类 Mat类数据类型读取 S 有符号整型 U 无符号整型 F 浮点型CV_8U - 8位无符号整数&#xff08;0…255&#xff09;CV_8S - 8位有符号整数&#xff08;-128…127&#xff…

Python基础(1)——Python简介

Python基础&#xff08;1&#xff09;——Python简介 文章目录 Python基础&#xff08;1&#xff09;——Python简介目标Python介绍Python版本总结 目标 了解PythonPython的应用领域Python的版本 Python介绍 Python是时下最流行、最火爆的编程语言之一&#xff0c;具体原因如…

SAP从入门到放弃系列之BOM组-Part1

目录 BOM组两种模式&#xff1a; 创建BOM的方式 方式一&#xff1a;直接在每个工厂分别创建BOM。 方式二&#xff1a;创建BOM组&#xff0c;然后每个工厂参考创建 方式三&#xff1a;创建BOM组&#xff0c;每个工厂参考创建&#xff0c;针对有特殊的工厂复制BOM组后进行调…

人机交互学习-9 以用户为中心的设计

以用户为中心的设计 以用户为中心的设计思想工程设计过程的三个假设以人为本设计思想三个方面的假设以用户为中心的设计四原则UDC项目包含的方法 用户参与设计用户参与的重要性选择哪些用户&#xff1f;为什么要让用户参与&#xff1f; 用户参与的形式参与式设计PICTIVECARD 理…

Nacos-手写注册中心基本原理

本文已收录于专栏 《中间件合集》 目录 概念说明需求分析核心功能代码实现AService模块BService模块NacosService模块NacosSDK模块 注意事项总结提升 概念说明 注册中心是微服务架构中的纽带&#xff0c;类似于“通讯录”&#xff0c;它记录了服务和服务地址的映射关系。在分布…

香蕉派(Banana Pi) BPI-M2 Zero 评测试,与树莓派 Zero同尺寸的开发板

Banana Pi M2 Zero 是一款微型计算机&#xff0c;配备四核处理器并内置 Wi-Fi 和蓝牙。这是一款非常适合基本计算任务甚至轻度游戏的小型设备。在这篇评论中&#xff0c;我们将了解 M2 Zero 的性能、功能和价值。 什么是 Banana Pi BPI-M2 Zero Banana Pi M2 Zero 是由深圳公…

MongoDB(Windows版)安装

首先需要下载 官网&#xff1a;MongoDB: The Developer Data Platform | MongoDB 安装过程 需要安装的版本 第一步&#xff1a;安装时&#xff0c;Custom是指可以自定义安装路径&#xff0c;然后傻瓜式安装即可&#xff08;注意&#xff1a;先不要安装图形化工具&#xff0…

2023-06-17 LeetCode每日一题(分割圆的最少切割次数)

2023-06-17每日一题 一、题目编号 2481. 分割圆的最少切割次数二、题目链接 点击跳转到题目位置 三、题目描述 圆内一个 有效切割 &#xff0c;符合以下二者之一&#xff1a; 该切割是两个端点在圆上的线段&#xff0c;且该线段经过圆心。该切割是一端在圆心另一端在圆上…

【Vue】学习笔记-Vue Router activated deactivated 路由守卫

Vue Router activated deactivated 路由守卫 activated deactivated路由守卫1.全局守卫2.独享守卫3.组件内守卫全局路由守卫路由器的两种工作模式 activated deactivated activated 和 deactivated 是路由组件所独有的两个钩子&#xff0c;用于捕获路由组件的激活状态 具体使用…

管理类联考——英语——趣味篇——阅读——考题的来源

Part One考研英语阅读——Part A 1.卫报 《卫报》( The Guardian)是英国的全国性综合内容日报。与《泰晤士报》、《每日电讯报》被合称为英国三大报。由约翰爱德华容泰勒创办于1821年5月5日。该报注重报道国际新闻&#xff0c;擅长发表评论和分析性专题文章。一般公众视《卫报…

【数据分析之道-Matplotlib(九)】Matplotlib棉棒图

文章目录 专栏导读1、Matplotlib棉棒图stem()基本语法2、Matplotlib棉棒图stem()定义样式2.1linefmt参数2.2markerfmt参数2.3举例一&#xff1a;直线样式2.4举例二&#xff1a;圆点样式 3、棉棒图案例实战3.1绘制每月销量的棉棒图3.2绘制每月销量与平均销量之差 专栏导读 ✍ 作…

Prometheus介绍安装和快速入门

Prometheus介绍安装和快速入门 1、Prometheus介绍 1.1 什么是 Prometheus? Prometheus&#xff08;普罗米修斯&#xff09;是古希腊的一个神明&#xff0c;名字的意思是「先见之明」。从它的名字可以看出&#xff0c; Prometheus 是做「先见之明」的监控告警用途。维基百科…

HJ26 字符串排序

题目&#xff1a; HJ26 字符串排序 题解&#xff1a; 规则 1 &#xff1a;英文字母从 A 到 Z 排列&#xff0c;不区分大小写。 统一转换&#xff0c;通过减去对应字母的起始值&#xff0c;得到一个相对值&#xff0c;抹平大小写&#xff0c;例如&#xff1a;B - A&#xff…

【Java项目】使用LruCache提高DSP广告主需求方 (Demand Side Platform)系统性能

文章目录 背景LruCache简介LruCache在DSP系统中的应用场景LruCacheRedis增加LruCache数据过期清除机制ConcurrentHashMapLruCache零拷贝机制 源码 背景 我之前工作的一家公司是一家传媒公司&#xff0c;公司的主要盈利方式为在公司项目中接入广告&#xff0c;以及自媒体广告宣…

Windows下编译安装Acise

Acise是济南友泉软件公司自主研发的一套跨平台的通用CAx(CAD/CAE)软件开发框架&#xff0c;本文旨在记录Windows下编译安装Acise的流程。 零、系统环境 操作系统Windows 10编译器Visual Studio 2019 CommunityCMake2.24.2Boost1.80.0Qt5.14.0OpenCASCADE7.6.0VTK9.0.0 一、依…