Nodejs学习笔记(七)--- Node.js + Express 构建网站简单示例

2020-12-13 05:22

阅读:343

YPE html>

目录

前言

  上一篇学习了一些构建网站会用到的一些知识点   http://www.cnblogs.com/zhongweiv/p/nodejs_express_webapp1.html

  这一篇主要结合前面讲到的知识,去构建一个较为完整的网站应用程序,对前面学到的一些知道做一个串联加深并灵活运用!

  功能主要用MySQL数据库,包括登录、注册、主页三部分;下面就一步步开始吧!

新建项目、建立数据库以及其它准备工作

  1.新建express + ejs 项目:sampleEjs

soscw.com,搜素材
cd 工作目录
express -e sampleEjs
cd sampleEjs && npm install
soscw.com,搜素材

  2.创建数据库 (还第四篇数据库一致: http://www.cnblogs.com/zhongweiv/p/nodejs_mysql.html)

soscw.com,搜素材
CREATE DATABASE IF NOT EXISTS nodesample CHARACTER SET UTF8;

USE nodesample;

SET FOREIGN_KEY_CHECKS=0;

DROP TABLE IF EXISTS `userinfo`;
CREATE TABLE `userinfo` (
  `Id` int(11) NOT NULL AUTO_INCREMENT COMMENT 主键,
  `UserName` varchar(64) NOT NULL COMMENT 用户名,
  `UserPass` varchar(64) NOT NULL COMMENT 用户密码,
  PRIMARY KEY (`Id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT=用户信息表;
soscw.com,搜素材

  3.修改package.json文件,安装session和mysql模块

soscw.com,搜素材
{
  "name": "sampleEjs",
  "version": "0.0.1",
  "private": true,
  "scripts": {
    "start": "node ./bin/www"
  },
  "dependencies": {
    "express": "~4.0.0",
    "static-favicon": "~1.0.0",
    "morgan": "~1.0.0",
    "cookie-parser": "~1.0.1",
    "body-parser": "~1.0.0",
    "debug": "~0.7.4",
    "ejs": "~0.8.5",
    "express-session" : "latest",
    "mysql" : "latest"
  }
}
soscw.com,搜素材

  主要是标红部分(mysql如果是在这篇文章 http://www.cnblogs.com/zhongweiv/p/nodejs_mysql.html 中按照我的步骤安装过了的,其实可以不用再安装),然后在cmd中运行

soscw.com,搜素材
npm install
soscw.com,搜素材

 

  安装完成后,打开app.js文件,添加如下代码

soscw.com,搜素材
var express = require(‘express‘);
var path = require(‘path‘);
var favicon = require(‘static-favicon‘);
var logger = require(‘morgan‘);
var cookieParser = require(‘cookie-parser‘);
var bodyParser = require(‘body-parser‘);
var session = require(‘express-session‘);

...      

//这里传入了一个密钥加session id
app.use(cookieParser(‘Wilson‘));
//使用靠就这个中间件
app.use(session({ secret: ‘wilson‘}));

...
soscw.com,搜素材

  4.样式和JQuery文件

  样式:  样式主要使用了bootstrap 3.0.3  https://github.com/twbs/bootstrap/releases/tag/v3.0.3

  JQuery:     jquery  1.11.1           http://jquery.com/download/ 

  

  添加以上文件到项目中,目录结构如下

  soscw.com,搜素材

 

  不在bootstrap包中两个css文件样式如下:

soscw.com,搜素材
soscw.com,搜素材soscw.com,搜素材
body {
  min-height: 2000px;
}

.navbar-static-top {
  margin-bottom: 19px;
}
navbar-static-top.css
soscw.com,搜素材
soscw.com,搜素材soscw.com,搜素材
body {
  padding-top: 40px;
  padding-bottom: 40px;
  background-color: #eee;
}

.form-signin {
  max-width: 330px;
  padding: 15px;
  margin: 0 auto;
}
.form-signin .form-signin-heading,
.form-signin .checkbox {
  margin-bottom: 10px;
}
.form-signin .checkbox {
  font-weight: normal;
}
.form-signin .form-control {
  position: relative;
  font-size: 16px;
  height: auto;
  padding: 10px;
  -webkit-box-sizing: border-box;
     -moz-box-sizing: border-box;
          box-sizing: border-box;
}
.form-signin .form-control:focus {
  z-index: 2;
}
.form-signin input[type="text"] {
  margin-bottom: 10px;
  border-bottom-left-radius: 0;
  border-bottom-right-radius: 0;
}
.form-signin input[type="password"] {
  margin-bottom: 10px;
  border-top-left-radius: 0;
  border-top-right-radius: 0;
}
signin.css

  5.清理项目冗余文件,并添加监听

   删除自带的style.css文件

   删除routes目录下user.js文件,并在app.js中删除如下代码

soscw.com,搜素材
var users = require(‘./routes/users‘);

...

app.use(‘/users‘, users);
soscw.com,搜素材

    app.js文件中添加8000端口监听

soscw.com,搜素材
...
app.listen(8000);
...
soscw.com,搜素材

  

  到这里,示例环境准备完毕!

规划路由,并新建相关文件

  1.路由

  首页:/

  注册页:/reg

  登录页:/login

  安全退出:/logout

  (红色表示需要新建的)

 

  2.routes目录下新建如下文件

  reg.js

  login.js

  logout.js

 

  3.views目录下新建:

  header.ejs

  reg.ejs

  login.ejs

 

  4.打开app.js文件,添加如下代码

soscw.com,搜素材
...

var routes = require(‘./routes/index‘);
var reg = require(‘./routes/reg‘);
var login = require(‘./routes/login‘);
var logout = require(‘./routes/logout‘);

...

app.use(‘/‘, routes);
app.use(‘/reg‘, reg);
app.use(‘/login‘, login);
app.use(‘/logout‘, logout);

...
soscw.com,搜素材

实现登录和注册需要的数据访问方法

  我们新建一个models文件夹,在其中新建user.js,实例代码如下

soscw.com,搜素材
var mysql = require(‘mysql‘);
var DB_NAME = ‘nodesample‘;

var pool  = mysql.createPool({
    host     : ‘192.168.0.200‘,
    user     : ‘root‘,
    password : ‘abcd‘
});

pool.on(‘connection‘, function(connection) {  
    connection.query(‘SET SESSION auto_increment_increment=1‘); 
});  

function User(user){
    this.username = user.username;
    this.userpass = user.userpass;
};
module.exports = User;

pool.getConnection(function(err, connection) {

    var useDbSql = "USE " + DB_NAME;
    connection.query(useDbSql, function (err) {
         if (err) {
            console.log("USE Error: " + err.message);
            return;
         }
         console.log(‘USE succeed‘);
    });

    //保存数据
    User.prototype.save = function save(callback) {
        var user = {
            username: this.username,
            userpass: this.userpass
        };

        var insertUser_Sql = "INSERT INTO userinfo(id,username,userpass) VALUES(0,?,?)";

        connection.query(insertUser_Sql, [user.username, user.userpass], function (err,result) {
            if (err) {
                console.log("insertUser_Sql Error: " + err.message);
                return;
            }

            connection.release();

            console.log("invoked[save]");
            callback(err,result);                     
        });       
    };

    //根据用户名得到用户数量
    User.getUserNumByName = function getUserNumByName(username, callback) {

        var getUserNumByName_Sql = "SELECT COUNT(1) AS num FROM userinfo WHERE username = ?";

        connection.query(getUserNumByName_Sql, [username], function (err, result) {
            if (err) {
                console.log("getUserNumByName Error: " + err.message);
                return;
            }

            connection.release();

            console.log("invoked[getUserNumByName]");
            callback(err,result);                     
        });        
    };

    //根据用户名得到用户信息
    User.getUserByUserName = function getUserNumByName(username, callback) {

        var getUserByUserName_Sql = "SELECT * FROM userinfo WHERE username = ?";

        connection.query(getUserByUserName_Sql, [username], function (err, result) {
            if (err) {
                console.log("getUserByUserName Error: " + err.message);
                return;
            }

            connection.release();

            console.log("invoked[getUserByUserName]");
            callback(err,result);                     
        });        
    };
 
});
soscw.com,搜素材

  有这三个方法,基本登录注册就够了^_^!

注册

  1.先来布局一下HTML和CSS,加上前端验证及一些提示信息显示(reg.ejs)

soscw.com,搜素材
if (locals.success) { %>
if (locals.error) { %>
; $("#errorTip,#alt_sucess,#alt_warning").remove(); if(txtUserNameVal.length == 0) { $("#container").prepend(errorTip.format(‘用户名不能为空‘)); $txtUserName.focus(); return false; } if(txtUserPwdVal.length == 0) { $("#container").prepend(errorTip.format(‘密码不能为空‘)); $txtUserPwd.focus(); return false; } if(txtUserRePwdVal.length == 0) { $("#container").prepend(errorTip.format(‘重复密码不能为空‘)); $txtUserRePwd.focus(); return false; } if(txtUserPwdVal != txtUserRePwdVal) { $("#container").prepend(errorTip.format(‘两次密码不一致‘)); $txtUserPwd.focus(); return false; } return true; }) });
soscw.com,搜素材

  2.实现注册功能(reg.js)

soscw.com,搜素材
var express = require(‘express‘),
    router = express.Router(),
    User = require(‘../models/user.js‘),
    crypto = require(‘crypto‘),
    TITLE_REG = ‘注册‘;

router.get(‘/‘, function(req, res) {
  res.render(‘reg‘,{title:TITLE_REG});
});

router.post(‘/‘, function(req, res) {
  var userName = req.body[‘txtUserName‘],
      userPwd = req.body[‘txtUserPwd‘],
      userRePwd = req.body[‘txtUserRePwd‘],      
      md5 = crypto.createHash(‘md5‘);
 
      userPwd = md5.update(userPwd).digest(‘hex‘);

  var newUser = new User({
      username: userName,
      userpass: userPwd
  });

  //检查用户名是否已经存在
  User.getUserNumByName(newUser.username, function (err, results) {        
             
      if (results != null && results[0][‘num‘] > 0) {
          err = ‘用户名已存在‘;
      }

      if (err) {
          res.locals.error = err;
          res.render(‘reg‘, { title: TITLE_REG });
          return;
      }

      newUser.save(function (err,result) {
          if (err) {
              res.locals.error = err;
              res.render(‘reg‘, { title: TITLE_REG }); 
              return;            
          }        

          if(result.insertId > 0)
          {
              res.locals.success = ‘注册成功,请点击    登录 ‘ ;
          }
          else
          {
              res.locals.error = err;
          }
         
          res.render(‘reg‘, { title: TITLE_REG });
          });    
    });          
});

module.exports = router;
soscw.com,搜素材

  3.运行查看效果(命令行中cd到项目根目录,运行: node app)

  运行程序后,在浏览器中输入:  http://localhost:8000/reg   ,运行后如下图

  soscw.com,搜素材

  前端提示如下图:

  soscw.com,搜素材

  注册成功后提示与检查数据库中插入数据!

  soscw.com,搜素材

  soscw.com,搜素材

 

  到这里,注册功能完成(比如判断用户名是否已存在等情况显示就不列举了,自已写出来运行再看!)

登录

  1.先来布局一下HTML和CSS,加上前端验证及一些提示信息显示(login.ejs)

soscw.com,搜素材soscw.com,搜素材

  
if (locals.success) { %>
if (locals.error) { %>
; $("#errorTip,#alt_warning").remove(); if(txtUserNameVal.length == 0) { $("#container").prepend(errorTip.format(‘用户名不能为空‘)); $txtUserName.focus(); return false; } if(txtUserPwdVal.length == 0) { $("#container").prepend(errorTip.format(‘密码不能为空‘)); $txtUserPwd.focus(); return false; } return true; }) });
login.ejs 示例

  2.再来完成后端代码(包括保存session和cookies记录密码)

soscw.com,搜素材soscw.com,搜素材
var express = require(‘express‘),
    router = express.Router(),
    User = require(‘../models/user.js‘),
    crypto = require(‘crypto‘),
    TITLE_LOGIN = ‘登录‘;

router.get(‘/‘, function(req, res) {
    res.render(‘login‘,{title:TITLE_LOGIN});
});

router.post(‘/‘, function(req, res) {
    var userName = req.body[‘txtUserName‘],
        userPwd = req.body[‘txtUserPwd‘],
        isRem = req.body[‘chbRem‘],
        md5 = crypto.createHash(‘md5‘);
       
    User.getUserByUserName(userName, function (err, results) {                            
        
        if(results == ‘‘)
        {
            res.locals.error = ‘用户不存在‘;
             res.render(‘login‘,{title:TITLE_LOGIN});
             return;
        }

         userPwd = md5.update(userPwd).digest(‘hex‘);
         if(results[0].UserName != userName || results[0].UserPass != userPwd)
         {
             res.locals.error = ‘用户名或密码有误‘;
             res.render(‘login‘,{title:TITLE_LOGIN});
             console.log(1);
             return;
         }
         else
         {
             if(isRem)
             {
                res.cookie(‘islogin‘, userName, { maxAge: 60000 });                 
             }

             res.locals.username = userName;
             req.session.username = res.locals.username;  
             console.log(req.session.username);                        
             res.redirect(‘/‘);
             return;
         }     
    });              
});

module.exports = router;
login.js

  3.运行查看效果

  运行程序后,在浏览器中输入:  http://localhost:8000/login   ,运行后如下图

  soscw.com,搜素材

  提示效果如下:

  soscw.com,搜素材

  

  登录成功后,会跳转到首页,下面我们就着手把首页完成!

首页

  首页主要为了测试登录注册的功能是否可可用,虽然首页基本没什么功能,但是我还是把它头部放到了header.ejs文件中!

  1.头部HTML和CSS

soscw.com,搜素材soscw.com,搜素材
header.ejs

  2.index.ejs

soscw.com,搜素材soscw.com,搜素材

  
index.ejs

  3.index.js实现(主要是cookies,session登录状态判断)

soscw.com,搜素材soscw.com,搜素材
var express = require(‘express‘),
    router = express.Router();

router.get(‘/‘, function(req, res) {
  if(req.cookies.islogin)
  { 
         console.log(‘cookies:‘ + req.cookies.islogin);
       req.session.username = req.cookies.islogin;
  }  

  if(req.session.username)
  {    
          console.log(‘session:‘ + req.session.username);
        res.locals.username = req.session.username;      
  }
  else
  {
        res.redirect(‘/login‘);
        return;    
  }

  res.render(‘index‘,{title:‘主页‘});
});

module.exports = router;
index.js

  4.运行,登录后,查看效果

  登录,不勾选自动登录,运行http://localhost:8000 会自动跳到登录页

  输入正确的用户名和密码登录成功后,页面显示如下(右上角的部分显示了用户名)

  soscw.com,搜素材

  关闭浏览器,再次输入http://localhost:8000 ,跳转到登录页,需要重新登录!

  

  再次登录,勾选自动登录

  进行首页如上图;

  关闭浏览器,再次输入http://localhost:8000,不会跳转到登录页,而是直接登录了!(cookies起了作用)

  关闭浏览器,过一分钟后,再输入http://localhost:8000 ,跳转到登录页 (cookies失效)

安全退出

  安全退出主要就是清除session(logout.js)

soscw.com,搜素材soscw.com,搜素材
var express = require(‘express‘),
     router = express.Router();

router.get(‘/‘, function(req, res) {
  req.session.destroy();
  res.redirect(‘/login‘);
});

module.exports = router;
logout.js

  

  实现效果:在不是自动登录的情况下,登录后点击安全退出,不关闭浏览器,通过url再访问首页,无法直接进入,会跳转到登录页!

前言

  这一篇主要是对前面所学的知识的运用,把零散的知识能真正变成能帮助你实现功能的技能!

 

  示例并没有过多去优化代码,有很多可优化的部分,在学习的过程中去思考提升,比如:

  1.怎么把代码写的更高效和优美

  2.&am


评论


亲,登录后才可以留言!