使用 Node.js 和 ajax 进行长轮询
Posted
技术标签:
【中文标题】使用 Node.js 和 ajax 进行长轮询【英文标题】:Long polling with Node.js and ajax 【发布时间】:2013-10-01 08:39:09 【问题描述】:我有以下服务器代码。
var http = require('http');
var mysql = require('mysql');
var querystring = require('request');
var util = require('util');
var url = require('url');
var singer_name;
var currentmodif, lastmodif;
var requests=[];
var response;
var connection = mysql.createConnection(
host : 'localhost',
user : 'someone',
password : 'xxxxxxx',
database : 'rest', //mysql database to work with (optional)
);
connection.connect(); //connect to mysql
connection.query('SELECT * FROM musics WHERE id=1', function(err, rows, fields)
if (err) throw err;
singer_name=rows[0].singer_name;
currentmodif=rows[0].time_added;
);
http.createServer(function (req, res)
console.log('request received');
requests.push(
response: res,
timestamp: new Date().getTime()
);
if(req.method=='GET')
var url_parts = url.parse(req.url,true);
lastmodif = url_parts.query.timestamp;
//check_update(req, res);
).listen(9000);
setInterval(function()
var expiration = new Date().getTime() - 30000;
for (var i = requests.length - 1; i >= 0; i--)
//console.log("Request timestamp: "+requests[i].timestamp+" Expiration : "+expiration);
response = requests[i].response;
if (requests[i].timestamp < expiration)
console.log("The condition is met");
response.writeHead(200,
'Content-Type' : 'text/plain',
'Access-Control-Allow-Origin' : '*'
);
// return response
response.write('_testcb(\'ok\')', 'utf8');
response.end();
//break;
connection.query('SELECT * FROM musics WHERE id=1', function(err, rows, fields)
if (err) throw err;
currentmodif=rows[0].time_added;
//console.log("currentmodif: "+currentmodif+" lastmodif: "+lastmodif);
if (currentmodif > lastmodif)
singer_name=rows[0].singer_name;
var _arrays = 'singer_name': singer_name, 'time': currentmodif
var data = "_testcb"+"("+JSON.stringify(_arrays)+")";
response.writeHead(200,
'Content-Type' : 'text/plain',
'Access-Control-Allow-Origin' : '*'
);
if (response.end(data))
console.log("Response successfully sent");
//return false;
);
, 2000);
和客户端代码:
<html>
<head>
<script src="http://code.jquery.com/jquery-1.10.1.min.js"></script>
<title>Node.js Ajax test</title>
</head>
<body>
</body>
<script>
var timestamp = "1380020402";
function callNode()
var time = "1380020402";
$.ajax(
url: 'http://xx.xxx.xx.xxx:9000/',
dataType: "jsonp",
data: "timestamp":timestamp,
type: 'POST',
jsonpCallback: "_testcb",
cache: false,
timeout: 35000,
success: function(response, code, xhr)
if ('ok' == response)
callNode();
return false;
console.log(response);
timestamp = response.time;
// make new call
callNode();
,
error: function(jqXHR, textStatus, errorThrown)
console.log('error ' + textStatus + " " + errorThrown);
);
$(function ()
callNode();
);
</script>
</html>
我正在尝试进行长轮询。因此,在更新数据库中的数据之前,应该暂停对 ajax 请求的响应,但上述代码不起作用。我正在从不同的域发出 ajax 请求,因此使用 jsonp。
确切的问题是,当前当数据库中的数据发生更改时,不会发送响应。它时不时地工作,但并不始终可靠。
另一个问题是超时的代码块不起作用。如果请求是 30 秒,则应发送空白响应以避免 ajax 超时。
如果有人可以提供帮助,我将不胜感激。
干杯。
【问题讨论】:
【参考方案1】:我已经想通了。修改后的代码如下:
客户端:
<html>
<head>
<script src="http://code.jquery.com/jquery-1.10.1.min.js"></script>
<title>Node.js Ajax test</title>
</head>
<body>
</body>
<script>
var timestamp;
function callNode()
$.ajax(
url: 'http://xx.xxx.xx.xxx:9000/',
dataType: "jsonp",
data: "timestamp":timestamp,
//type: 'POST', //don't need this with jsonp
jsonpCallback: "_testcb",
cache: false,
timeout: 35000,
success: function(response, code, xhr)
if ('ok' == response)
console.log(response);
callNode();
return false;
console.log(response);
timestamp = response.time;
// make new call
callNode();
,
error: function(jqXHR, textStatus, errorThrown)
console.log('error ' + textStatus + " " + errorThrown);
);
$(function ()
setTimeout(callNode, 1); //call function with setTimeout in order to avoid ugly constant browser loading
);
</script>
</html>
服务器端(server.js):
var http = require('http');
var mysql = require('mysql');
var util = require('util');
var url = require('url');
var singer_name, currentmodif, lastmodif, request, response, time_of_request;
//var requests=[];
var connection = mysql.createConnection(
host : 'localhost',
user : 'someone',
password : 'xxxxxx',
database : 'rest', //mysql database to work with (optional)
);
connection.connect(); //connect to mysql
connection.query('SELECT * FROM musics WHERE id=1', function(err, rows, fields)
if (err) throw err;
singer_name=rows[0].singer_name;
currentmodif=rows[0].time_added;
);
http.createServer(function (req, res)
request = req;
response = res;
time_of_request = new Date().getTime();
console.log('request received');
if(req.method=='GET')
var url_parts = url.parse(req.url,true);
lastmodif = url_parts.query.timestamp;
req.on('error', function(e)
console.log('problem with request: ' + e.message);
);
//checkupdate();
).listen(9000);
var response;
function checkupdate()
var expiration = new Date().getTime() - 30000;
//for (var i = requests.length - 1; i >= 0; i--)
//console.log("Request timestamp: "+time_of_request+" Expiration : "+expiration);
if (time_of_request < expiration)
console.log("The condition is met");
// return response
response.write('_testcb(\'ok\')', 'utf8');
response.end();
//
connection.query('SELECT * FROM musics WHERE id=1', function(err, rows, fields)
if (err) throw err;
currentmodif=rows[0].time_added;
if (lastmodif == undefined)
lastmodif = 0;
console.log("currentmodif: "+currentmodif+" lastmodif: "+lastmodif);
if (currentmodif > lastmodif)
singer_name=rows[0].singer_name;
var _arrays = 'singer_name': singer_name, 'time': currentmodif
var data = "_testcb"+"("+JSON.stringify(_arrays)+")";
//response.writeHead(200, 'content-type':'application/json',
//'Access-Control-Allow-Origin' : '*');
//response.write(data);
response.end(data);
console.log("Response successfully sent");
//return false;
);
;
setInterval(checkupdate, 2000);
问题出在服务器端。当服务器想要回复并因此未发送响应时,响应对象不可用(未定义)。我可能忽略了 node.js 控制台中的错误。
这几乎是一个使用 MYSQL 数据库使用 node.js 进行长轮询的完整示例。此脚本将在回复 ajax 请求之前等待新数据可用。如果新数据(在 MYSQL 中)在请求后 30 秒内不可用,则会做出虚假回复,以便请求不会超时。 ajax 的成功回调中有一个条件,即在收到此演示响应时重新启动此 ajax 请求,因此使其成为无限循环。
我已经成功测试了上面的代码,它似乎工作正常。我运行了脚本,然后更新了我的数据库中的数据(主要是 time_added 字段),这触发了对来自 node.js 服务器的新数据的等待 ajax 调用的回复。
我希望这段代码可以帮助到那里的人。
在这里结帐教程以获得进一步的解释:http://www.sahilsaid.com/blog/long-polling-node-js-mysql-database-ajax/
【讨论】:
嗨。一直给函数递归调用会导致栈溢出吗? 我认为不会。我的服务器现在持续运行了一个多月,还没有造成任何溢出。也许 node.js 有解决这个问题的机制? 不会导致堆栈溢出。它实际上不是递归,因为callNode
是由 $.ajax 请求的 success
回调而不是 callNode
调用自身调用的。只要callNode
发出请求,它就会退出。以上是关于使用 Node.js 和 ajax 进行长轮询的主要内容,如果未能解决你的问题,请参考以下文章