您的位置:首页 > Web前端 > Node.js

原生node实现简易留言板

2018-10-19 21:23 731 查看

原生node实现简易留言板

学习node,实现一个简单的留言板小demo

1. 使用模块

  • http模块

    创建服务

  • fs模块

    操作读取文件

  • url模块

    便于path操作并读取表单提交数据

  • art-template模块(需npm安装)

    服务端渲染

2. 服务端

2.1 服务端代码

var http = require('http')
var fs = require('fs')
var url = require('url')
var template = require('art-template')

var comments = [
{
name:'阿孔',
message: 'Yo~~',
dataTime: '2018-09-27'
},
{
name: '老许',
message: 'Yo~@@@~',
dataTime: '2018-09-27'
}
]
http.createServer(function (req, res) {
var parseObj = url.parse(req.url, true)

var pathname = parseObj.pathname

if (pathname === '/') {
fs.readFile('./views/留言本.html', function (err, data) {
if (err) {
return res.end('404 Not Found')
}
var htmlStr = template.render(data.toString(), {
comments: comments
})
res.end(htmlStr)
})
} else if (pathname.indexOf('/public/') === 0) {
fs.readFile('.' + pathname, function (err, data) {
if (err) {
return res.end('404 Not Found')
}
res.end(data)
})
} else if (pathname === '/post') {
fs.readFile('./views/post.html', function (err, data) {
if (err) {
return res.end('404 Not Found')
}
res.end(data)
})
} else if (pathname === '/pinglun') {
// 追加到数组
var comment = parseObj.query
comment.dataTime = '2018-10-01'
comments.unshift(comment)
// 跳转首页
// 如何通过服务器让客户端重定向
//  1. 状态码设置为 302 临时重定向 statusCode
//  2. 在响应头中通过 location 告诉客户端去哪重定向
res.statusCode = 302
res.setHeader('Location', '/')
res.end()
}else {
fs.readFile('./views/404.html', function(err, data) {
res.end(data)
})
}

}).listen(3000, function () {
console.log('http://localhost:3000')
})

2.2 页面渲染

使用了 第三方 的渲染模板工具

art-template
,在请求 / 主页时,在服务端将页面中的数据渲染出来 数据来自comments数组

if (pathname === '/') {
fs.readFile('./views/留言本.html', function (err, data) {
if (err) {
return res.end('404 Not Found')
}
var htmlStr = template.render(data.toString(), {
comments: comments
})
res.end(htmlStr)
})
}

2.3 评论提交

评论提交的实现,在post提交页面中 form表单的行为action指定

/pinglun
,随后在服务端的app.js中处理
/pinglun
路径请求逻辑

else if (pathname === '/pinglun') {
// 追加到数组
var comment = parseObj.query
comment.dataTime = '2018-10-01'
comments.unshift(comment)
// 跳转首页
// 如何通过服务器让客户端重定向
//  1. 状态码设置为 302 临时重定向 statusCode
//  2. 在响应头中通过 location 告诉客户端去哪重定向
res.statusCode = 302
res.setHeader('Location', '/')
res.end()
}

其中,有几点需要注意

  1. parseObj

    parseObj是通过url模块解析每次请求的路径得到的

    var parseObj = url.parse(req.url, true)

    url中的parse方法可以将url解析为一段一段的数据

    例如,我们在 node 中 执行下面代码

    var parseObj = url.parse('http://localhost:3000/pinglun?name=aaa&message=bbbbbbb', true)
    console.log(parseObj)
    
    Url {
    protocol: 'http:',
    slashes: true,
    auth: null,
    host: 'localhost:3000',
    port: '3000',
    hostname: 'localhost',
    hash: null,
    search: '?name=aaa&message=bbbbbbb',
    query: { name: 'aaa', message: 'bbbbbbb' },
    pathname: '/pinglun',
    path: '/pinglun?name=aaa&message=bbbbbbb',
    href: 'http://localhost:3000/pinglun?name=aaa&message=bbbbbbb' }

    可以看到,url的方法直接将url解析成一个对象,里面包含着很多我们能方便使用的属性,入其中的query,这里面就包含着我们提交的数据信息,url.parse方法的第二个参数默认为 false ,若默认false的话,query就不是一个对象,而是一个字符串,设置为true后就会自动帮我们解析成一个对象,方便我们使用。

    我们评论的追加就是取得这个parseObj.query对象中的数据

    var comment = parseObj.query
    comment.dataTime = '2018-10-01'
    comments.unshift(comment)

    最后在使用unshift数组方法追加到最前面,这样我们的评论功能就能实现了。

