原生node实现简易留言板
原生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()
}
其中,有几点需要注意
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代码
原生node实现简易留言板的更多相关文章
- JSP简易留言板
写在前面 在上篇博文JSP内置对象中介绍JSP的9个内置对象的含义和常用方法,但都是比较理论的知识.今天为大家带来一个小应用,用application制作的简易留言板. 包括三个功能模块:留言提交.留 ...
- DOM操作相关案例 模态对话框,简易留言板,js模拟选择器hover,tab选项卡,购物车案例
1.模态框案例 需求: 打开网页时有一个普通的按钮,点击当前按钮显示一个背景图,中心并弹出一个弹出框,点击X的时候会关闭当前的模态框 代码如下: <!DOCTYPE html> <h ...
- Flask学习之旅--简易留言板
一.写在前面 正所谓“纸上得来终觉浅,方知此事要躬行”,在看文档和视频之余,我觉得还是要动手做点什么东西才能更好地学习吧,毕竟有些东西光看文档真的难以理解,于是就试着使用Flask框架做了一个简易留言 ...
- php实现简易留言板效果
首先是Index页面效果图 index.php <?php header('content-type:text/html;charset=utf-8'); date_default_timezo ...
- 原生JS实现简单留言板功能
原生JS实现简单留言板功能,实现技术:css flex,原生JS. 因为主要是为了练手js,所以其中布局上的一些细节并未做处理. <!DOCTYPE html> <html lang ...
- 微信小程序实现简易留言板
微信小程序现在很火,于是也就玩玩,做了一个简易的留言板,让大家看看,你们会说no picture you say a j8 a,好吧先上图. 样子就是的,功能一目了然,下面我们就贴实现的代码,首先是H ...
- vue实现简易留言板
首先引入vue.js <script src="vue.js"></script> 布局 <div id="div"> &l ...
- js简易留言板
<!DOCTYPE html> <html lang="en"> <head> <meta charset="U ...
- js 实现简易留言板功能
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
随机推荐
- django orm多条件查询及except处理不存在记录的样码
日常所用的功能, 记录下来加深记忆. try: sv_set = Sv.objects.get(project=site_name, version=deploy_version) print sv_ ...
- 【bzoj1025】【SCOI2009】【游戏】【dp】
Description windy学会了一种游戏.对于1到N这N个数字,都有唯一且不同的1到N的数字与之相应.最開始windy把数字按顺序1,2.3.--,N写一排在纸上. 然后再在这一排以下写上它们 ...
- Maven学习--------基础2
继续学习Maven. 4.2 变量 1) 问:怎样使用变量替换?项目中的某个配置文件比方jdbc.properties使用了一些pom中的变量,怎样在公布中使用包括真实内容的终于结果文件? 答:使用资 ...
- _DataStructure_C_Impl:基数排序
#include<stdio.h> #include<stdlib.h> #include<math.h> #include<string.h> #de ...
- Unity3D与JSP TomCatserver传递数据和文件( 二 ) Unity3D向java传输表单
扫码关注微信公众号,获取最新资源 经历了一天的工作.我又来更新啦...白天手欠,把上一个给删了.明天重写吧.. 废话不多说.我们先去Unity里创建一个能够输入username和password的登录 ...
- [cocos2dx笔记012]一定简易的UI配置类
使用cocostudio能够装载编辑好的UI,可是过于复杂.特别是在加截UI后,发现触屏事件有些问题. 假设直接使用程序写死载入UI又过于麻烦.花点时间,添加了一个基于ini的UI配置类,眼下仅仅实现 ...
- Linux命令(二)——目录和文件管理命令
一.Linux系统的目录结构 1.根目录(/):顶层目录,某些系统中的唯一分区. 2./bin命令文件目录:包含Linux命令的二进制可执行文件. 3./boot目录:存放系统的内核文件和引导装载程序 ...
- use 在php 用法中的总结
1.命名空间 2.匿名函数 3.多继承 4.暂时想到这三个,如果有请补充在评论区
- 《Head First 设计模式》学习笔记——复合模式
模型-视图-控制器(MVC模式)是一种很经典的软件架构模式.在UI框架和UI设计思路中扮演着很重要的角色.从设计模式的角度来看,MVC模式是一种复合模式.它将多个设计模式在一种解决方式中结合起来,用来 ...
- hdoj-1827-Summer Holiday(scc+缩点)
Summer Holiday Time Limit: 10000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...