[JAVA]流控及超流控后的延迟处理

流控检查(每半秒累计,因此最小留空阀值只能做到每秒2条):

import java.text.SimpleDateFormat;
import java.util.Date;
import java.lang.Thread;

/**
 * 流量控制
 *
 * @author chenx
 */
public class OverflowController {

	private int maxSendCountPerSecend; // 该条链路上流控阀值
	private Date sendTime = new Date();
	private int sendCount = 0; // 该条链路上发送的数量

	public OverflowController(int maxSendCountPerSecend) {
		if (maxSendCountPerSecend < 2) {
			maxSendCountPerSecend = 2;
		}

		this.maxSendCountPerSecend = maxSendCountPerSecend;
	}

	public int getMaxSendCountPerSecend() {
		if (getMilliseconds(new Date()) >= 500) {
			return maxSendCountPerSecend / 2;
		}

		return maxSendCountPerSecend - (maxSendCountPerSecend / 2);
	}

	/**
	 * 是否超流控
	 */
	public boolean isOverflow(int sendNum) {
		synchronized (this) {
			Date now = new Date();
			if (now.getTime() - sendTime.getTime() >= 500) {
				sendTime = now;
				sendCount = sendNum;
			} else {
				if (sendCount + sendNum > getMaxSendCountPerSecend()) {
					return true;
				} else {
					sendCount += sendNum;
				}
			}

			return false;
		}
	}

	/**
	 * 获取指定时间的毫秒数
	 */
	private int getMilliseconds(Date date) {
		SimpleDateFormat df = new SimpleDateFormat("SSS");
		return Integer.valueOf(df.format(date));
	}

	public static void main(String[] args) throws InterruptedException {
		OverflowController oc = new OverflowController(50);
		SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SSS");
		for (int i = 0; i <= 100; i++) {
			if (oc.isOverflow(1)) {
				System.out.println(i + "-isOverflow-" + df.format(new Date()));
			} else {
				System.out.println(i + "-sendOk-" + df.format(new Date()));
			}

			Thread.sleep(10);
		}
	}
}

超流控后的延迟处理,由于java中没有.net的“延迟委托”一说:

ThreadPool.RegisterWaitForSingleObject(
 WaitHandle waitObject,
      WaitOrTimerCallback callBack,
      Object state,
     int millisecondsTimeOutInterval,
     bool executeOnlyOnce

)

Java下需实现一个简单的延迟队列:

import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;

public class DelayEntry implements Delayed {

	private int count;
	private long dequeuedTimeMillis; // 出队列时间

	public int getCount() {
		return count;
	}

	public void setCount(int count) {
		this.count = count;
	}

	public long getDequeuedTimeMillis() {
		return dequeuedTimeMillis;
	}

	public DelayEntry(long delayMillis) {
		dequeuedTimeMillis = System.currentTimeMillis() + delayMillis;
	}

	@Override
	public int compareTo(Delayed o) {
		DelayEntry de = (DelayEntry) o;
		long timeout = dequeuedTimeMillis - de.dequeuedTimeMillis;
		return timeout > 0 ? 1 : timeout < 0 ? -1 : 0;
	}

	@Override
	public long getDelay(TimeUnit unit) {
		return dequeuedTimeMillis - System.currentTimeMillis();
	}
}
import java.util.concurrent.DelayQueue;

public class DelayService {

	public void run() {
		DelayQueue<DelayEntry> queue = new DelayQueue<DelayEntry>();
		DelayConsumer delayConsumer = new DelayConsumer(queue);
		delayConsumer.start();

		for (int i = 0; i < 100; i++) {
			DelayEntry de = new DelayEntry(5000);
			de.setCount(i);
			System.out.println(System.currentTimeMillis() + "--------" + de.getCount());
			queue.add(de);
		}
	}

	class DelayConsumer extends Thread {
		DelayQueue<DelayEntry> queue;
		public DelayConsumer(DelayQueue<DelayEntry> queue) {
			this.queue = queue;
		}