2.4 服务端重定向

我们在追加我们的评论后,需要跳转到我们的 / 主页,这是我们的 comments已经追加新数据, 在渲染的时候,自然就能将我们的新数据展现到我们的页面中。

关于重定向,此次有两部操作

res.statusCode = 302
res.setHeader('Location', '/')

第一步设置res请求的状态码,状态302 是临时重定向的状态码,

而后我们设置响应头的Location设置为 主页

访问别的页面:response.setStatus(302); response.setHeader("location","url");

2.5 代码杂点笔记

  • 模块加载在node中是同步的,通常把当前模块所有的依赖项都声明再文件模块最上面

  • 为了让目录结构保持统一清晰,所以我们约定,把所有的 HTML 文件都放到 views(视图) 目录中,把所有的静态资源都存放在 public 目录中

    ​ 浏览器收到 HTML 响应内容之后,就要开始从上到下依次解析,

    ​ 当在解析的过程中,如果发现:

    link
  • script
  • img
  • iframe
  • video
  • audio

​ 等带有 src 或者 href(link) 属性标签(具有外链的资源)的时候,浏览器会自动对这些资源发起新的请求。

  • 在这中服务端中,请求文件路径不要再去写相对路径了,因为这个时候所有的资源都是通过url来获取的,我们服务器开放了/public/目录,这里的静态资源的请求,如bootstrap.css都写为:

    <link rel="stylesheet" href="/public/lib/bootstrap/dist/css/bootstrap.css">

    浏览器在真正发请求的时候会最终把http://localhost:3000 拼在前面

  • 3. 客户端

    3.1 客户端代码

    <body>
    <div class="header container">
    <div class="page-header">
    <h1>留言板  <small>--- a small demo of node</small></h1>
    <a class="btn btn-success" href="/post">发表留言</a>
    <hr>
    </div>
    </div>
    <div class="comments container">
    <ul class="list-group">
    {{each comments}}
    <li class="list-group-item">{{$value.name}}说: {{$value.message}}<span class="float-right">{{$value.dataTime}}</span></li>
    {{/each}}
    </ul>
    </div>
    </body>
    <!DOCTYPE html>
    <html lang="en">
    <head>
    <meta charset="UTF-8">
    <title>Document</title>
    <link rel="stylesheet" href="/public/lib/bootstrap/dist/css/bootstrap.css">
    </head>
    
    <body>
    <div class="header container">
    <div class="page-header">
    <h1><a href="/">首页</a> <small>发表评论</small></h1>
    </div>
    </div>
    <div class="comments container">
    <!--
    以前表单是如何提交的?
    表单中需要提交的表单控件元素必须具有 name 属性
    表单提交分为:
    1. 默认的提交行为
    2. 表单异步提交
    
    action 就是表单提交的地址,说白了就是请求的 url 地址
    method 请求方法
    get
    post
    -->
    <form action="/pinglun" method="get">
    <div class="form-group">
    <label for="input_name">你的大名</label>
    <input type="text" class="form-control" required minlength="2" maxlength="10" id="input_name" name="name" placeholder="请写入你的姓名">
    </div>
    <div class="form-group">
    <label for="textarea_message">留言内容</label>
    <textarea class="form-control" name="message" id="textarea_message" cols="30" rows="10" required minlength="5" maxlength="20"></textarea>
    </div>
    <button type="submit" class="btn btn-default">发表</button>
    </form>
    </div>
    </body>
    
    </html>

    所有的代码都在github仓库github代码

    内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
    标签: