Express

Express 框架

第一章 Express简介

npm提供了大量的第三方模块,其中不乏许多Web框架,我们没有必要重复发明轮子,因而选择使用 Express作为开发框架,因为它是目前最稳定、使用最广泛,而且Node.js官方推荐的唯一一个 Web开发框架。
Express是一个轻量级的Web框架,多数功能只是对HTTP协议中常用操作的封装,更多的功能需要插件或者整合其他模块来完成.

Express中文官网 http://www.expressjs.com.cn/

第二章 初始化一个Express应用

  1. 我们想要安装Express,首先需要安装Epxress生成器,在全局通过命令安装
npm install express-generator -g
  1. 等待安装完毕,查看帮助信息:
express --help

可以看到,

Usage: express [options] [dir]

我们可以用express 在当前目录创建dir,并且依据配置选项
我们可以通过option设置express使用的模板,这里我们用ejs,也可以设置css的引擎,比如less和sass,这里我们先使用原生的css
所以只需要输入

express myapp --view ejs

express会给我们快速的搭建一个app框架,然后根据提示进入myapp文件夹,运行

npm install

npm会自动帮助我们安装当前应用所需要的模块,等待安装完毕之后,启动:

npm start

浏览器打开:"localhost:3000",可以看到express框架给我返回的信息。

在调试过程中,如果我们不想反复的重启服务端,可以npm安装nodemon,使用nodemon启动express项目

第三章 目录分析

回到我们安装express之前,看一下express快速搭建的目录结构:

 create : taobao
   create : taobao/package.json //依赖关系文件
   create : taobao/app.js // 入口文件
   create : taobao/public //公共资源文件夹
   create : taobao/public/images
   create : taobao/public/stylesheets
   create : taobao/public/stylesheets/style.css
   create : taobao/public/javascripts
   create : taobao/views //视图文件夹
   create : taobao/views/index.ejs
   create : taobao/views/error.ejs
   create : taobao/routes //路由文件夹
   create : taobao/routes/index.js
   create : taobao/routes/users.js
   create : taobao/bin
   create : taobao/bin/www //配置文件

由此可见,express是一个标准的MVC框架,通过路由分配视图,并且可以在路由中写上我们的控制结构.
修改 routes下的index.js

router.get('/', function(req, res, next) {
  res.render('index', { title: '你好express' });
});

发现我们的页面出现了改变,这个路由跟我们自己写的路由作用是差不多的,具体用法我们后边再谈.

第四章 模板的使用方式

我们在安装express的时候,使用的模板是ejs,具体什么是模板呢?ejs又怎么用呢?
简单来讲,模板就是帮助我们在view层进行输出的工具.我们打开view下的index.ejs,可以看到,这个模板其实就是在ejs下写了我们常见的html代码,只是多了几个不一样的标签:

<!DOCTYPE html>
<html>
  <head>
    <title><%= title %></title>
    <link rel='stylesheet' href='/stylesheets/style.css' />
  </head>
  <body>
    <h1><%= title %></h1>
    <p>Welcome to <%= title %></p>
  </body>
</html>

<%= title %>可以帮助我们将路由中的{ title: '你好express' }输出到模板当中,其中的render就是模板渲染的方法,帮助我们渲染第一个参数index所对应的模板.类似angular,我们也可以在标签内部写自己的表达式

<h1><%= title.toUpperCase() %></h1>

4.1 模板引擎的用法

ejs有三种常用的标签

  1. <% code %>:运行 JavaScript 代码,不输出
  2. <%= code %>:显示转义后的 HTML内容
  3. <%- code %>:显示原始 HTML 内容

注意:<%= code %><%- code %>都可以是JavaScript表达式生成的字符串,当变量code为普通字符串时,两者没有区别。当 code 比如为 <h1>hello</h1> 这种字符串时,<%= code %> 会原样输出 <h1>hello</h1>,而 <%- code %> 则会显示H1大的hello字符串

