Redis
 sql >> Cơ Sở Dữ Liệu >  >> NoSQL >> Redis

Node.js, Socket.io, Redis pub / sub âm lượng lớn, độ trễ thấp khó khăn

Tôi nghĩ đây là một câu hỏi hợp lý và đã nghiên cứu nó một thời gian ngắn. Tôi đã dành một ít thời gian để tìm kiếm các ví dụ mà bạn có thể chọn được một số mẹo hữu ích.

Ví dụ

Tôi muốn bắt đầu với các ví dụ đơn giản:

  • mã mẫu tôi nhẹ
  • Bản trình diễn Node.js + Redis Pub / Sub + socket.io

Mẫu nhẹ là một trang duy nhất (lưu ý rằng bạn sẽ muốn thay thế redis-node-client bằng một cái gì đó giống như node_redis của Matt Ranney:

/*
 * Mclarens Bar: Redis based Instant Messaging
 * Nikhil Marathe - 22/04/2010

 * A simple example of an IM client implemented using
 * Redis PUB/SUB commands so that all the communication
 * is offloaded to Redis, and the node.js code only
 * handles command interpretation,presentation and subscribing.
 * 
 * Requires redis-node-client and a recent version of Redis
 *    http://code.google.com/p/redis
 *    http://github.com/fictorial/redis-node-client
 *
 * Start the server then telnet to port 8000
 * Register with NICK <nick>, use WHO to see others
 * Use TALKTO <nick> to initiate a chat. Send a message
 * using MSG <nick> <msg>. Note its important to do a
 * TALKTO so that both sides are listening. Use STOP <nick>
 * to stop talking to someone, and QUIT to exit.
 *
 * This code is in the public domain.
 */
var redis = require('./redis-node-client/lib/redis-client');

var sys = require('sys');
var net = require('net');

var server = net.createServer(function(stream) {
    var sub; // redis connection
    var pub;
    var registered = false;
    var nick = "";

    function channel(a,b) {
    return [a,b].sort().join(':');
    }

    function shareTable(other) {
    sys.debug(nick + ": Subscribing to "+channel(nick,other));
    sub.subscribeTo(channel(nick,other), function(channel, message) {
        var str = message.toString();
        var sender = str.slice(0, str.indexOf(':'));
        if( sender != nick )
        stream.write("[" + sender + "] " + str.substr(str.indexOf(':')+1) + "\n");
    });
    }

    function leaveTable(other) {
    sub.unsubscribeFrom(channel(nick,other), function(err) {
        stream.write("Stopped talking to " + other+ "\n");
    });
    }

    stream.addListener("connect", function() {
    sub = redis.createClient();
    pub = redis.createClient();
    });

    stream.addListener("data", function(data) {
    if( !registered ) {
        var msg = data.toString().match(/^NICK (\w*)/);
        if(msg) {
        stream.write("SERVER: Hi " + msg[1] + "\n");
        pub.sadd('mclarens:inside', msg[1], function(err) {
            if(err) {
            stream.end();
            }
            registered = true;
            nick = msg[1];
// server messages
            sub.subscribeTo( nick + ":info", function(nick, message) {
            var m = message.toString().split(' ');
            var cmd = m[0];
            var who = m[1];
            if( cmd == "start" ) {
                stream.write( who + " is now talking to you\n");
                shareTable(who);
            }
            else if( cmd == "stop" ) {
                stream.write( who + " stopped talking to you\n");
                leaveTable(who);
            }
            });
        });
        }
        else {
        stream.write("Please register with NICK <nickname>\n");
        }
        return;
    }

    var fragments = data.toString().replace('\r\n', '').split(' ');
    switch(fragments[0]) {
    case 'TALKTO':
        pub.publish(fragments[1]+":info", "start " + nick, function(a,b) {
        });
        shareTable(fragments[1]);
        break;
    case 'MSG':
        pub.publish(channel(nick, fragments[1]),
            nick + ':' +fragments.slice(2).join(' '),
              function(err, reply) {
              if(err) {
                  stream.write("ERROR!");
              }
              });
        break;
    case 'WHO':
        pub.smembers('mclarens:inside', function(err, users) {
        stream.write("Online:\n" + users.join('\n') + "\n");
        });
        break;
    case 'STOP':
        leaveTable(fragments[1]);
        pub.publish(fragments[1]+":info", "stop " + nick, function() {});
        break;
    case 'QUIT':
        stream.end();
        break;
    }
    });

    stream.addListener("end", function() {
    pub.publish(nick, nick + " is offline");
    pub.srem('mclarens:inside', nick, function(err) {
        if(err) {
        sys.debug("Could not remove client");
        }
    });
    });
});

server.listen(8000, "localhost");

Tài liệu

Có rất nhiều tài liệu ở đó và apis đang thay đổi nhanh chóng trên loại ngăn xếp này, vì vậy bạn sẽ phải cân nhắc mức độ liên quan về thời gian của mỗi tài liệu.

  • luồng hoạt động của nút
  • ví dụ về xưởng đúc đám mây
  • cách đóng nút redis pubsub
  • độ trễ redis
  • redis cookbook Sử dụng Pub / Sub cho Giao tiếp Không đồng bộ
  • các mẹo chung của linkedin
  • nút redis ràng buộc
  • câu hỏi về nhóm nodejs của google

Câu hỏi liên quan

Chỉ cần một vài câu hỏi liên quan, đây là một chủ đề nóng hổi:

  • Redis pub / sub cho máy chủ trò chuyện trong node.js
  • Cách thiết kế redis pub / sub cho hệ thống nhắn tin tức thì?

Mẹo đáng chú ý (ymmv)

Tắt hoặc tối ưu hóa tổng hợp ổ cắm, sử dụng các liên kết hiệu quả, theo dõi độ trễ và đảm bảo rằng bạn không lặp lại công việc (tức là không cần xuất bản cho tất cả người nghe hai lần).




  1. Redis
  2.   
  3. MongoDB
  4.   
  5. Memcached
  6.   
  7. HBase
  8.   
  9. CouchDB
  1. Redis có sử dụng tên người dùng để xác thực không?

  2. Không thể kết nối Redis Cluster trong Elasticache với PHP bằng thư viện phpredis

  3. Làm thế nào để nhận được tất cả các công việc đang chờ xử lý trong hàng đợi laravel trên redis?

  4. 6 công cụ phân tích bộ nhớ Redis miễn phí hàng đầu

  5. Giao điểm của hai hoặc nhiều tập hợp được sắp xếp