Примечания к изучению Node.js — простой пример веб-сайта разработки Node.js + Express

Node.js Express
Примечания к изучению Node.js — простой пример веб-сайта разработки Node.js + Express

содержание

  • предисловие
  • Создание нового проекта, создание базы данных и другие подготовительные работы
    • Создайте новый проект express + ejs: sampleEjs
    • создать базу данных
    • Измените файл package.json и установите модули сеанса и mysql.
    • Стили и файлы JQuery
    • Очистите избыточные файлы проекта и добавьте слушателей
  • Планируйте маршруты и создавайте новые связанные файлы
  • Реализовать методы доступа к данным, необходимые для входа и регистрации
  • регистр
  • Авторизоваться
  • титульная страница
  • Отступить безопасно
  • написано после

предисловие

В прошлой статье я узнал некоторые знания, которые будут использованы для создания веб-сайта.http://www.cnblogs.com/zhongweiv/p/nodejs_express_webapp1.html

Эта статья в основном объединяет знания, упомянутые выше, для создания относительно полного веб-приложения, а также углубляет и гибко использует некоторые знания, полученные ранее!

Функция в основном использует базу данных MySQL, состоящую из трех частей: вход в систему, регистрация и домашняя страница; давайте начнем шаг за шагом!

Создание нового проекта, создание базы данных и другие подготовительные работы

1. Создайте новый проект express + ejs: sampleEjs

cd 工作目录
express -e sampleEjs
cd sampleEjs && npm install

2. Создайте базу данных(Также четвертая база данных непротиворечива:http://www.cnblogs.com/zhongweiv/p/nodejs_mysql.html)

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='用户信息表';

3. Измените файл package.json и установите модули сеанса и mysql.

{
  "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"
  }
}

В основном красная часть (mysql, если она есть в этой статьеhttp://www.cnblogs.com/zhongweiv/p/nodejs_mysql.htmlОн был установлен в соответствии с моими шагами в , на самом деле вам не нужно его устанавливать снова), а затем запустите его в cmd

npm install

После завершения установки откройте файл app.js и добавьте следующий код.

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'}));

...

4. Стили и файлы JQuery

Стиль: стиль в основном использует бутстрап 3.0.3.https://github.com/twbs/bootstrap/releases/tag/v3.0.3

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

Добавьте вышеуказанные файлы в проект, структура каталогов выглядит следующим образом

Два файла css, которых нет в пакете начальной загрузки, оформлены следующим образом:

navbar-static-top.css

body {
  min-height: 2000px;
}

.navbar-static-top {
  margin-bottom: 19px;
}

signin.css

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;
}

5. Очистите лишние файлы проекта и добавьте мониторинг

Удалите встроенный файл style.css.

Удалите файл user.js в каталоге маршрутов и удалите следующий код в app.js.

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

...

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

Добавьте порт 8000 для прослушивания файла app.js.

...
app.listen(8000);
...

На этом пример среды готов!

Планируйте маршруты и создавайте новые связанные файлы

1. Маршрутизация

титульная страница:/

Страница регистрации: /reg

Страница входа: /login

Безопасный выход: /logout

(красный означает, что нужно создать заново)

2. Создайте следующие файлы в каталоге маршрутов.

reg.js

login.js

logout.js

3. Новое в каталоге представлений:

header.ejs

reg.ejs

login.ejs

4. Откройте файл app.js и добавьте следующий код.

...

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);

...

Реализовать методы доступа к данным, необходимые для входа и регистрации

Мы создаем новую папку моделей и создаем в ней новый user.js Код примера выглядит следующим образом

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);                     
        });        
    };
 
});

Есть три способа, базовая регистрация входа достаточно ^ _ ^!

регистр

1. Давайте сначала выложим HTML и CSS, а также проверим переднюю часть и отобразим подсказку (reg.ejs)

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">       
    <title><%= title %></title>
  
    <link rel="stylesheet" type="text/css" href="/stylesheets/bootstrap.min.css" />
    <link rel="stylesheet" type="text/css" href="/stylesheets/signin.css" />
  </head>

  <body>

    <div id="container" class="container">
       <% if (locals.success) { %> 
          <div id="alt_sucess" class="alert alert-success"> 
            <%- success %> 
          </div> 
       <% } %> 

      <% if (locals.error) { %> 
        <div id="alt_warning" class="alert alert-warning"> 
          <%= error %> 
        </div> 
      <% } %> 

      <form class="form-signin" role="form" method="post">
        <h2 class="form-signin-heading">注册</h2>

        <input id="txtUserName" name="txtUserName" type="text" class="form-control" placeholder="用户名" required autofocus />
        <input id="txtUserPwd" name="txtUserPwd" type="password" class="form-control" placeholder="密码" required/>
        <input id="txtUserRePwd"  name="txtUserRePwd" type="password" class="form-control" placeholder="重复密码" required/>   

        <button id="btnSub" class="btn btn-lg btn-primary" type="submit">注  册</button>
        <a class="btn btn-link" href="/login" role="button">登  录</a>
      </form>  
      
    </div> 
  </body>