下面的例子可以帮我们解释<% code %>的用法:

index.ejs

<!DOCTYPE html>
<html>
  <head>
    <title><%= title %></title>
    <link rel='stylesheet' href='/stylesheets/style.css' />
  </head>
  <body>
    <h1><%= title %></h1>
    <ul>
    <% for(var i=0; i<contents.length; i++) { %>
        <li><%= contents[i] %></li>
    <% } %>
    </ul>
  </body>
</html>

index.js

router.get('/', function(req, res, next) {
    res.render('index',{ 
        title: 'express',
        contents:['AngularJS','Node.js','Express']
    });
});

查看渲染之后的结果.
同样,如果你想要动态的调整js代码,可以将js代码在<script>标签中按照原样渲染.

4.3 渲染模板的方法

我们刚刚看到,routes下的index.js使用render可以将后边的参数渲染到模板之中,但是,我们在views下无法看到users.ejs,这是因为在user.js下使用了send方法,可以将后边的信息直接渲染而不通过模板.

4.4 includes

我们使用模板引擎通常不是一个页面对应一个模板,这样就失去了模板的优势,而是把模板拆成可复用的模板片段组合使用,如在 views 下新建 header.ejs 和 footer.ejs,并修改users.ejs

views/header.ejs

<!DOCTYPE html>
<html>
  <head>
    <style type="text/css">
      body {padding: 50px;font: 14px "Lucida Grande", Helvetica, Arial, sans-serif;}
    </style>
  </head>
  <body>

views/footer.ejs

  </body>
</html>

views/users.ejs

<%- include('header') %>
  <h1><%= name.toUpperCase() %></h1>
  <p>hello, <%= name %></p>
<%- include('footer') %>

routes/users.js

router.get('/', function(req, res, next) {
  // res.send('respond with a resource');
  res.render('users',{ 
        name: 'express'
    });
});

访问'localhost:3000/users'可以看到渲染后的结果.
我们将users.ejs拆成出了header.ejs和footer.ejs,并在users.ejs通过 ejs 内置的 include方法引入,从而实现了跟以前一个模板文件相同的功能.

注意:要用 <%- include('header') %> 而不是 <%= include('header') %>

第五章 路由

我们在路由中将数据渲染并且分配到模板当中,那么我们如何写一个路由呢?

5.1 入口文件

首先,我们先要了解入口文件的组成,打开app.js,我们可以看到

var express = require('express');
var path = require('path');
var favicon = require('serve-favicon');
var logger = require('morgan');
var cookieParser = require('cookie-parser');
var bodyParser = require('body-parser');

这几句话表示我们引入的模块

var index = require('./routes/index');
var users = require('./routes/users');

路由都是模块写法,而入口文件将这些路由引入,我们就可以直接使用了.

app.use('/', index);
app.use('/users', users);

这则使用了我们的路由

// view engine setup
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'ejs');

这两句话设置了我们的模板引擎,如果我们想要使用sass或者less等css引擎,同样可以向下追加.

// catch 404 and forward to error handler
...
// error handler
...

剩下的部分则是规定了404和500错误下的页面渲染方法.
由此,我们可以在routes下添加的路由文件,并且在入口文件引入

5.2 新增路由

在routes下添加新的路由:
routers/tests.js

var express = require('express');
var router = express.Router();
router.get('/', function(req, res, next) {
   res.send('这是新的路由');
});
module.exports = router;

同样,在入口文件添加
app.js

var tests = require('./routes/tests');
app.use('/tests', tests);

访问'localhost:3000/tests'则可以获取我们新增加的路由信息.

5.3 路由中的参数

其中 req,res是我们熟悉的请求和响应对象,当我们的地址栏访问模板的时候,可以获取参数

ruoter.get('/:name', function(req, res, next) {
  res.send('hello, ' + req.params.name);
});

我们在地址栏请求'localhost:3000/tests/Y18',我们可以看到响应信息