		public void run() {
			while (true) {
				try {
					DelayEntry de = queue.take();
					System.out.println("queue size=" + queue.size());
					System.out.println(de.getCount());
					System.out.println(System.currentTimeMillis());
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}

	public static void main(String[] args) {
		DelayService ds = new DelayService();
		ds.run();
	}
}
时间: 2024-11-05 21:33:53

[JAVA]流控及超流控后的延迟处理的相关文章

Java socket中关闭IO流后,发生什么事?(以关闭输出流为例)

声明:该博文以socket中,关闭输出流为例进行说明. 为了方便讲解,我们把DataOutputstream dout = new DataOutputStream(new BufferedOutputStream(mySocket.getOutputStream()));中的dout做为Socket输出流的代言.同样的,din是输入流的代言. 可以造成dout被关闭的操作有: 1.调用dout.close();或din.close();因为使用这种流关闭,会造成socket被关闭,所以输入输出

简练软考知识点整理-因流直散捡排控

项目质量管理中有七种基本质量工具,也称7QC工具,用于在PDCA循环的框架内解决与质量相关的问题.七种基本质量工具是(因流直散捡排控):因果图.流程图.核查表.帕累托图.直方图.控制图.散点图. (1)因果图,问题陈述放在鱼骨的头部,作为起点,用来追溯问题来源,回推到可行动的根本原因.在问题陈述中,通常把问题描述为一个要被弥补的差距或要达到的目标.通过看问题陈述和问"为什么"来发现原因,直到发现可行动的根本原因,或者列尽每根鱼骨上的合理可能性. (2)流程图,用来显示在一个或多个输入转

无比迅速敏捷地开发IOS超精美控件

目录 前言 设计 编码 PaintCode 前言 自从人生第一篇博客<IOS中的预编译指令的初步探究>问世以来 浏览量竟然达到了360多,(路过的大神勿笑!)这些浏览量使我兴奋异常但又令我黯然神伤,为何我会眼里常含泪水?因为国人伸手党达90%!!!区区只有可怜的三个评论,可怜的三个评论~ 没有鼓励~ 没有鲜花~ 也没有谩骂~ 但是我不哭 因为贱人会笑!我深信: 一日伸手党,bug终身随! 好久没打篮球了,“教练,我想打篮球”. 这次的东西标题为<无比迅速敏捷地开发IOS超精美控件>

Java I/O— 梳理各种“流”

背景 Java的核心库java.io提供了全面的IO接口.包括:文件读写.标准设备输出等.Java中IO是以流为基础进行输入输出的,所有数据被串行化写入输出流,或者从输入流读入. -- 百度百科 Java.io中有很多类,我们如何清楚地知道什么时候该用什么类呢? 目标 本文的目标是梳理java.io中各个类的关系,了解各个类的应用场景. 概念 数据源,数据流,目的地.在看关于java io的相关代码时,每次都要问问自己此时这三者各指的是谁.而本文主要是说数据流的. 归类 按不同分类方法,将各种"

Java I/O流操作(二)---缓冲流[转]

转自:http://blog.csdn.net/johnny901114/article/details/8710403 一.BufferWriter类 IO的缓冲区的存在就是为了提高效率,把要操作的数据放进缓冲区,然后一次性把缓冲区的内容写到目的地,而不是写一次就往目的地写一次. 在这里要注意的是当我们关闭了缓冲区对象实际也关闭了与缓冲区关联的流对象. BufferWriter类 try { FileWriter fw =new FileWriter("test.txt"); //使

java高级特性之IO流

缓冲流 转换流 对象流 打印流 标准输入输出流 随机访问流 数组流 有关flush():所有的处理流的输出流,最外层流需要刷新. javaIO流 1认识File类 File类的对象表示一个文件或者一个文件目录 绝对路径:包含盘符的文件完整路径 相对路径:在当前路径下的文件路径 File类中的方法,涉及到文件或文件目录的新建.删除.获取文件的路径.获取文件的大小.并没有涉及到向文件中写入或读出内容.这样的读取或写入的功能就需要IO流来完成 一般通过将File类的对象作为参数传递到流的构造器中,作为

黑马程序员——java基础---IO(input output)流字符流

黑马程序员——java基础---IO(input output)流字符流 ------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- io(input output)流特点: 1,io流用来处理数据之间的传输 2,java对数据的操作是通过流的方式: 3,java用于操作流的对象都在io包中: 4,流按操作数据分为两种:字节流和字符流: 5,流按流向分为:输入流和输出流. 注意:流只能操作数据,而不能操作文件. 3.IO流的常用基类: 1)字节流的抽象

Java基础---IO(一)---IO流概述、字符流

   IO概述 概述 1.IO流:即InputOutput的缩写. 2.特点: 1)IO流用来处理设备间的数据传输. 2)Java对数据的操作是通过流的方式. 3)Java用于操作流的对象都在IO包中. 4)流按操作数据分为两种:字节流和字符流. 5)流按流向分为:输入流和输出流. 注意:流只能操作数据,而不能操作文件. 3.IO流的常用基类: 1)字节流的抽象基流:InputStream和OutputStream 2)字符流的抽象基流:Reader和Writer 注:此四个类派生出来的子类名称

Java基础---IO(一)---IO流概述、字符流、字节流、流操作规律

第一讲     IO概述 概述 1.IO流:即InputOutput的缩写. 2.特点: 1)IO流用来处理设备间的数据传输. 2)Java对数据的操作是通过流的方式. 3)Java用于操作流的对象都在IO包中. 4)流按操作数据分为两种:字节流和字符流. 5)流按流向分为:输入流和输出流. 注意:流只能操作数据,而不能操作文件. 3.IO流的常用基类: 1)字节流的抽象基流:InputStream和OutputStream 2)字符流的抽象基流:Reader和Writer 注:此四个类派生出来