Nodejs Express4.x开发框架

开发环境:

Ubuntu

MonogoDB: v2.6.4

nodejs:v0.11.2

npm 2.1.10 ( 如果nodejs安装的时候是1.2.19版本,本文升级到了2.x版本)

1. 建立工程

进入工程目录

mkdir workplace

cd workplace

全局安装express,express作为命令被安装到了系统中.

npm install -g express

查看express版本

express -V

注意:在express4.x版本中已经不含有express命令了。

需要安装 express-generator

npm install express-generator -g

详细安装过程见:《准备Nodejs开发环境Ubuntu》

使用express命令创建工程,并支持ejs

hadoop@sven:~/workspace/nodeJs$express -e nodejs-demo

create : nodejs-demo (项目名)

create : nodejs-demo/package.json (项目包的信息)

create : nodejs-demo/app.js (主程序)

create : nodejs-demo/public (公开目录)

create : nodejs-demo/public/images

create : nodejs-demo/public/javascripts

create : nodejs-demo/public/stylesheets

create : nodejs-demo/public/stylesheets/style.css

create : nodejs-demo/routes (路由目录,以后在这个目录下开发程序,然后在app.js里use)

create : nodejs-demo/routes/index.js

create : nodejs-demo/routes/users.js

create : nodejs-demo/views (视图目录,视图模板文件等)

create : nodejs-demo/views/index.ejs

create : nodejs-demo/views/error.ejs

create : nodejs-demo/bin

create : nodejs-demo/bin/www (启动文件,用于启动app.js)

install dependencies:

$ cd nodejs-demo && npm install

run the app:

$ DEBUG=nodejs-demo ./bin/www

项目创建成功。

根据上述提示安装依赖:

 
 
  1. cd nodejs-demo && npm install

根据提示启动web:

 
 
  1. $ DEBUG=nodejs-demo ./bin/www

不过我们这里不打算使用该方法启动程序。原因是我们在开发过程中需要使用nodemon这么一个工具。

nodemon用于动态识别开发过程中项目的改变,然后动态加载(这是Eclipse种开发java web类似)。该工具是开发web的必备啊

安装nodemon:

npm install nodemon -g

那么为什么我们上面不使用./bin/www脚本呢?

原因是nodemon ./bin/www 这样是没有办法识别项目的改动。(我个人实验的,如有知道的大牛,望赐教)

修改app.js:

把最有一行//module.exports = app;注释掉

换成:app.listen(3000);

使用下面命令启动app.js主程序:

hadoop@sven:~/workspace/nodeJs/nodejs-demo$nodemon app.js

然后修改程序,看看是否会动态加载。会有下面提示:

1 Dec 16:22:07 – [nodemon] restarting due to changes…

1 Dec 16:22:07 – [nodemon] starting `node app.js`

表示生效。

测试:

本地的3000端口被打开,通过浏览器访问: localhost:3000

2. 目录结构

./

drwxrwxr-x 5 hadoop hadoop 4096 12月 1 15:57 ../

-rw-rw-r– 1 hadoop hadoop 1495 12月 1 16:22 app.js

-rw-r–r– 1 hadoop hadoop 12288 12月 1 16:22 .app.js.swp

drwxr-xr-x 2 hadoop hadoop 4096 12月 1 15:57 bin/

drwxrwxr-x 9 hadoop hadoop 4096 12月 1 15:58 node_modules/

-rw-rw-r– 1 hadoop hadoop 326 12月 1 15:57 package.json

drwxr-xr-x 5 hadoop hadoop 4096 12月 1 15:57 public/

drwxr-xr-x 2 hadoop hadoop 4096 12月 1 15:57 routes/

drwxr-xr-x 2 hadoop hadoop 4096 12月 1 15:57 views/

目录介绍:

node_modules, 存放所有的项目依赖库。(每个项目管理自己的依赖,与Maven,Gradle等不同)

package.json,项目依赖配置及开发者信息

app.js,程序主入口

public,静态文件(css,js,img)