</html>

<script src="/javascripts/jquery-1.11.1.min.js" type="text/javascript"></script>

<script type="text/javascript">
     String.prototype.format = function (args) {
            var result = this;
            if (arguments.length > 0) {
                if (arguments.length == 1 && typeof (args) == "object") {
                    for (var key in args) {
                        if (args[key] != undefined) {
                            var reg = new RegExp("({" + key + "})", "g");
                            result = result.replace(reg, args[key]);
                        }
                    }
                }
                else {
                    for (var i = 0; i < arguments.length; i++) {
                        if (arguments[i] != undefined) {
                            var reg = new RegExp("({)" + i + "(})", "g");
                            result = result.replace(reg, arguments[i]);
                        }
                    }
                }
            }
            return result;
    }

    $(function(){
        $('#btnSub').on('click',function(){
            var 
            $txtUserName = $('#txtUserName'),
            txtUserNameVal = $.trim($txtUserName.val()),            
            $txtUserPwd = $('#txtUserPwd'),
            txtUserPwdVal = $.trim($txtUserPwd.val()),
            $txtUserRePwd = $('#txtUserRePwd'),
            txtUserRePwdVal = $.trim($txtUserRePwd.val()),
            errorTip = '<div id="errorTip" class="alert alert-warning">{0}</div> ';

            $("#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;
        })
    });

</script>

2. Реализовать функцию регистрации (reg.js)

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 = '注册成功,请点击   <a class="btn btn-link" href="/login" role="button"> 登录 </a>' ;
          }
          else
          {
              res.locals.error = err;
          }
         
          res.render('reg', { title: TITLE_REG });
          });    
    });          
});

module.exports = router;

3. Запустите для просмотра эффекта (перейдите в корневую директорию проекта в командной строке, запустите: node app)

После запуска программы введите в браузере: http://localhost:8000/reg, как показано ниже после запуска

Подсказка переднего плана выглядит следующим образом:

После успешной регистрации он запросит и проверит введенные данные в базе данных!

На этом функция регистрации завершена (например, определение того, существует ли уже имя пользователя и т. д., оно не будет отображаться в списке, запишите его и запустите самостоятельно!)

Авторизоваться

1. Давайте сначала выложим HTML и CSS, а также проверим переднюю часть и отобразим некоторую подсказку (login.ejs)

login.ejs

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">       
    <title><%= title %></title>
  
  <link rel="stylesheet" type="text/css" href="/stylesheets/bootstrap.min.css" />
    <link rel="stylesheet" type="text/css" href="/stylesheets/signin.css" />
  </head>

  <body>

    <div id="container" class="container">
       <% if (locals.success) { %> 
          <div id="alt_sucess" class="alert alert-success"> 
            <%- success %> 
          </div> 
       <% } %> 

      <% if (locals.error) { %> 
        <div id="alt_warning" class="alert alert-warning"> 
          <%= error %> 
        </div> 
      <% } %> 

      <form class="form-signin" role="form" method="post">
        <h2 class="form-signin-heading">登录</h2>
        <input id="txtUserName" name="txtUserName" type="text" class="form-control" placeholder="用户名" required autofocus />
            <input id="txtUserPwd" name="txtUserPwd" type="password" class="form-control" placeholder="密码" required/> 
        <label class="checkbox">
          <input name="chbRem" id="chbRem" type="checkbox" value="remember-me"> 记住密码
        </label>              
        <button id="btnSub" class="btn btn-lg btn-primary" type="submit">登  录</button>
        <a class="btn btn-link" href="/reg" role="button">注  册</a>

      </form>
            
    </div> 
  </body>
</html>

<script src="/javascripts/jquery-1.11.1.min.js" type="text/javascript"></script>

