[zmq]-消息中间件,循环队列

该文章展示了一个基于ZMQ的消费者实现,它使用Dealer模式连接到生产者,并通过ExecutorService创建工作线程池来处理接收到的数据。然后,文章进一步介绍了如何使用Disruptor和RingBuffer来替代线程池,以提高数据处理的效率和并发性能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

import org.zeromq.SocketType;
import org.zeromq.ZContext;
import org.zeromq.ZMQ;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class ZMQConsumer {

    private static final String ENDPOINT = "tcp://localhost:5555";
    private static final int NUM_WORKERS = 4;

    public static void main(String[] args) {
        // 创建ZeroMQ上下文
        try (ZContext context = new ZContext()) {
            // 创建Dealer模式的套接字
            ZMQ.Socket socket = context.createSocket(SocketType.DEALER);
            // 连接到生产者端点
            socket.connect(ENDPOINT);

            // 创建工作线程池
            ExecutorService workerThreadPool = Executors.newFixedThreadPool(NUM_WORKERS);

            for (int i = 0; i < NUM_WORKERS; i++) {
                workerThreadPool.submit(new Worker(context, socket));
            }

            // 循环监听套接字的事件
            while (!Thread.currentThread().isInterrupted()) {
                try {
                    // 从套接字接收数据
                    byte[] data = socket.recv();
                    
                    // 将数据发送给工作线程池处理
                    workerThreadPool.submit(() -> {
                        // 对数据进行处理
                        // ...
                        // 将结果发送回生产者
                        socket.send(data);
                    });
                } catch (ZMQException e) {
                    if (!e.getMessage().contains("Operation cannot be accomplished in current state")) {
                        throw e;
                    }
                }
            }

        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }

    private static class Worker implements Runnable {
        private final ZMQ.Socket workerSocket;

        public Worker(ZContext context, ZMQ.Socket socket) {
            this.workerSocket = context.createSocket(SocketType.DEALER);
            // 连接到生产者端点
            this.workerSocket.connect(ENDPOINT);
        }

        @Override
        public void run() {
            while (true) {
                try {
                    // 从队列中取出任务
                    Runnable task = workerThreadPool.take();
                    // 执行任务
                    task.run();
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                }
            }
        }
    }
}

优化的代码

import com.lmax.disruptor.RingBuffer;
import com.lmax.disruptor.dsl.Disruptor;
import org.zeromq.SocketType;
import org.zeromq.ZContext;
import org.zeromq.ZMQ;

import java.util.concurrent.Executors;

public class ZMQConsumer {

    private static final String ENDPOINT = "tcp://localhost:5555";
    private static final int NUM_WORKERS = 4;
    private static final int BUFFER_SIZE = 1024;

    public static void main(String[] args) {
        // 创建ZeroMQ上下文
        try (ZContext context = new ZContext()) {
            // 创建Dealer模式的套接字
            ZMQ.Socket socket = context.createSocket(SocketType.DEALER);
            // 连接到生产者端点
            socket.connect(ENDPOINT);

            // 创建Disruptor实例
            Disruptor<byte[]> disruptor = new Disruptor<>(() -> new byte[BUFFER_SIZE], BUFFER_SIZE, Executors.defaultThreadFactory());
            RingBuffer<byte[]> ringBuffer = disruptor.getRingBuffer();

            // 创建工作线程池
            for (int i = 0; i < NUM_WORKERS; i++) {
                disruptor.handleEventsWith(new Worker(context, socket));
            }

            // 启动Disruptor
            disruptor.start();

            // 循环监听套接字的事件
            while (!Thread.currentThread().isInterrupted()) {
                try {
                    // 从套接字接收数据
                    byte[] data = socket.recv();

                    // 将数据添加到Disruptor中
                    long sequence = ringBuffer.next();
                    try {
                        byte[] buffer = ringBuffer.get(sequence);
                        System.arraycopy(data, 0, buffer, 0, data.length);
                    } finally {
                        ringBuffer.publish(sequence);
                    }
                } catch (ZMQException e) {
                    if (!e.getMessage().contains("Operation cannot be accomplished in current state")) {
                        throw e;
                    }
                }
            }

        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }

    private static class Worker implements EventHandler<byte[]> {
        private final ZMQ.Socket workerSocket;

        public Worker(ZContext context, ZMQ.Socket socket) {
            this.workerSocket = context.createSocket(SocketType.DEALER);
            // 连接到生产者端点
            this.workerSocket.connect(ENDPOINT);
        }

        @Override
        public void onEvent(byte[] data, long sequence, boolean endOfBatch) {
            // 对数据进行处理
            // ...

            // 将结果发送回生产者
            workerSocket.send(data);
        }
    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值