routes,路由文件(MVC中的C,controller)

Views,页面文件(Ejs模板)

nodejs-demo/bin/www (启动文件,用于启动app.js)

打开app.js查看内容:

 
 
  1. /**
  2. * 模块依赖
  3. */
  4. var express = require('express')
  5. , routes = require('./routes')
  6. , user = require('./routes/user')
  7. , http = require('http')
  8. , path = require('path');
  9. var app = express();
  10. //环境变量
  11. app.set('port', process.env.PORT || 3000);
  12. app.set('views', __dirname + '/views');
  13. app.set('view engine', 'ejs');
  14. app.use(express.favicon());
  15. app.use(express.logger('dev'));
  16. app.use(express.bodyParser());
  17. app.use(express.methodOverride());
  18. app.use(app.router);
  19. app.use(express.static(path.join(__dirname, 'public')));
  20. // 开发模式
  21. if ('development' == app.get('env')) {
  22. app.use(express.errorHandler());
  23. }
  24. // 路径解析
  25. app.get('/', routes.index);
  26. app.get('/users', user.list);
  27. // 启动及端口
  28. http.createServer(app).listen(app.get('port'), function(){
  29. console.log('Express server listening on port ' + app.get('port'));
  30. });

4. Ejs模板使用

让ejs模板文件,使用扩展名为html的文件。

修改:app.js

var ejs = require('ejs'); //引入ejs。重新安装依赖>

app.engine('.html', ejs.__express);

app.set('view engine', 'html');  // app.set('view engine', 'ejs');

