使用React、Node.js、MongoDB、Socket.IO开发一个角色投票应用的学习过程(三)
大家好,今天本人给大家带来文章《使用React、Node.js、MongoDB、Socket.IO开发一个角色投票应用的学习过程(三)》,文中内容主要涉及到MySQL、Node.js、javascript、react.js、waterline,如果你对数据库方面的知识点感兴趣,那就请各位朋友继续看下去吧~希望能真正帮到你们,谢谢!
前篇
使用React、Node.js、MongoDB、Socket.IO开发一个角色投票应用的学习过程(一)
使用React、Node.js、MongoDB、Socket.IO开发一个角色投票应用的学习过程(二)
原文第十三步,Express API路由
第一个路由是用来创建角色的
app.post('/api/characters',(req,res,next) => { let gender = req.body.gender; let characterName = req.body.name; let characterIdLookupUrl = 'https://api.eveonline.com/eve/CharacterId.xml.aspx?names=' + characterName; const parser = new xml2js.Parser(); async.waterfall([ function(callback) { request.get(characterIdLookupUrl,(err,request,xml) => { if(err) return next(err); parser.parseString(xml,(err,parsedXml) => { try { let characterId = parsedXml.eveapi.result[0].rowset[0].row[0].$.characterID; app.models.character.findOne({ characterId: characterId},(err,model) => { if(err) return next(err); if(model) { return res.status(400).send({ message: model.name + ' is alread in the database'}); } callback(err,characterId); }); } catch(e) { return res.status(400).send({ message: ' xml Parse Error'}); } }); }); }, function(characterId) { let characterInfoUrl = 'https://api.eveonline.com/eve/CharacterInfo.xml.aspx?characterID=' + characterId; console.log(characterInfoUrl); request.get({ url: characterInfoUrl },(err,request,xml) => { if(err) return next(err); parser.parseString(xml, (err,parsedXml) => { if (err) return res.send(err); try{ let name = parsedXml.eveapi.result[0].characterName[0]; let race = parsedXml.eveapi.result[0].race[0]; let bloodline = parsedXml.eveapi.result[0].bloodline[0]; app.models.character.create({ characterId: characterId, name: name, race: race, bloodline: bloodline, gender: gender },(err,model) => { if(err) return next(err); res.send({ message: characterName + ' has been added successfully!'}); }); } catch (e) { res.status(404).send({ message: characterName + ' is not a registered citizen of New Eden',error: e.message }); } }); }); } ]); });
是不是看起来和原文的基本一模一样,只不过把var 变成了let 匿名函数变成了ES6的'=>'箭头函数,虽然我用的是warterline而原文中用的是mongoose但是包括方法名基本都一样,所以我感觉waterline是在API上最接近mongoose
顺便说一下,我为什么不喜欢mongodb,仅仅是因为有一次我安装了,只往里面写了几条测试数据,按文本算最多几kb,但第二天重启机器的时候,系统提示我,我的/home分区空间不足了(双系统分区分给linux分小了本来就不大),结果一查mongodb 的data文件 有2G多,我不知道什么原因,可能是配置不对还是别的什么原因,反正,当天我就把它删除了,
完成了这个API我们就可以往数据库里添加东西了,不知道哪些用户名可以用?相当简单,反正我用的全是一名人的名字(英文名),外国人也喜欢抢注名字,嘿嘿嘿

