相关教程推荐:node.js视频教程
简单开始
1.安装node。https://nodejs.org/en/
2.安装ws模块
ws:是nodejs的一个websocket库,可以用来创建服务。 https://github.com/websockets/ws
3.server.js
在项目里面新建一个server.js,创建服务,指定8181端口,将收到的消息log出来。
var websocketserver = require('ws').server,wss = new websocketserver({ port: 8181 });wss.on('connection', function (ws) { console.log('client connected'); ws.on('message', function (message) { console.log(message); });});
4.建立一个client.html。
在页面上建立一个websocket的连接。用send方法发送消息。
var ws = new websocket(ws://localhost:8181); ws.onopen = function (e) { console.log('connection to server opened'); } function sendmessage() { ws.send($('#message').val()); }
页面:
<html xmlns="http://www.w3.org/1999/xhtml"><head> <meta http-equiv="content-type" content="text/html; charset=utf-8" /> <title>websocket echo demo</title> <meta name="viewport" content="width=device-width, initial-scale=1"/> <link href="../bootstrap-3.3.5/css/bootstrap.min.css" rel="stylesheet" /> <script src="../js/jquery-1.12.3.min.js"></script> <script src="../js/jquery-1.12.3.min.js"></script> <script src="../bootstrap-3.3.5/js/bootstrap.min.js"></script> <script> var ws = new websocket(ws://localhost:8181); ws.onopen = function (e) { console.log('connection to server opened'); } function sendmessage() { ws.send($('#message').val()); } </script></head><body > <p class="vertical-center"> <p class="container"> <p> </p> <form role="form" id="chat_form" onsubmit="sendmessage(); return false;"> <p class="form-group"> <input class="form-control" type="text" name="message" id="message" placeholder="type text to echo in here" value="" /> </p> <button type="button" id="send" class="btn btn-primary" onclick="sendmessage();"> send! </button> </form> </p> </p></body></html>
view code
运行之后如下,服务端即时获得客户端的消息。
模拟股票
上面的例子很简单,只是为了演示如何运用nodejs的ws创建一个websocket服务器。且可以接受客户端的消息。那么下面这个例子演示股票的实时更新。客服端只需要连接一次,服务器端会不断地发送新数据,客户端收数据后更新ui.页面如下,有五只股票,开始和停止按钮测试连接和关闭。
服务端:
1.模拟五只股票的涨跌。
var stocks = { aapl: 95.0, msft: 50.0, amzn: 300.0, goog: 550.0, yhoo: 35.0}function randominterval(min, max) { return math.floor(math.random() * (max - min + 1) + min);}var stockupdater;var randomstockupdater = function() { for (var symbol in stocks) { if(stocks.hasownproperty(symbol)) { var randomizedchange = randominterval(-150, 150); var floatchange = randomizedchange / 100; stocks[symbol] += floatchange; } } var randommstime = randominterval(500, 2500); stockupdater = settimeout(function() { randomstockupdater(); }, randommstime);}randomstockupdater();
2.连接建立之后就开始更新数据
wss.on('connection', function (ws) { var sendstockupdates = function (ws) { if (ws.readystate == 1) { var stocksobj = {}; for (var i = 0; i < clientstocks.length; i++) { var symbol = clientstocks[i]; stocksobj[symbol] = stocks[symbol]; } if (stocksobj.length !== 0) { ws.send(json.stringify(stocksobj));//需要将对象转成字符串。websocket只支持文本和二进制数据 console.log("更新", json.stringify(stocksobj)); } } } var clientstockupdater = setinterval(function () { sendstockupdates(ws); }, 1000); ws.on('message', function (message) { var stockrequest = json.parse(message);//根据请求过来的数据来更新。 console.log("收到消息", stockrequest); clientstocks = stockrequest['stocks']; sendstockupdates(ws); });
客户端:
建立连接:
var ws = new websocket("ws://localhost:8181");
onopen直接只有在连接成功后才会触发,在这个时候将客户端需要请求的股票发送给服务端。
var isclose = false; var stocks = { "aapl": 0, "msft": 0, "amzn": 0, "goog": 0, "yhoo": 0 }; function updataui() { ws.onopen = function (e) { console.log('connection to server opened'); isclose = false; ws.send(json.stringify(stock_request)); console.log("sened a mesg"); } //更新ui var changestockentry = function (symbol, originalvalue, newvalue) { var valelem = $('#' + symbol + ' span'); valelem.html(newvalue.tofixed(2)); if (newvalue < originalvalue) { valelem.addclass('label-danger'); valelem.removeclass('label-success'); } else if (newvalue > originalvalue) { valelem.addclass('label-success'); valelem.removeclass('label-danger'); } } // 处理受到的消息 ws.onmessage = function (e) { var stocksdata = json.parse(e.data); console.log(stocksdata); for (var symbol in stocksdata) { if (stocksdata.hasownproperty(symbol)) { changestockentry(symbol, stocks[symbol], stocksdata[symbol]); stocks[symbol] = stocksdata[symbol]; } } }; } updataui();
运行效果如下:只需要请求一次,数据就会不断的更新,效果是不是很赞,不用轮询,也不用长连接那么麻烦了。文章末尾会附上所有源码。
(美股的涨跌和a股的颜色是反的,即红跌绿涨)
实时聊天
上面的例子是连接建立之后,服务端不断给客户端发送数据。接下来例子是一个简单的聊天室类的例子。可以建立多个连接。
1.安装node-uuid模块,用来给每个连接一个唯一号。
2.服务端消息发送
消息类型分notification和message两种,前者是提示信息,后者是聊天内容。消息还包含一个id、昵称和消息内容。在上一节有学习到readystate有四个值,open表示连接建立可以发送消息。如果页面关闭了,为websocket.close。
function wssend(type, client_uuid, nickname, message) { for (var i = 0; i < clients.length; i++) { var clientsocket = clients[i].ws; if (clientsocket.readystate === websocket.open) { clientsocket.send(json.stringify({ "type": type, "id": client_uuid, "nickname": nickname, "message": message })); } }}
3.服务端处理连接
每新增加一个连接,都会发送一条匿名用户的加入的提示消息,如果消息中带有“/nick” 认为这一个修改昵称的消息。然后更新客户端的昵称。其他都会当做聊天消息处理。
wss.on('connection', function(ws) { var client_uuid = uuid.v4(); var nickname = "anonymoususer" + clientindex; clientindex += 1; clients.push({ "id": client_uuid, "ws": ws, "nickname": nickname }); console.log('client [%s] connected', client_uuid); var connect_message = nickname + " has connected"; wssend("notification", client_uuid, nickname, connect_message); console.log('client [%s] connected', client_uuid); ws.on('message', function(message) { if (message.indexof('/nick') === 0) { var nickname_array = message.split(' '); if (nickname_array.length >= 2) { var old_nickname = nickname; nickname = nickname_array[1]; var nickname_message = client + old_nickname + changed to + nickname; wssend(nick_update, client_uuid, nickname, nickname_message); } } else { wssend(message, client_uuid, nickname, message); } });
处理连接关闭:
var closesocket = function(custommessage) { for (var i = 0; i < clients.length; i++) { if (clients[i].id == client_uuid) { var disconnect_message; if (custommessage) { disconnect_message = custommessage; } else { disconnect_message = nickname + has disconnected; } wssend(notification, client_uuid, nickname, disconnect_message); clients.splice(i, 1); } } }; ws.on('close', function () { closesocket(); });
4.客户端
没有启动时,页面如下,change按钮用来修改昵称。
<p class="vertical-center"> <p class="container"> <ul id="messages" class="list-unstyled"></ul> <hr/> <form role="form" id="chat_form" onsubmit="sendmessage(); return false;"> <p class="form-group"> <input class="form-control" type="text" id="message" name="message" placeholder="type text to echo in here" value="" autofocus/> </p> <button type="button" id="send" class="btn btn-primary" onclick="sendmessage();"> send message </button> </form> <p class="form-group"><span>nikename:</span><input id="name" type="text" /> <button class="btn btn-sm btn-info" onclick="changname();">change</button></p> </p> </p>
view code
js:
ws = websocket(ws://localhost:8181 nickname = = 'connection to server opened' ( message == undefined) messages = document.getelementbyid('messages' messageelem = document.createelement(li (type === 'notification'= <span class=\"label label-info\">*</span> (type == 'nick_update'= <span class=\"label label-warning\">*</span>= <span class=\"label label-success\"> + nickname + </span> message_text = <h2> + preface_label + + message + </h2>= ws.onmessage = data ==id: [%s] = %s= connection closedconnection closed messagefield = document.getelementbyid('message' (ws.readystate ==== '' name = $(#name (ws.readystate ===/nick +
运行结果:
页面关闭之后,连接马上断开。
这种实时响应的体验简直不能太爽,代码也清爽了,前端体验也更好,客户端不用一直发请求,服务端不用等着被轮询。
相信看了本文案例你已经掌握了方法,更多精彩请关注其它相关文章!
推荐阅读:
javascript的var与this,{}与function
怎样将node.js部署到heroku
以上就是怎样用nodejs搭建服务器的详细内容。