重命名:views/*.ejs 为 views/*.html

访问localhost:3000正确

主要要重命名index.ejs等文件

5. 增加Bootstrap界面框架

其实就是把js,css文件复制到项目中对应该的目录里。 包括4个文件:

复制到public/stylesheets目录

bootstrap.min.css

bootstrap-responsive.min.css

复制到public/javascripts目录

bootstrap.min.js

jquery-1.9.1.min.js

接下来,我们把index.html页面切分成3个部分:header.html, index.html, footer.html

header.html, 为html页面的头部区域

index.html, 为内容显示区域

footer.html,为页面底部区域

header.html

 
 
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="utf-8">
  5. <title><%=: title %></title>
  6. <!-- Bootstrap -->
  7. <link href="http://www.geedoo.info/stylesheets/bootstrap.min.css" rel="stylesheet" media="screen">
  8. <!-- <link href="http://www.geedoo.info/css/bootstrap-responsive.min.css" rel="stylesheet" media="screen"> -->
  9. </head>
  10. <body screen_capture_injected="true">
  11. index.html
  12. <% include header.html %>
  13. <h1><%= title %></h1>
  14. <p>Welcome to <%= title %></p>
  15. <% include footer.html %>
  16. footer.html
  17. <script src="http://www.geedoo.info/javascripts/jquery-1.9.1.min.js"></script>
  18. <script src="http://www.geedoo.info/javascripts/bootstrap.min.js"></script>
  19. </body>
  20. </html>

访问localhost:3000正确。

我们已经成功的使用了EJS模板的功能,把公共的头部和底部从页面中分离出来了。

并已经引入了bootstrap界面框架。

6. 路由功能

我们设计一下用户登陆业务需求。

访问路径:/,页面:index.html,不需要登陆,可以直接访问。

访问路径:/home,页面:home.html,必须用户登陆后,才可以访问。

访问路径:/login,页面:login.html,登陆页面,用户名密码输入正确,自动跳转到home.html

访问路径:/logout,页面:无,退出登陆后,自动回到index.html页面

打开app.js文件,在增加路由配置

 
 
  1. app.get('/',routes.index);
  2. app.route('/login')
  3. .get(routes.login)
  4. post(routes.doLogin);
  5. app.get('/logout',routes.logout);
  6. app.get('/home',routes.home);

注:get为get请求,post为post请求,all为所有针对这个路径的请求

我们打开routes/index.js文件,增加对应的方法。

 
 
  1. exports.index=function(req, res) {
  2.  res.render('index', { title: 'index' });
  3. };
  4. exports.login=function(req,res){
  5.  res.render('login',{title: '用户登录'});
  6. };
  7. exports.doLogin=function(req,res){
  8.  var user = {
  9.  username:"admin",
  10.  password:"admin"
  11. }
  12. if(req.body.username==user.username && req.body.password==user.password){
  13.  res.redirect('/home');
  14. }
  15. res.redirect('/login');
  16. };
  17. exports.logout = function(req,res){
  18.  res.redirect('/');
  19. };
  20. exports.home = function(req,res){
  21.  var user = {
  22.  username:'admin',
  23.  password:'admin'
  24.  }
  25.  res.render('home',{title:'Home',user:user});
  26. };

创建views/login.html和views/home.html两个文件

login.html

 
 
  1. <% include header.html %> 
  2. <div class="container-fluid">
  3. <form class="form-horizontal" method="post">
  4. <fieldset>
  5. <legend>用户登陆</legend>
  6. <div class="control-group">
  7. <label class="control-label" for="username">用户名</label>
  8. <div class="controls">
  9. <input type="text" class="input-xlarge" id="username" name="username">
  10. </div>
  11. </div>
  12. <div class="control-group">
  13. <label class="control-label" for="password">密码</label>
  14. <div class="controls">
  15. <input type="password" class="input-xlarge" id="password" name="password">
  16. </div>
  17. </div>
  18. <div class="form-actions">
  19. <button type="submit" class="btn btn-primary">登陆</button>
  20. </div>
  21. </fieldset>
  22. </form>
  23. </div>
  24. <% include footer.html %>
  25. home.html:
  26. <% include header.html %>
  27. <h1>Welcome <%= user.username %>, 欢迎登陆!!</h1>
  28. <a claa="btn" href="http://www.geedoo.info/logout">退出</a>
  29. <% include footer.html %>

修改index.html,增加登陆链接

index.html

 
 
  1. <% include header.html %>
  2. <h1>Welcome to <%= title %></h1>
  3. <p><a href="http://www.geedoo.info/login">登陆</a></p>
  4. <% include footer.html %>

路由及页面我们都写好了,快去网站上试试吧。

7. Session使用

从刚来的例子上面看,执行exports.doLogin时,如果用户名和密码正确,我们使用redirect方法跳转到的home

res.redirect('/home');

执行exports.home时,我们又用render渲染页面,并把user对象传给home.html页面

res.render('home', { title: 'Home',user: user});

为什么不能在doLogin时,就把user对象赋值给session,每个页面就不再传值了。

session这个问题,其实是涉及到服务器的底层处理方式。

像Java的web服务器,是多线程调用模型。每用户请求会打开一个线程,每个线程在内容中维护着用户的状态。

像PHP的web服务器,是交行CGI的程序处理,CGI是无状态的,所以一般用cookie在客户的浏览器是维护用户的状态。但cookie在客户端维护的信息是不够的,所以CGI应用要模仿用户session,就需要在服务器端生成一个session文件存储起来,让原本无状态的CGI应用,通过中间文件的方式,达到session的效果。

Nodejs的web服务器,也是CGI的程序无状态的,与PHP不同的地方在于,单线程应用,所有请求都是异步响应,通过callback方式返回数据。如果我们想保存session数据,也是需要找到一个存储,通过文件存储,redis,Mongdb都可以。

接下来,我将演示如何通过mongodb来保存session,并实现登陆后用户对象传递。

app.js文件

在相应位置添加下面代码:

 
 
  1. var session = require('express-session');
  2. var connect = require('connect');
  3. var SessionStore = require("session-mongoose")(connect);
  4. var store = new SessionStore({
  5. url:"mongodb://localhost/session",
  6.  interval: 120000
  7. });
  8. app.use(session({
  9.  secret: 'test.com',
  10.  store: store,
  11.  cookie:{maxAge:10000} //expire session in 10 seconds
  12. }));
  13. //用于把登录用户设置到res.locals里面,在home.html里显示
  14. app.use(function(req,res,next){
  15.  res.locals.user = req.session.user;
  16.  console.log('Session is = ',req.session.user);
  17.  next();
  18. });

需要添加中间件connect、session-mongoose。

其中session-mongoose是用于连接mongodb数据库。然后自动写入session信息到数据库中(也可以用mongoose中间件,但是要自己实现session的写入)

app.use(session(….)) 这里是全局设置了session的信息及session信息存储的方式。

注:app.js文件有顺序要求,一定要注意!!!

安装session-mongoose依赖库

npm install connect

npm install session-mongoose

npm install session-mongoose

安装有错误但是没关系。

访问:http://localhost:3000/login,正常

修改routes/index.js文件

exports.doLogin方法

 
 
  1. exports.doLogin = function(req, res){
  2.  var user={ username:'admin', password:'admin' } 
  3.  if(req.body.username===user.username && req.body.password===user.password){ 
  4.   req.session.user=user; 
  5.   return res.redirect('/home');
  6.  } else { 
  7.   return res.redirect('/login'); 
  8.  }
  9. };

exports.logout方法

 
 
  1. exports.logout = function(req, res){
  2.  req.session.user=null;
  3.  res.redirect('/');
  4. };

exports.home方法

 
 
  1. exports.home = function(req, res){
  2.  res.render('home', { title: 'Home'});
  3. };

这个时候session已经起作用了,exports.home的user显示传值已经被去掉了。 是通过app.js中app.use的res.locals变量,通过框架进行的赋值。

 
 
  1. app.use(function(req, res, next){
  2.  res.locals.user = req.session.user;
  3.  next();
  4. });

注:这个session是express4.10.4的写法,与express4之前的版本是不一样的。

访问页面可以查看mongodb有没有数据:

nodejs-mongodb

nodejs-mongodb

由于上面配置的 cookie:{maxAge:10000} //expire session in 10 seconds

过期时间,因此你会看到mongodb里面的数据过一段时间就被清除了。

参考:

Mongoose:http://mongoosejs.com/

关于express4.2.0与express3.x操作的区别:http://blog.csdn.net/u013758116/article/details/38758351

8. 页面提示

登陆的大体我们都已经讲完了,最后看一下登陆失败的情况。

我们希望如果用户登陆时,用户名或者密码出错了,会给用户提示,应该如何去实现。

打开app.js的,增加res.locals.message

登陆的大体我们都已经讲完了,最后看一下登陆失败的情况。

我们希望如果用户登陆时,用户名或者密码出错了,会给用户提示,应该如何去实现。

打开app.js的,增加res.locals.message

 
 
  1. app.use(function(req, res, next){
  2.  res.locals.user = req.session.user;
  3.  var err = req.session.error;
  4.  delete req.session.error;
  5.  res.locals.message = '';
  6.  if (err) res.locals.message = '<div class="alert alert-danger">' + err + '</div>';
  7.  next();
  8. });

修改login.html页面,<%- message %>

 
 
  1. <% include header.html %>
  2. <div class="container-fluid">
  3. <form class="form-horizontal" method="post">
  4. <fieldset>
  5. <legend>用户登陆</legend>
  6. <%- message %>
  7. <div class="control-group">
  8. <label class="control-label" for="username">用户名</label>
  9. <div class="controls">
  10. <input type="text" class="input-xlarge" id="username" name="username" value="admin">
  11. </div>
  12. </div>
  13. <div class="control-group">
  14. <label class="control-label" for="password">密码</label>
  15. <div class="controls">
  16. <input type="password" class="input-xlarge" id="password" name="password" value="admin">
  17. </div>
  18. </div>
  19. <div class="form-actions">
  20. <button type="submit" class="btn btn-primary">登陆</button>
  21. </div>
  22. </fieldset>
  23. </form>
  24. </div>
  25. <% include footer.html %>

修改routes/index.js,增加req.session.error

 
 
  1. exports.doLogin = function(req, res){
  2.  var user={
  3.  username:'admin',
  4.  password:'admin'
  5.  }
  6.  if(req.body.username===user.username && req.body.password===user.password){
  7.  req.session.user=user;
  8.  return res.redirect('/home');
  9.  } else {
  10.  req.session.error='用户名或密码不正确';
  11.  return res.redirect('/login');
  12.  }
  13. };

让我们来看看效果: http://localhost:3000/login输入错误的和密码, 用户名:dad,密码:da

boostrap-nodejs

9. 页面访问控制

网站登陆部分按照我们的求已经完成了,但网站并不安全。

localhost:3000/home,页面本来是登陆以后才访问的,现在我们不要登陆,直接在浏览器输入也可访问。

页面报错了,提示<%= user.username %> 变量出错。

GET /home?user==a 500 15ms

TypeError: D:\workspace\project\nodejs-demo\views\home.html:2

1| <% include header.html %>

>> 2| <h1>Welcome <%= user.username %>, 欢迎登陆!!</h1>

3| <a claa="btn" href="http://www.geedoo.info/logout">退出</a>

4| <% include header.html %>

Cannot read property 'username' of null

at eval (eval at <anonymous> (D:\workspace\project\nodejs-demo\node_modules\ejs\lib\ejs.js:

at eval (eval at <anonymous> (D:\workspace\project\nodejs-demo\node_modules\ejs\lib\ejs.js:

at D:\workspace\project\nodejs-demo\node_modules\ejs\lib\ejs.js:249:15

at Object.exports.render (D:\workspace\project\nodejs-demo\node_modules\ejs\lib\ejs.js:287:

at View.exports.renderFile [as engine] (D:\workspace\project\nodejs-demo\node_modules\ejs\l

at View.render (D:\workspace\project\nodejs-demo\node_modules\express\lib\view.js:75:8)

at Function.app.render (D:\workspace\project\nodejs-demo\node_modules\express\lib\applicati

at ServerResponse.res.render (D:\workspace\project\nodejs-demo\node_modules\express\lib\res

at exports.home (D:\workspace\project\nodejs-demo\routes\index.js:36:8)

at callbacks (D:\workspace\project\nodejs-demo\node_modules\express\lib\router\index.js:161

这个页面被打开发,因为没有user.username参数。我们避免这样的错误发生。

还记录路由部分里说的get,post,all的作用吗?我现在要回到路由配置中,再做点事情。

修改app.js文件

app.get('/',routes.index);

app.route('/login')

.all(notAuthentication)

.get(routes.login)

.post(routes.doLogin);

app.route('/logout')

app.get('/',routes.index);

app.route('/login')

.all(notAuthentication)

.get(routes.login)

.post(routes.doLogin);

app.route('/logout')

.get(authentication)

.get(routes.logout);

app.route('/home')

.get(authentication)

.get(routes.home);

访问控制:

/ ,谁访问都行,没有任何控制

/login,用all拦截所有访问/login的请求,先调用authentication,用户登陆检查

/logout,用get拦截访问/login的请求,先调用notAuthentication,用户不登陆检查

/home,用get拦截访问/home的请求,先调用Authentication,用户登陆检查

修改app.js文件,增加authentication,notAuthentication两个方法

 
 
  1. function authentication(req, res, next) {
  2.  if (!req.session.user) {
  3.  req.session.error='请先登陆';
  4.  return res.redirect('/login');
  5.  }
  6.  next();
  7. }
  8. function notAuthentication(req, res, next) {
  9.  if (req.session.user) {
  10.   req.session.error='已登陆';
  11.   return res.redirect('/home');
  12.  }
  13.  next();
  14. }

配置好后,我们未登陆,直接访问localhost:3000/home时或者localhost:3000/logout,就会跳到/login页面

登录后, 访问localhost:3000/login 则直接跳到/home页面

到此,express4 相关内容到此为止。



原文转载至:https://teakki.com/p/57dfa81e3c20b02e90a0d142

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值