我有一个生产者和消费者框架。每个生产者推送到队列,而消费者从队列中消费。在任何时间点都可以有一个或多个队列,每个消费者都从单个队列中消费。但是生产者可以生产到任何队列。如果消费者很慢,它会不断堆积消息。我试图提供一个框架,在其中可以平衡使用者的负载,以便所有使用者队列都具有几乎相等的消息,而不管使用者的速度如何。
示例:
这里假设Q1-Q3队列具有几乎相等的消息,而与C1-C3消费者的比率无关。我现在使用的默认策略是对生产者进行轮询,但是如果任何消费者使用速度较慢,它将继续添加消息以排队。所有消息都属于同一类型,因此它会进入任何队列。
任何开始的建议都是有帮助的。
答案 0 :(得分:3)
简单-添加到具有最少项目数的队列中。
答案 1 :(得分:0)
以下是我已实施的解决方案。使用的算法如下。
生产者代码:
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.BlockingQueue;
public class Producer implements Runnable{
private List<BlockingQueue<Integer>> blockingQueues = new ArrayList<>();
private List<Integer> fullPartitions;
private List<Integer> activePartitions;
long timer = System.currentTimeMillis();
int THRESHOLD = 10000;
int currentQueue = 0;
public Producer(List<BlockingQueue<Integer>> blockingQueues, List<Integer> fullPartitions, List<Integer> activePartitions) {
this.blockingQueues = blockingQueues;
this.fullPartitions = fullPartitions;
this.activePartitions = activePartitions;
}
@Override
public void run() {
long start = System.currentTimeMillis();
while(true) {
blockingQueues.get(getNextID()).offer(new Random().nextInt(100000));
try {
if(System.currentTimeMillis()-start<300000)
Thread.sleep(1);
else
break;
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
private int getNextID() {
if(System.currentTimeMillis()-timer>30000) {
activePartitions = new ArrayList<>();
long mean = 0l;
for(int i=0;i<fullPartitions.size();i++)
mean += blockingQueues.get(i).size();
mean = mean/blockingQueues.size();
for(int i=0;i<fullPartitions.size();i++)
if(blockingQueues.get(i).size()-mean<THRESHOLD)
activePartitions.add(i);
timer = System.currentTimeMillis();
}
int partitionID = activePartitions.get(currentQueue%activePartitions.size());
currentQueue++;
return partitionID;
}
}
消费者:
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
public class Consumer implements Runnable{
private BlockingQueue<Integer> blockingQueue = new ArrayBlockingQueue<>(100000000);
private int delayFactor;
public Consumer(BlockingQueue<Integer> blockingQueue, int delayFactor, int consumerNo) {
this.blockingQueue = blockingQueue;
this.delayFactor = delayFactor;
}
@Override
public void run() {
long start = System.currentTimeMillis();
while(true) {
try {
blockingQueue.take();
if(blockingQueue.isEmpty())
System.out.println((System.currentTimeMillis()-start)/1000);
Thread.sleep(delayFactor);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
主线程:
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
public class KafkaLoadBalancer {
private static int MAX_PARTITION = 4;
public static void main(String args[]) throws InterruptedException {
List<BlockingQueue<Integer>> blockingQueues = new ArrayList<>();
List<Integer> fullPartitions = new ArrayList<Integer>();
List<Integer> activePartitions = new ArrayList<Integer>();
System.out.println("Creating Queues");
for(int i=0;i<MAX_PARTITION;i++) {
blockingQueues.add(new ArrayBlockingQueue<>(1000000));
fullPartitions.add(i);
activePartitions.add(i);
}
System.out.println("Starting Producers");
for(int i=0;i<MAX_PARTITION;i++) {
Producer producer = new Producer(blockingQueues,fullPartitions,activePartitions);
new Thread(producer).start();
}
System.out.println("Starting Consumers");
for(int i=0;i<MAX_PARTITION;i++) {
Consumer consumer = new Consumer(blockingQueues.get(i),i+1,i);
new Thread(consumer).start();
}
System.out.println("Starting Display Thread");
DisplayQueue dq = new DisplayQueue(blockingQueues);
new Thread(dq).start();
}
}
DispayQueue:显示队列大小
import java.util.List;
import java.util.concurrent.BlockingQueue;
public class DisplayQueue implements Runnable {
private List<BlockingQueue<Integer>> blockingQueues;
public DisplayQueue(List<BlockingQueue<Integer>> blockingQueues) {
this.blockingQueues = blockingQueues;
}
@Override
public void run() {
long start = System.currentTimeMillis();
while(true) {
if(System.currentTimeMillis()-start>30000) {
for(int i=0;i<blockingQueues.size();i++)
System.out.println("Queue "+i+" size is=="+blockingQueues.get(i).size());
start = System.currentTimeMillis();
}
}
}
}