node.js+react全栈实践
利用业余时间写了个简单的项目,使用react+node.js做的一个全栈实践项目,前端参考了[React-Admin-Starter](https://github.com/veryStarters/react-admin-starter)这个项目,这个项目的自动配置路由,自动页面骨架的思路很新颖。后端是node.js+express提供接口访问,最主要的内容是mysql.js的使用和使用nginx反向代理来跨域。
1.前端parttime
前端基于框架React-Admin-Starter基本没有改动。这是一个后台管理系统,最常用的功能也就是增删改查,这里做了一些自己的调整。
1.1.统一的字段名
开发PC端这种后台项目,产品经理经常会提一些临时需求。比如原型上一个表格字段“编辑时间”,做到一般快结尾了或者已经快上线了,说要改成“更新时间”。这个时候就比较蛋疼了,当然最直接的办法就是Ctrl+H全局查找,一个一个替换,但是遇到新手连编辑器都不是很熟的小伙伴就要捉急了(我见过一些刚入门的小伙子,用的是vscode,还真不知道全局查找,快速跳转这些快捷键)。
前端项目中使用的是ant.design for react,table有两个地方需要注意,数据源和显示列名:
// 数据源
const dataSource = [
{ key: '1', name: '胡彦斌', age: 32, address: '西湖区湖底公园1号' },
{ key: '2', name: '胡彦祖', age: 42, address: '西湖区湖底公园1号' }
]; // 显示列
const columns = [
{ title: '姓名', dataIndex: 'name', key: 'name' },
{ title: '年龄', dataIndex: 'age', key: 'age' },
{ title: '住址', dataIndex: 'address', key: 'address' }
]
这里可以把所有字段单独写在一个文件里面,从同一个地方引用这个字段,这样只修改这一个字段所有的名字都改过来了。如下,columns.js 定义字段:
const id = { title: 'ID', dataIndex: 'id', key: 'id', type: 'input' }
const name = { title: '姓名', dataIndex: 'name', key: 'name', type: 'input' }
const mobile = { title: '手机号', dataIndex: 'mobile', key: 'mobile', type: 'input' }
const email = { title: '邮箱', dataIndex: 'email', key: 'email', type: 'input' }
const thumb = { title: '头像', dataIndex: 'thumb', key: 'thumb', render: src => <img alt='' src={ src }/> }
const user = [id, name, email, mobile, thumb, createTime, updateTime]
export {
user
}
user/list/index.js使用字段:
import { user } from './../../../columns' <Table
dataSource={userList}
pagination={paginationProps}
columns={user})}
rowKey='id'
size="middle"
bordered/>
问题来了,如果有编辑,删除字段怎么办呢?这个时候就需要和引用它的地方交互了。这里可以使用给子组件传递函数的方法来实现:
const action = props => {
let { handleDelete, handleEdit } = props
return {
title: '操作',
key: 'action',
render: (text, record) => <span>
<Popconfirm title='确定删除?' onConfirm={() => handleDelete(record)} okText="确定" cancelText="取消">
<Icon type="delete" className={style.deleteLink}/>
</Popconfirm>
<Divider type="vertical"/>
<Icon type="edit" onClick={() => handleEdit(record)}/>
</span>
}
}
const user = { column: props => [id, name, email, mobile, thumb, createTime, updateTime, action(props)] }
在使用这个字段的时候就可以调用一个函数:
handleDelete(record) {
api.user.deleteUser({ id: record.id }).then(res => {
if (res.success) {
this.search()
}
})
} <Table
dataSource={userList}
pagination={paginationProps}
columns={user.column({ handleDelete: this.handleDelete.bind(this), handleEdit: this.handleEdit.bind(this) })}
rowKey='id'
size="middle"
bordered/>
这里给Table的columns属性赋的是一个函数,函数参数是一个也是一个函数,这样子组件就可以调用到这个函数,有点拗口,你懂就好。columns.js中的action字段只是一个桥梁作用,根据具体逻辑传递进去的函数执行不同的操作,不同场合执行的操作不同,但是操作是类似的,基本都是删除,和编辑两个逻辑。
分页也有类似的问题,比如那天产品经理说:“分页样式统一起来,每个地方可选的每页个数都是20, 30, 50, 100”。我们也可以把这个定义在同一个地方,方便修改。这里仍然定义在columns.js中
const pageSet = { current: 1, pageSize: 2, total: 0, showQuickJumper: true, showSizeChanger: true, pageSizeOptions: ['20', '30', '50', '100'] }
使用的,如果我们要需要某些场合需要覆盖掉部分信息,可以在state中使用...扩展运算符,然后后面跟上同名属性来覆盖,例如:
import { user, pageSet } from './../../../columns'
constructor(props) {
super(props)
this.state = {
showAdd: false,
pageSet: { ...pageSet, pageSizeOptions: ['2', '10'] }
}
}
这样就不需要在每个业务逻辑里都去定义列名,只需要在columns.js中去定义,组合,导出字段就好了。这样可能也会有不妥的地方,理论上这里应该包含这个系统中所有要显示的列名,大一点的系统如果有成千上万个字段,这里就多起来了。不过话说回来这总比在每个界面自己定义字段写的代码要少。
1.2 使用同一个新增弹框
新增数据,无非是一个弹出框,一个Form加上两个按钮,没有必要为每一个界面写一个,如果能给这个弹框传入属性,包含要新增的字段,点击确定的时候调用父组件中的新增方法。这样这个弹出框被公用起来,只起到收集数据,验证数据的作用。
传入要新增的字段,一样在columns.js这个文件里做文章,一般要新增的字段和显示在表里的字段是类似的,二般不一样就难办了,这样最好还是区分开来,顶多是组合字段而已。再者,如果新增的字段时间类型,下拉框选择,上传的文件,图片怎么办呢? 可以在这个字段里加上一个type字段,表示控件类型,如下:
const email = { title: '邮箱', dataIndex: 'email', key: 'email', type: 'input' }
const createTime = { title: '创建时间', dataIndex: 'createTime', key: 'createTime', type: 'time' }
const user = { column: props => [id, name, email, mobile, thumb, createTime, updateTime, action(props)], field: [name, email, mobile, thumb]}
引入field,传递给新增组件
import { user, pageSet } from './../../../columns'
<AddComp
field={user.field}
showAdd={showAdd}
onAddData={this.addUser.bind(this)}
title={route.title}/>
在AddComp组件中使用传入的字段:
import React, { Component } from 'react'
import { Form, Modal, Input, message } from 'antd' class AddDataComp extends Component {
constructor(props) {
super(props)
this.state = {
}
}
componentWillReceiveProps(nextProps, nextContext) {
this.setState({ showAdd: nextProps.showAdd })
}
// 取消,关闭,调用父组件关闭弹框
hideModel() {
this.props.onClose()
}
// 确认,调用父组件,添加数据
confirmForm() {
this.props.form.validateFields((err, values) => {
if (err) {
message.error(err)
}
this.props.onAddData(values)
})
}
render() {
let { showAdd } = this.state
let { field, title } = this.props
let { getFieldDecorator } = this.props.form
const formItemLayout = { labelCol: { span: 6 }, wrapperCol: { span: 18 }}
return <Modal
visible={showAdd}
title={'添加' + title}
centered
onCancel={this.hideModel.bind(this)}
onOk={this.confirmForm.bind(this)}>
<Form {...formItemLayout}>
{field.map((f, index) => <Form.Item key={f.key} label={f.title}>
{getFieldDecorator(f.key, {
validateTrigger: ['onChange', 'onBlur'],
rules: [
{ required: true, whitespace: true, message: `${f.title}不能为空` },
],
})(<Input placeholder={'请输入' + f.title}/>)}
</Form.Item>)}
</Form>
</Modal>
}
}
const AddComp = Form.create({ name: 'add_comp' })(AddDataComp)
export default AddComp
未解决问题:
- 验证,不同的字段验证不同,可以在字段中传入一个RegExp来验证,复杂的验证比如密码比较,字段之间有关联的验证如何通过字段来验证,目前本人没有想到好办法
- 复杂字段,比如文件上传,传入file或者img字段可以明确表示需要上传的字段类型,这种一般是上传文件后得到一个链接,返回这个链接并写入到数据库中,暂时没有实现。
1.3 使用同一个搜索组件
同样,搜索也是根据几个字段来查询信息,这里我们可以把搜索分成两种类型:
- 简单搜索,按照更新时间来搜索,比如昨天,今天,当月,上月,名称搜索,其中昨天,今天,当月,上月做成tab的形式,名称直接输入框,并且回车搜索。这个能满足最普遍的搜索功能。
- 复杂搜索,简单搜索的基础上加上要搜索的字段。
简单搜索
复杂搜索
复杂搜索中要搜索的字段照样放在common.js中,如下:
const user = { column: props => [id, name, email, mobile, thumb, createTime, updateTime, action(props)], field: [name, email, mobile, thumb], searchField: [name, email, mobile, createTime] }
引用并使用:
import { user, pageSet } from './../../../columns'
<AddComp
field={user.field}
showAdd={showAdd}
onAddData={this.addUser.bind(this)}
title={route.title}/>
SearchComp组件:
import React, { Component } from 'react'
import { Tabs, Input, Button, DatePicker } from 'antd'
const { TabPane } = Tabs
const { Search } = Input
const { RangePicker } = DatePicker
import style from './../static/css/index.pcss'
import { Type } from 'utils' class SearchComp extends Component {
constructor(props) {
super(props)
this.state = {
moreSearch: true, // 显示更多搜索
timeSpan: [{ name: 'today', title: '今天' },
{ name: 'yesterday', title: '昨天' },
{ name: 'currentMonth', title: '本月' },
{ name: 'lastMonth', title: '上月' }],
searchObj: {}
}
}
componentDidMount() {
}
// 搜索条件
setSearchState(event, column) {
let { searchObj } = this.state
if (event.type === 'time') {
if (column[0]) {
searchObj[`${event.dataIndex}Start`] = column[0].format('YYYY-MM-DD hh:mm')
} else {
delete searchObj[`${event.dataIndex}Start`]
}
if (column[1]) {
searchObj[`${event.dataIndex}End`] = column[1].format('YYYY-MM-DD hh:mm')
} else {
delete searchObj[`${event.dataIndex}End`]
}
} else {
if (event.target.value) {
searchObj[event.target.name] = event.target.value
} else {
delete searchObj[event.target.name]
}
}
this.setState(searchObj)
}
// 简单搜索,默认搜索第一个字段
searchKeyword(value) {
let searchObj = {}
let { searchField } = this.props
if (searchField.length > 0) {
searchObj[searchField[0].key] = value
this.onSearch(searchObj)
}
}
// 回车搜索
searchEnterKeyword(e) {
if (e.target.value) {
let searchObj = {}
let { searchField } = this.props
if (searchField.length > 0) {
searchObj[searchField[0].key] = e.target.value
this.onSearch(searchObj)
}
}
}
// 条件搜索
searchClick() {
let { searchObj } = this.state
this.onSearch(searchObj)
}
// 触发父组件搜索
onSearch(searchObj) {
this.props.onSearch(searchObj)
}
// 添加,触发父组件,弹出添加框
popUpAdd() {
this.props.onAdd()
}
getSearchItem = () => {
let { searchField } = this.props
return (<div className={style.searchItem}>
{searchField.map((s, index) => {
if (s.type === 'input') { // 文本框
return <div key={s.key}>
<label htmlFor={s.key}>{s.title}</label>
<Input name={s.key} id={s.key} allowClear placeholder={s.title} onChange={this.setSearchState.bind(this)} className={style.itemInput}/>
</div>
} else if (s.type === 'time') { // 时间搜索
return <div key={s.key}>
<label htmlFor={s.key}>{s.title}</label>
<RangePicker name={s.key} id={s.key} allowClear onChange={ this.setSearchState.bind(this, s) } className={style.itemInput}/>
</div>
} else {
return null
}
})}
<div key='submit-button'>
<Button>重置</Button>
<Button type="primary" className={style.commonMarginLeft} onClick={this.searchClick.bind(this)}>搜索</Button>
</div>
</div>)
}
render() {
let { timeSpan, moreSearch } = this.state
let { onAdd } = this.props
return (<div>
<div className={style.search}>
<Tabs>{ timeSpan.map((t, i) => <TabPane tab={t.title} key={i}/>) }</Tabs>
<div className={style.searchBox}>
<Search
allowClear
className={style.itemInput}
placeholder="请输入关键字"
onPressEnter={this.searchEnterKeyword.bind(this)}
onSearch={this.searchKeyword.bind(this)}/>
<Button
onClick={() => this.setState({ moreSearch: !moreSearch })}
icon="search"
className={style.commonMarginLeft}/>
{Type.isFunction(onAdd) ? <Button
onClick={this.popUpAdd.bind(this)}
className={style.commonMarginLeft}
type="primary"
icon="plus"/> : null}
</div>
</div>
{moreSearch ? this.getSearchItem() : null}
</div>)
}
}
export default SearchComp
这里使用onChange方法来收集搜索数据,原理是给Input组件设置name,值是key,也就是字段名,onChange方法中,使用event.target.name获取字段名字,使用event.target.value获取Input的输入值,这样组成搜索数据searchObj,最后把searchObj返回给父组件。
未解决问题:
- 时间搜索一般是一个时间段,这个暂时没有实现。
- 如果搜索条件是一个下拉框选择出来的,这个要给条件渲染成下拉框,这个暂时没有实现。
1.4 mock数据和代理跨域
原框架提供自动生成mock文件的功能,项目启动后使用express启用了http应用(parttime\scripts\addone\mock-server.js),端口是10086,专门监听mock请求,在fetch(parttime\src\common\utils\fetch.js),proxyTable(parttime\src\rasConfig.js)中代理。如果不想走mock,就修改代理的target。不过上项目之后很少使用mock,增加了工作量不是?再说已经全栈开发了还要mock个啥呢?
2.后端parttimeApp
后端开发采用的express,mysql.js,pug实现的,注意这里主要写接口,pug模板基本上没有用到。这个子项目基本上是按照官方文档来写的。
使用express-generator来生成项目骨架,express的模板引擎好多,也不知道那个好,就按照官方文档中的例子给个pug来生成项目。项目中有个www文件,是启动文件,可以直接运行这个文件启动。
2.1 数据库访问
要访问接口要添加中间件body-parser,因为post,put,patch三种请求中包含请求提,node.js原生的http模块中,请求提是基于流的方式来接受,body-parser可以解析JSON,Raw,文本,URL-encoded格式的请求体。
var bodyParser = require('body-parser');
//解析 application/json
app.use(bodyParser.json());
//解析 application/x-www-form-urlencoded
app.use(bodyParser.urlencoded({ extended: false }));
//转发api/base请求
app.use('/api/base', indexRouter);
//转发api/user请求
app.use('/api/user', usersRouter);
在usersRouter就是具体的接口请求了,如下:
var express = require('express');
var router = express.Router();
var config = require('./../conf/index') /* GET users listing. */
router.get('/', function (req, res, next) {
res.send('respond with a resource');
});
这里简单的分了个层,和java,.net代码一样有router层(相当于业务逻辑层),dao层(数据访问层)。dao层里使用mysql.js访问mysql数据库。
这个地方说一下分页的逻辑,分页查询使用的是limit offset,pageSize方式,但是有个重要的信息要返回,就是数据行数,所以需要执行两次请求,这就意味这要使用回调嵌套了,这就不是很爽了,代码会成一坨。所幸mysql.js生成连接池的时候有个选项multipleStatements,把它设置成true,就可以一次执行两个sql语句,有点类似存储过程。
查询接口一般是select column1,column2 ... from table where column1=value1 and column2=value2 ... order by updateTime desc limit offset, pageSize,这样的,为了避免每次都拼接sql语句,这里写了一个统一处理函数,另外还使用current,pageSize生成offSet。
接口请求中出列current,pageSize,current字段之外的字段默认都是需要查询的字段,使用for...of方法轮询查询对象,生成where后缀。方法如下:
paging: (sql, param) => {
// 如果请求中有pageSize,使用current,pageSize生成offSet
if (param.hasOwnProperty('pageSize')) {
param.pageSize = parseInt(param.pageSize)
param.offSet = param.current <= 1 ? 0 : (param.current - 1) * param.pageSize
}
for(let key in param) {
if(!['pageSize', 'current', 'offSet'].includes(key)) {
sql[0]+= ` AND ${key}=:${key}`
sql[1]+= ` AND ${key}=:${key}`
}
}
sql[0] += ' ORDER BY updateTime DESC LIMIT :offSet, :pageSize;'
sql[1] += ' ORDER BY updateTime DESC;'
return {sql: sql.join(''), param: param}
}
2.2 转义
默认情况下使用?转义,但是我觉得这种情况有点怪,例如select * from t_user where name=? and age=? and sex=?;这样要传入的参数是一个数组,并且要时刻注意数组的顺序和sql语句中?的顺序保持一致,这是不是反人类?所幸mysql.js有提供一个配置queryFormat,自定义转义,代码如下:
queryFormat: function (sqlString, values) {
if (!values) return sqlString;
return sqlString.replace(/\:(\w+)/g, function (txt, key) {
if (values.hasOwnProperty(key)) {
return this.escape(values[key]);
}
return txt;
}.bind(this))
}
这个函数的原理是使用字符串的replace方法将sql语句中的:columnname替换成转义后的请求值,这样写sql语句就方便多了,select * from t_user where name=:name and age=:age and sex=:sex; 还有传入参数的时候就可以直接传入一个对象就好,例如{name: '张三', age: 18, sex: 'man'},见名知义,岂不是很爽?
未解决问题:
- 暂时没有考虑like,between,>,<等情况。
- 这里默认接口请求传入的字段名字和数据库中表的字段名字一致,这是不安全的。
- 使用multipleStatements设置一次执行多条语句,也不是很安全,会有sql注入危险。
3. 部署上线
部署上线首先要有域名和空间,这没啥好说的,就是买买买,不过域名不是必须的。
服务器我用的是阿里云的Ubuntu,要在里面安装nginx,node.js,npm,mysql,pm2或者forever。
mysql装好之后命令可以连接,查看,但是这不是影响工作效率,所有要用客户端连接,我用的是navicat for mysql。首先要在阿里云服务器里当前实例的安全组里配置端口访问规则,mysql使用的是3306,截图如下:
还要允许root用户从外网登陆,要修改mysql里的user表,这里不再赘述。
使用pm2启动node.js项目,防止因出错造成自动退出。pm2工具的使用就不再赘述。
最后前端使用proxyTable代理解决跨域问题的那一套,部署在服务器上就不管用了,这里没有在后端修改服务器响应头Access-Control-Allow-Origin,而是使用nginx代理,具体做法是使用vhost,将来自localhost:3332/api/路径的请求代理到本地127.0.0.1:3333。具体做法是在nginx的vhost目录下新建一个parttime.conf,内容如下:
server {
listen 3332; # 端口
server_name www.hzyayun.net hzyayun.net; # 域名
root /usr/local/app/parttime; # 站点根目录
index index.html; # 默认首页
location /api/ {
proxy_pass http://127.0.0.1:3333; # 请求转发的地址
proxy_connect_timeout 6000; # 连接超时设置
proxy_read_timeout 6000;
proxy_redirect off; # 不修改请求url
}
}
在nginx的配置文件ngxin.conf内修改http对象,在http配置的最后一行跟上include /etc/nginx/vhost/*.conf; 然后重启nginx。最后还要开放3332,3333两个端口。如下:
最后如果想用域名访问,需要在阿里云上解析域名,需要备案,太麻烦我就没有弄,直接使用域名访问:http://120.27.214.189:3332/
git地址:https://github.com/tylerdong/parttimejob
node.js+react全栈实践的更多相关文章
- node.js+react全栈实践-Form中按照指定路径上传文件并
书接上回,讲到“使用同一个新增弹框”中有未解决的问题,比如复杂的字段,文件,图片上传,这一篇就解决文件上传的问题.这里的场景是在新增弹出框中要上传一个图片,并且这个上传组件放在一个Form中,和其他文 ...
- Vue2.0+Node.js+MongoDB全栈打造商城系统 免费下载
<ignore_js_op> 课程目录||--第01章 课程介绍| 01-01 课程-导学.mp4| 01-02 前端框架回顾.mp4| 01-03 vue概况以及核心思 ...
- vue2.0+node.js+mongodb全栈打造商城
Github地址:https://github.com/ccyinghua/vue-node-mongodb-project 一.构建项目所用: vue init webpack vue-node-m ...
- Vue2.0+Node.js+MongoDB全栈打造商城系统
vue.js +axios mock数据 在main.js中 import axios from 'axios' Vue.prototype.$ajax = axios webpack.dev.con ...
- react技术栈实践(1)
本文来自网易云社区 作者:汪洋 背景 最近开发一个全新AB测试平台,思考了下正好可以使用react技术开发. 实践前技术准备 首先遇到一个概念,redux.这货还真不好理解,大体的理解:Store包含 ...
- Node.js编写CLI的实践
导语:通常而言,Node.js的应用场景有前后端分离.海量web页面渲染服务.命令行工具和桌面端应用等等.本篇文章选取CLI(Command Line Tools)这子领域,来谈谈Node.js编写C ...
- react技术栈实践(2)
本文来自网易云社区 作者:汪洋 这时候还没完,又有两个问题引出来了. 按照上面的配置,第三方库 antd 竟然也被编译了,导致样式失败. react中,一旦包裹了子组件,子组件没办法直接使用 styl ...
- Node.js + React + MongoDB 实现 TodoList 单页应用
之前用 Ant Design 开发了一个项目,因此对 React 的特性有了一定的了解,React 使用封装组件的思想,组件各自维护自己的状态和 UI, 组件之间通过 props 传递数据和方法.当状 ...
- Jhipster 一个Spring Boot + Angular/React 全栈框架
Jhipster 一个Spring Boot + Angular/React 全栈框架: https://www.jhipster.tech/
随机推荐
- Python环境的搭建(windows系统)
1.首先访问http://www.python.org/download/去下载最新的python版本. 2.安装下载包,一路next. 3.为计算机添加安装目录搭到环境变量,如图把python的安装 ...
- 从零开始用刚买的阿里云实例搭建lnmp环境(非集成包)
一.安装前 1. 更新系统软件: yum update 2. 查看是否已安装wget: rpm -qa wget 否则安装: yum install wget 3. 查看是否已安装编译器: rpm - ...
- Flink中接收端反压以及Credit机制 (源码分析)
先上一张图整体了解Flink中的反压 可以看到每个task都会有自己对应的IG(inputgate)对接上游发送过来的数据和RS(resultPatation)对接往下游发送数据, 整个反压机制通 ...
- 面经-VIVO
面试时间:2019.09.26 现场面试 面试岗位:广告推荐算法工程师/一面/正式批 面试时长:50Min 面试内容: 自我介绍 阶乘与阶乘和(复杂度高) 文本-视频论文讲解 视频排序讲解 概率题(2 ...
- Codeforces 游记
早就对这个比赛平台有所耳闻(事实上,之前打过一场div2惨的一批……)今天去打了一场div3. 首先还是吐槽一下这个毛子时区的比赛时间,从十点三十五到零点三十五……这种时间要不是在家根本没法打嘛…… ...
- Java对象的"后事处理"——垃圾回收(二)
1 先谈Finalize() finalize()能做的所有工作,使用try-finally或者其他方式都可以做得更好.更及时,所以笔者建议大家完全可以忘掉Java语言中有这个方法的存在. ——< ...
- CSPS模拟 86
看见异或两个字就孩怕 T1 按位? T2 这道异或稍水啊233 貌似可以打表找找规律 emm七种转移,有重复刷表 优化一下? T3 skyh已经接了2杯水了(实际情况他已经ak了) cbx开始抬头傻笑 ...
- 「NOIP模拟赛」数位和乘积(dp,高精)
统计方案数,要么组合数,要么递推(dp)了. 这是有模拟赛历史以来爆炸最狠的一次 T1写了正解,也想到开long long,但是开错了地方然后数组开大了结果100->0 T3看错题本来简单模拟又 ...
- 创建基于OData的Web API - Knowledge Builder API, Part II:Project Setup
本篇为Part II:Project Setup 查看第一篇<Part I: Business Scenario> 第一步,准备步骤. 准备步骤一,下载.NET Core 2.2 SDK ...
- django_1:配置文件
工程下: settings.py(建议设置成如下) DATABASES #数据库配置 DEBUG = True ...