Java使用多线程发送消息
在后台管理用户信息的时候,经常会用到批量发送提醒消息,首先想到的有:
(1)、循环发送列表,逐条发送。优点是:简单,如果发送列表很少,而且没有什么耗时的操作,是比较好的一种选择,缺点是:针对大批量的发送列表,不可取,耗时,程序会出现严重的阻塞问题。
(2)、使用队列(BlockingQueue),开启多个线程,分为三个部分。一部分负责处理将发送列表放入队列;一部分负责从队列中读取并发送消息;第三部分负责监视队列是否为空及后续的操作。
(3)、以下说到的这种模式,使用Future、Callable来返回发送结果,觉得是一种比较好的方式,很简单代码也很详细,就不介绍了。
代码如下:
public class PublishMsgTest {
//创建固定大小为100 的线程池
private static ExecutorService service = Executors.newFixedThreadPool(100);
//发送消息的业务逻辑方法
public int sendMsg(List<Integer> receivers,String content){
long begin = System.nanoTime();
AtomicInteger ai = new AtomicInteger(0);
List<Future<Integer>> list = new ArrayList<>();
//循环发送消息
for(int i=0;i<receivers.size();i++){
Integer receiver = receivers.get(i);
//使用Future,Callable实现发送消息后返回发送结果
Future<Integer> future = service.submit(new Callable<Integer>() {
@Override
public Integer call() throws Exception {
//调用相对比较耗时的发送消息接口
Thread.sleep(200);
//发送消息
int resultStatus = sendMsg(receiver,content);
System.out.println("接收者【"+receiver+"】,发送结果【"+resultStatus+"】");
return resultStatus;
}
});
list.add(future);
}
System.out.println("-----------------------"+(System.nanoTime()-begin)/1000_000d+"-----------------------");
//循环接收发送结果,相当于一个使线程同步的过程,这个过程是比较耗时的
for(int i=0;i<list.size();i++){
try {
int resultStatus = list.get(i).get();
if(resultStatus == 0){//发送成功
ai.incrementAndGet();
}
} catch (Exception e) {
e.printStackTrace();
}
}
System.out.println("发送消息结束,耗时:"+(System.nanoTime()-begin)/1000_000d);
return ai.get();
}
public static void main(String[] args){
PublishMsgTest pmt = new PublishMsgTest();
//待接收人
List<Integer> receivers = new ArrayList<Integer>();
for(int i=0;i<1000;i++){
receivers.add(i);
}
String content = "群发消息_测试代码";
int successCount = pmt.sendMsg(receivers, content);
System.out.println("共有【"+receivers.size()+"】接收者,发送成功【"+successCount+"】");
}
//完成发送消息
private int sendMsg(Integer receiver, String content) {
if(receiver%2 == 0){//模拟被2整除,即为发送成功
return 0;
}
return 1;
}
以上代码的执行结果:
-----------------------14.786889-----------------------
接收者【2】,发送结果【0】
接收者【3】,发送结果【1】
接收者【4】,发送结果【0】
接收者【0】,发送结果【0】
接收者【6】,发送结果【0】
接收者【7】,发送结果【1】
接收者【1】,发送结果【1】
接收者【5】,发送结果【1】
接收者【10】,发送结果【0】
.
.
.
.
接收者【994】,发送结果【0】
接收者【993】,发送结果【1】
接收者【992】,发送结果【0】
接收者【996】,发送结果【0】
接收者【999】,发送结果【1】
接收者【998】,发送结果【0】
接收者【997】,发送结果【1】
发送消息结束,耗时:2033.053433
共有【1000】接收者,发送成功【500】
Java使用多线程发送消息的更多相关文章
- java向邮箱发送消息失败!
出现的错误如下: org.apache.commons.mail.EmailException: Sending the email to the following server failed : ...
- Java多线程分批发送消息的小例子
需求: 假设有10万个用户,现在节假日做活动,需要给每个用户发送一条活动短信,为了提高程序的效率,建议使用多线程分批发送. 这里值得注意的是: 每开一个线程都会占用CPU的资源,所以线程根据所需要的条 ...
- java socket 一个服务器对应多个客户端,可以互相发送消息
直接上代码,这是网上找的demo,然后自己根据需求做了一定的修改.代码可以直接运行 服务器端: package socket; import java.io.BufferedReader; impor ...
- Java企业微信开发_05_消息推送之发送消息(主动)
一.本节要点 1.发送消息与被动回复消息 (1)流程不同:发送消息是第三方服务器主动通知微信服务器向用户发消息.而被动回复消息是 用户发送消息之后,微信服务器将消息传递给 第三方服务器,第三方服务器接 ...
- java 操作 RabbitMQ 发送、接受消息
例子1 Producer.java import java.io.IOException; import java.util.concurrent.TimeoutException; import c ...
- java集成WebSocket向指定用户发送消息
一.WebSocket简单介绍 随着互联网的发展,传统的HTTP协议已经很难满足Web应用日益复杂的需求了.近年来,随着HTML5的诞生,WebSocket协议被提出,它实现了浏览器与服务器的全双工通 ...
- java访问微信接口发送消息
最近在开发activiti流程的时候有个需求:流程到达每个审批节点后,需要向该节点的审批人发送一个消息,提示有审批需要处理. 参考了一下微信的开发者文档和网络上的一些技术博客,现在记录一下.以便后续继 ...
- Java企业微信开发_04_消息推送之发送消息(主动)
源码请见: Java企业微信开发_00_源码及资源汇总贴 一.本节要点 1.发送消息与被动回复消息 (1)流程不同:发送消息是第三方服务器主动通知微信服务器向用户发消息.而被动回复消息是 用户发送消息 ...
- 2.技巧: 用 JAXM 发送和接收 SOAP 消息—Java API 使许多手工生成和发送消息方面必需的步骤自动化
转自:https://www.cnblogs.com/chenying99/archive/2013/05/23/3094128.html 技巧: 用 JAXM 发送和接收 SOAP 消息—Java ...
随机推荐
- [转]Vue项目全局配置微信分享思路详解
这篇文章给大家介绍了vue项目全局配置微信分享思路讲解,使用vue作为框架,使用vux作为ui组件库,具体内容详情大家跟随脚本之家小编一起学习吧 这个项目为移动端项目,主要用于接入公众号服务.项目采用 ...
- 冲刺周五——Fifth Day
#一.Fifth Day照片 #二.今日份燃尽图 #三.项目进展 * 码云团队协同环境构建完毕 * 利用Leangoo制作任务分工及生成燃尽图 * 完成AES加解密部分代码 * 用代码实现对文件的新建 ...
- day49—JavaScript阻止浏览器默认行为
转行学开发,代码100天——2018-05-04 今天主要说明一下通过JavaScript对浏览器默认行为的阻止操作.比如右键菜单的行为. 阻止默认行为的语句为: return false; 例如,阻 ...
- 接口自动化之ddt
接口自动化会用到数据驱动模式,也就是一个ddt模块 目录 1.环境准备 2.调用时标准格式 3.应用(结合excle来传值) 1.环境准备 首先,需要安装ddt模块 pip install ddt 2 ...
- Week13 - 376. Wiggle Subsequence
Week13 - 376. Wiggle Subsequence A sequence of numbers is called a wiggle sequence if the difference ...
- 关于staticmethod() 函数
说实话,我就不知这个是干什么的. 菜鸟教程写的无需实例化, 自己可以调用自己. 在同一个类面我使用到了 因为一个类了, 我可能会方法间互相调用. 类中间使用.不加这个,就会报错.无法识别这个 orig ...
- ES6标准入门 第二章:块级作用域 以及 let和const命令
一.块级作用域 1.为什么需要块级作用域? ES5中只有全局作用域和函数作用域,带来很多不合理的场景. (1)内层变量可能会覆盖外层变量: var tem = new Date(); function ...
- vue 父子component生命周期
如今前端框架都流行组件化,页面元素都可以使用组件进行高度概括,那么处理组件之间的关系就如同处理页面架构一样重要.正确理解组件之间的关系,才能让代码按照我们与预料方式工作.最近参与了一个Vue.js的项 ...
- sql select as
as 可理解为:用作.当成,作为:一般式重命名列名或者表名.例如有表table, 列 column_1,column_2 你可以写成 select column_1 as 列1,column_2 as ...
- FileSystemObject详解
FSO是FileSystemObject 或 Scripting.FileSystemObject 的缩写,为 IIS 内置组件,用于操作磁盘.文件夹或文本文件.FSO 的对象.方法和属性非常的多,这 ...