hello, Y18

不难看出:req 包含了请求来的相关信息,res 则用来返回该请求的响应。下面介绍几个常用的

req 的属性:

  • req.query: 解析后的 url 中的 querystring,如 ?name=haha,req.query 的值为 {name: 'haha'}
  • req.params: 解析 url 中的占位符,如 /:name,访问 /haha,req.params 的值为 {name: 'haha'}
  • req.body: 解析后请求体,需使用相关的模块,如 body-parser,请求体为 {"name": "haha"},则 req.body 为 {name: 'haha'},我们可以用接受post请求为例.
  • 注意,无论是send或者render,我们只能获取第一次符合标准的渲染页面

5.4 next和中间件

前面我们讲解了express中路由和模板引擎ejs的用法,但express的精髓并不在此,在于中间件的设计理念。

我们可以看到,在路由的回调函数中,有一个参数next,这就是我们使用中间件最重要的部分。
express中的中间件(middleware)是用来处理请求的,当一个中间件处理完,可以通过调用 next() 传递给下一个中间件,如果没有调用 next(),则请求不会往下传递,如内置的 res.render 其实就是渲染完 html 直接返回给客户端,如果调用了next(),那么请求会继续向下传递,访问路由组中的下一个路由。

next中间件分为应用级中间件和路由级中间件

应用级中间件

我们回头再来看一下入口文件:

app.use(logger('dev'));
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({ extended: false }));
app.use(cookieParser());
app.use(express.static(path.join(__dirname, 'public')));

app.use('/', index);
app.use('/users', users);
app.use('/tests', tests);

其实,这些app.use()也是给我们加载中间件,只是这些中间件是在应用中使用的,应用级中间件绑定到 app 对象 使用 app.use() 和 app.METHOD(), 其中, METHOD 是需要处理的 HTTP 请求的方法,例如 GET, PUT, POST 等等,全部小写。我们自定义一下全局中间件:

// 挂载到全局的中间件,任何请求都会执行它
app.use(function (req, res, next) {
  console.log('Time:', Date.now());
  next();
});
// 挂载至 /hello/:name 的中间件,任何指向 /hello/:name 的请求都会执行它
app.use('/hello/:name', function (req, res, next) {
  console.log('Request Type:', req.method);
  next();
});
// 路由和句柄函数(中间件系统),处理指向 /hello/:name 的 GET 请求
app.get('/hello/:name', function (req, res, next) {
  res.send('hello '+ req.params.name);
});

如果我们不指定路径,则默认匹配所有路径,我们在最后一个路由中并没有使用next,所以请求没有向下传递.
我们在上方引入的其余全局中间件,即第三方中间件,其实也在回调函数的末尾使用了next方法,使得请求能够继续向下传递.

在我们访问'localhost:3000/hello/Y18',可以在控制台和页面都能看到数据返回.

注意:

作为中间件系统的路由句柄,使得为路径定义多个路由成为可能。

// 一个中间件栈,处理指向 /user/:id 的 GET 请求
app.get('/hello/:id', function (req, res, next) {
  console.log('ID:', req.params.id);
  next();
}, function (req, res, next) {
  res.send('hello Info');
});

// 处理 /user/:id, 打印出用户 id
app.get('/hello/:id', function (req, res, next) {
  res.end(req.params.id);
});

但是由于第一个路由已经终止了请求-响应循环,第二个路由虽然不会带来任何问题,但却永远不会被调用.
如果需要在中间件栈中跳过剩余中间件,调用 next('route') 方法将控制权交给下一个路由

// 一个中间件栈,处理指向 /user/:id 的 GET 请求
app.get('/hello/:id', function (req, res, next) {
  console.log('ID:', req.params.id);
  if(req.params.id == 4) next('route');
  else next();
}, function (req, res, next) {
  res.send('hello Info');
});

// 处理 /user/:id, 打印出用户 id
app.get('/hello/:id', function (req, res, next) {
  res.end(req.params.id);
});