原文第十三步,Home组件
基本保持和原文一样,只是用lodash 替换了 underscore
一开始我看到网上介绍lodash是可以无缝替换underscore,中要修改引用就可以,但是我用的版本是4.11.2已经有很多方法不一样了,还去掉了不少方法(没有去关注underscore是不是也在最新版本中有同样的改动)
原文中:
...... import {first, without, findWhere} from 'underscore'; ...... var loser = first(without(this.state.characters, findWhere(this.state.characters, { characterId: winner }))).characterId; ......
修改为:
...... import {first, filter} from 'lodash'; ...... let loser = first(filter(this.state.characters,item => item.characterId != winner )).characterId;
findWhere 在最新版本的lodash中已经不存正,我用了filter来实现相同功能。
第十四步:Express API 路由(2/2)
GET /api/characters
原文的实现方法
/** * GET /api/characters * Returns 2 random characters of the same gender that have not been voted yet. */ app.get('/api/characters', function(req, res, next) { var choices = ['Female', 'Male']; var randomGender = _.sample(choices); Character.find({ random: { $near: [Math.random(), 0] } }) .where('voted', false) .where('gender', randomGender) .limit(2) .exec(function(err, characters) { if (err) return next(err); if (characters.length === 2) { return res.send(characters); } var oppositeGender = _.first(_.without(choices, randomGender)); Character .find({ random: { $near: [Math.random(), 0] } }) .where('voted', false) .where('gender', oppositeGender) .limit(2) .exec(function(err, characters) { if (err) return next(err); if (characters.length === 2) { return res.send(characters); } Character.update({}, { $set: { voted: false } }, { multi: true }, function(err) { if (err) return next(err); res.send([]); }); }); }); });
可以看到原文中用{ random: { $near: [Math.random(), 0] } }做为查询条件从而在数据库里取出两条随机的记录返回给页面进行PK,前文说过random的类型在mysql没有类似的,所以我把这个字段删除了。本来mysql,可以用order by rand() 之类的方法但是,waterline的sort(order by rand())不被支持,所以我是把所有符合条件的记录取出来,能过lodash的sampleSize方法从所有记录中获取两天随机记录。
app.get('/api/characters', (req,res,next) => { let choice = ['Female', 'Male']; let randomGender = _.sample(choice); //原文中是通过nearby字段来实现随机取值,waterline没有实现mysql order by rand()返回随机记录,所以返回所有结果,用lodash来处理 app.models.character.find() .where({'voted': false}) .exec((err,characters) => { if(err) return next(err); //用lodash来取两个随机值 let randomCharacters = _.sampleSize(_.filter(characters,{'gender': randomGender}),2); if(randomCharacters.length === 2){ //console.log(randomCharacters); return res.send(randomCharacters); } //换个性别再试试 let oppsiteGender = _.first(_.without(choice, randomGender)); let oppsiteCharacters = _.sampleSize(_.filter(characters,{'gender': oppsiteGender}),2); if(oppsiteCharacters === 2) { return res.send(oppsiteCharacters); } //没有符合条件的character,就更新voted字段,开始新一轮PK app.models.character.update({},{'voted': false}).exec((err,characters) => { if(err) return next(err); return res.send([]); }); }); });
在数据量大的情况下,这个的方法性能上肯定会有问题,好在我们只是学习过程,数据量也不大。将就用一下,能实现相同的功能就可以了。
GET /api/characters/search
这个API之前还有两个API,和原文基本一样,所做的修改只是用了ES6的语法,就不浪费篇幅了,可以去我的github看
这一个也只是一点mongoose和waterline的一点点小区别
原文中mongoose的模糊查找是用正则来做的,mysql好像也可以,但是warterline中没有找到相关方法(它的文档太简陋了)
所以原文中
app.get('/api/characters/search', function(req, res, next) { var characterName = new RegExp(req.query.name, 'i'); Character.findOne({ name: characterName }, function(err, character) { ......
我改成了
app.get('/api/characters/search', (req,res,next) => { app.models.character.findOne({name:{'contains':req.query.name}}, (err,character) => { .....
通过contains来查找,其实就是like %sometext%的方法来实现
下面还有两个方法修改的地方也大同小异,就不仔细讲了,看代码吧
GET /api/stats
这个是原文最后一个路由了,
原文中用了一串的函数来获取各种统计信息,原作者也讲了可以优化,哪我们就把它优化一下吧
app.get('/api/stats', (req,res,next) => { let asyncTask = []; let countColumn = [ {}, {race: 'Amarr'}, {race: 'Caldari'}, {race: 'Gallente'}, {race: 'Minmatar'}, {gender: 'Male'}, {gender: 'Female'} ]; countColumn.forEach(column => { asyncTask.push( callback => { app.models.character.count(column,(err,count) => { callback(err,count); }); }) }); asyncTask.push(callback =>{ app.models.character.find() .sum('wins') .then(results => { callback(null,results[0].wins); }); } ); asyncTask.push(callback => { app.models.character.find() .sort('wins desc') .limit(100) .select('race') .exec((err,characters) => { if(err) return next(err); let raceCount = _.countBy(characters,character => character.race); console.log(raceCount); let max = _.max(_.values(raceCount)); console.log(max); let inverted = _.invert(raceCount); let topRace = inverted[max]; let topCount = raceCount[topRace]; callback(err,{race: topRace, count: topCount}); }); }); asyncTask.push(callback => { app.models.character.find() .sort('wins desc') .limit(100) .select('bloodline') .exec((err,characters) => { if(err) return next(err); let bloodlineCount = _.countBy(characters,character => character.bloodline); let max = _.max(_.values(bloodlineCount)); let inverted = _.invert(bloodlineCount); let topBloodline = inverted[max]; let topCount = bloodlineCount[topBloodline]; callback(err,{bloodline: topBloodline, count: topCount}); }); }); async.parallel(asyncTask,(err,results) => { if(err) return next(err); res.send({ totalCount: results[0], amarrCount: results[1], caldariCount: results[2], gallenteCount: results[3], minmatarCount: results[4], maleCount: results[5], femaleCount: results[6], totalVotes: results[7], leadingRace: results[8], leadingBloodline:results[9] }); }) });
我把要统计数据的字段放入一个数组countColumn通过forEach把push到asyncTask,最后两个统计方法不一样的函数,单独push,最后用async.parallel方法执行并获得结果。
underscore的max方法可以从{a:1,b:6,d:2,e:3}返回最大值,但是lodash新版中的不行,只能通过_.max(_.values(bloodlineCount))这样的方式返回最大值。
理论要掌握,实操不能落!以上关于《使用React、Node.js、MongoDB、Socket.IO开发一个角色投票应用的学习过程(三)》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

- 上一篇
- 使用docker分离服务与数据库

- 下一篇
- QQ空间技术总监:腾讯QQ空间团队的人才培养与技术传承
-
- 霸气的白羊
- 很棒,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢作者分享技术文章!
- 2023-05-17 13:05:55
-
- 义气的豆芽
- 太详细了,已加入收藏夹了,感谢老哥的这篇博文,我会继续支持!
- 2023-05-17 12:25:31
-
- 有魅力的树叶
- 这篇技术贴真及时,太细致了,很棒,码起来,关注大佬了!希望大佬能多写数据库相关的文章。
- 2023-05-07 18:13:59
-
- 数据库 · MySQL | 1天前 |
- MySQL设置中文界面,超简单教程来了!
- 332浏览 收藏
-
- 数据库 · MySQL | 1天前 | mysql 索引提示
- MySQL进阶必看!FORCE/USE/IGNOREINDEX用法大揭秘
- 182浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- 手把手教你写MySQL存储过程,小白也能轻松上手
- 163浏览 收藏
-
- 数据库 · MySQL | 1天前 | mysql group by
- MySQL分组查询优化:GROUPBY原理+索引优化超全解析
- 324浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- MySQL设置中文语言,轻松拥有中文界面
- 211浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- MySQL建库语句从入门到精通:创建数据库+设置字符集&排序规则(附实例)
- 176浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- 从零开始学MySQL数据库操作,小白轻松变大神!
- 496浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- MySQL插入日期到时间字段,轻松搞定日期格式
- 484浏览 收藏
-
- 数据库 · MySQL | 1天前 | mysql 数据压缩
- MySQL怎么实现高效压缩存储?表压缩+列式存储详细解读
- 272浏览 收藏
-
- 数据库 · MySQL | 1天前 | mysql JOIN优化
- MySQL优化JOIN操作:七大技巧教你提升关联查询速度
- 106浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- MySQL出现中文乱码?超详细解决方案一次性搞定
- 211浏览 收藏
-
- 数据库 · MySQL | 1天前 |
- MySQL主从复制这样配!搞懂这些参数,replication稳了~
- 131浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 茅茅虫AIGC检测
- 茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
- 9次使用
-
- 赛林匹克平台(Challympics)
- 探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
- 46次使用
-
- 笔格AIPPT
- SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
- 53次使用
-
- 稿定PPT
- 告别PPT制作难题!稿定PPT提供海量模板、AI智能生成、在线协作,助您轻松制作专业演示文稿。职场办公、教育学习、企业服务全覆盖,降本增效,释放创意!
- 48次使用
-
- Suno苏诺中文版
- 探索Suno苏诺中文版,一款颠覆传统音乐创作的AI平台。无需专业技能,轻松创作个性化音乐。智能词曲生成、风格迁移、海量音效,释放您的音乐灵感!
- 54次使用
-
- golang MySQL实现对数据库表存储获取操作示例
- 2022-12-22 499浏览
-
- 搞一个自娱自乐的博客(二) 架构搭建
- 2023-02-16 244浏览
-
- B-Tree、B+Tree以及B-link Tree
- 2023-01-19 235浏览
-
- mysql面试题
- 2023-01-17 157浏览
-
- MySQL数据表简单查询
- 2023-01-10 101浏览