<script type="text/javascript">
     String.prototype.format = function (args) {
            var result = this;
            if (arguments.length > 0) {
                if (arguments.length == 1 && typeof (args) == "object") {
                    for (var key in args) {
                        if (args[key] != undefined) {
                            var reg = new RegExp("({" + key + "})", "g");
                            result = result.replace(reg, args[key]);
                        }
                    }
                }
                else {
                    for (var i = 0; i < arguments.length; i++) {
                        if (arguments[i] != undefined) {
                            var reg = new RegExp("({)" + i + "(})", "g");
                            result = result.replace(reg, arguments[i]);
                        }
                    }
                }
            }
            return result;
    }

    $(function(){
        $('#btnSub').on('click',function(){           
            var 
            $txtUserName = $('#txtUserName'),
            txtUserNameVal = $.trim($txtUserName.val()),            
            $txtUserPwd = $('#txtUserPwd'),
            txtUserPwdVal = $.trim($txtUserPwd.val()),
            errorTip = '<div id="errorTip" class="alert alert-warning">{0}</div> ';  

            $("#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;                
        })
    });

</script>

2. Затем введите внутренний код (включая сохранение сеанса и пароль для записи файлов cookie).

login.js

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;

3. Бегите, чтобы увидеть эффект

После запуска программы введите в браузере: http://localhost:8000/login, как показано ниже после запуска

Быстрый эффект следующий:

После успешного входа вы будете перенаправлены на домашнюю страницу, и сейчас мы начнем доделывать домашнюю страницу!

титульная страница

Основная цель домашней страницы - проверить, доступна ли функция входа и регистрации.Хотя домашняя страница в принципе не имеет никаких функций, я все же помещаю ее голову в файл header.ejs!

1. Заголовок HTML и CSS

header.ejs

<div class="navbar navbar-default navbar-static-top" role="navigation">
  <div class="container">
    <div class="navbar-header">         
      <a class="navbar-brand" href="/">Porschev - Nodejs + Express + Ejs + MySQL + Bootstrap 示例</a>
    </div>
    <div class="navbar-collapse collapse">
      <ul class="nav navbar-nav navbar-right">

        <% if (locals.username) { %> 
          <li><a href="#">        
                <%= username %>          
          </a></li>
          <li><a href="/logout">安全退出</a></li> 
        <% } %>           
      </ul>
    </div>              
  </div>
</div>

2.index.ejs

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">           
      <title><%= title %></title>
    <link rel="stylesheet" type="text/css" href="/stylesheets/bootstrap.min.css" />
    <link rel="stylesheet" type="text/css" href="/stylesheets/navbar-static-top.css" />   
  </head>
  <body>
      <% include header %> 
  </body>
</html>

3. Реализация Index.js (в основном файлы cookie, оценка состояния входа в сеанс)

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;

4. Запускаем, после авторизации проверяем эффект

Войдите, не проверяйте автоматический вход, запустите http://localhost:8000, чтобы автоматически перейти на страницу входа.

После ввода правильного имени пользователя и пароля для успешного входа в систему страница отображается следующим образом (имя пользователя отображается в правом верхнем углу)

Закройте браузер, снова введите http://localhost:8000, перейдите на страницу входа, вам нужно снова войти в систему!

Войдите снова, проверьте автоматический вход

Перейдите на домашнюю страницу, как показано выше;

Закройте браузер и снова введите http://localhost:8000, он не перейдет на страницу входа, а войдет напрямую! (куки работают)

Закройте браузер и через минуту снова введите http://localhost:8000, чтобы перейти на страницу входа (файлы cookie недействительны)

Отступить безопасно

Безопасный выход в основном для очистки сеанса (logout.js)

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

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

module.exports = router;

Достигнутый эффект: в случае не автоматического входа в систему нажмите безопасный выход после входа в систему, не закрывайте браузер, а затем перейдите на домашнюю страницу через URL-адрес, вы не можете войти напрямую и перейдете на страницу входа!

написано после

Эта статья в основном о применении полученных ранее знаний, а разрозненные знания действительно можно превратить в навыки, которые могут помочь вам в достижении функций!

Пример не слишком оптимизирует код, есть много частей, которые можно оптимизировать, и подумать над улучшением в процессе обучения, например:

1. Как писать код эффективнее и красивее

2. Если часть проверки входа написана как в index.js, что делать, если страниц слишком много?

3. Как лучше извлечь общедоступную часть страницы, например header.ejs

...

намекать:

1. В приведенном выше примере я упомянул «автоматический вход», а написал «запись пароля», все должны использовать его как автоматический вход ^_^!

2.По поводу логина и автоматического логина сессий и куки,примеры в основном отражают приложение.Мне лень доделывать по реальным спецификациям.Те у кого есть опыт в веб-разработке должны знать как это сделать и что информация для хранения Очистите сообщение или напишите мне!