路由级中间件

路由级中间件和应用级中间件一样,只是它绑定的对象为 express.Router()

修改index.js

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

/* GET home page. */
router.get('/:name', function(req, res, next) {
    // 如果 user id 为 0, 跳到下一个路由
    if (req.params.name != 'Y18') next('route');
    // 否则将控制权交给栈中下一个中间件
    else next(); 
}, function (req, res, next) {
    // 渲染常规页面
    res.send('你别来了,Y18');
});
// 处理name为Y18的路由, 渲染一个特殊页面
router.get('/:name', function (req, res, next) {
  res.send('欢迎你~ '+ req.params.name);
});

module.exports = router;

错误处理中间件

当我们访问一个不存在的路由时,虽然我们最后一个路由没有写next中间件,但是错误路由依旧执行,查看源码:

https://github.com/expressjs/express/blob/master/lib/router/index.js

function next(err) {
    ... //此处源码省略
    // find next matching layer
    var layer;
    var match;
    var route;

    while (match !== true && idx < stack.length) {
      layer = stack[idx++];
      match = matchLayer(layer, path);
      route = layer.route;

      if (typeof match !== 'boolean') {
        // hold on to layerError
        layerError = layerError || match;
      }

      if (match !== true) {
        continue;
      }
      ... //此处源码省略
    }
  ... //此处源码省略
    // this should be done for the layer
    if (err) {
        layer.handle_error(err, req, res, next);
    } else {
      layer.handle_request(req, res, next);
    }
  }

next函数内部有个while循环,每次循环都会从stack中拿出一个layer,这个layer中包含了路由和中间件信息,然后就会用layer和请求的path就行匹配,如果匹配成功就会执行layer.handle_request,调用中间件函数。但如果匹配失败,就会循环下一个layer(即中间件)。

express有成百上千的第三方中间件,在开发过程中我们首先应该去npm上寻找是否有类似实现的中间件,尽量避免造轮子,节省开发时间。下面给出几个常用的搜索 npm 模块的网站:

  1. http://npmjs.com(npm 官网)
  2. http://node-modules.com
  3. https://npms.io
  4. https://nodejsmodules.org

第六章 在express中使用less或sass

如果我们想要使用sass或者less,首先要对其进行学习;

Express支持sass,less等css引擎,我们现在使用sass编译模板的样式:
重新安装一个express应用:express app --view ejs -c sass
我们可以看到在入口文件中看到变化:

app.use(require('node-sass-middleware')({
  src: path.join(__dirname, 'public'),
  dest: path.join(__dirname, 'public'),
  indentedSyntax: true,
  sourceMap: true
}));

这其实就是引入了node的sass中间件。
同时,在public/stylesheets中可以看到style.sass这个sass文件。
启动express,访问'localhost:8000/'我们可以看到在这个文件夹中生成了两个新的文件:
'style.css'
'style.css.map'
这就是我们熟悉的sass经过编译后的文件。
同样,我们可以写自己的sass文件,决定样式:
修改style.sass

$fontStack:    Helvetica, sans-serif;
$primaryColor: #8A2BE2;

body
    font-family: $fontStack;
    padding: 50px;
p
  color: $primaryColor;

ul li
     color: #BDB76B;

a
  text-decoration: none;

修改index.ejs

<!DOCTYPE html>
<html>
  <head>
    <title><%= title %></title>
    <link rel='stylesheet' href='/stylesheets/style.css' />
  </head>
  <body>
    <h1><%= title %></h1>
    <p>Welcome to <%= title %></p>
    <ul>
        <li>AngularJS</li>
        <li>Node.js</li>
        <li>Express</li>
    </ul>
    <a id="test" href="www.zixue.it">自学it网</a>
  </body>
</html>

访问'localhost:3000',可以看到我们修改之后的样式.

我们可以修改indentedSyntax改为false,实现scss方式的书写

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

推荐阅读更多精彩内容