JAVAEE初阶相关内容第九弹--多线程(初阶)

news2024/11/22 10:13:11

目录

定时器

定时器是什么

标准库中的定时器

实现定时器

以上的代码存在两个问题

(1)MyTask没有指定优先级

(2)阻塞队列不满足还得塞回去

以上代码还存在一个问题:

完整代码实现


定时器

定时器是什么

类似于定闹钟一样。

(1)指定特定时刻提醒

(2)指定特定时间段后提醒【在书写代码的时候更倾向于这种,这里的定时器不是提醒,而是执行一个事先准备好的代码】

在网络通信卡顿的时候,就可以使用定时器来进行”止损“。

timer.schedule( ) 这个方法的效果就是给定时器注册一个具体的任务,这个任务不会立即执行,会有一个指定的时间进行执行。

方法中有两个参数,第一个参数是Runnable,这里用的是一个Timertask,第二个参数是指定一个时间。

标准库中的定时器

实现定时器

自己手动创建一个定时器

1.让被注册的任务能够在指定时间被执行

单独在定时器内部搞一个线程,让这个线程周期性的扫描,判定任务是否到时间了,如果时间到了就执行,如果没到就再等等。

这N个任务需要使用数据结构来保存。

2.一个定时器是可以执行N个任务的,N个任务可以按照最初约定的时间按顺序执行

综上对于定时器的核心有两部分的内容:

首先有一个扫描线程,负责判断时间到/执行任务。

其次还要有一个数据结构【优先级队列】,来保存所有被注册的线程。

优先级队列:扫描线程的时候只需要扫描队首元素即可,不必遍历整个队列

此处的优先级队列会在多线程环境下使用,很明显,调用schedule是一个线程的问题,扫描是另一个线程的问题,此时就会考虑到另一种选择:PriorityBlockingQueue 

代码:

首先创建一个类,创建一个扫描线程

private Thread t = null;

创建一个阻塞队列,用来保存任务

private PriorityBlockingQueue<>

 这里就会存在一个问提,这个阻塞队列里面存的是”任务“,此时的任务应该如何表示?

可以使用Runnable 表示任务,但是Runnable只是表示了任务的内容,还需要描述任务啥时候被执行。还需要将代码继续包装一下。继续创建一个类,如下:

class MyTask{
    public MyTask(Runnable runnable, long time) {
        this.runnable = runnable;
        this.time = time;
    }

    //要执行的任务
    private Runnable runnable;
    //任务在执行的时候使用毫秒级时间戳
    private long time;
}

 

定时器类要提供一个“schedule”方法来注册任务。 

扫描线程的主要逻辑:依次去判定我们的队首元素看是否满足时间上的要求

以上的代码存在两个问题

(1)MyTask没有指定优先级

谁减谁,这个顺序不可以背,试一下就ok

(2)阻塞队列不满足还得塞回去

“忙等”(在当前的场景中不好,但是在有的情况下就是好的选择)

思考:此处的等待需要等多久?等待的时间是否是明确的?

此处的等待时间看似是明确的,实际上并不是,随时可能有新的任务到来,随时可能有线程调用schedule添加新任务。这里不能使用sleep,还是需要使用wait与notify。每次有新的任务,就进行notify一下,重新计算一下需要等待的时间,并且wait也提供了一个带有“超时时间”的版本。

以上代码还存在一个问题:

完整代码实现

class MyTask implements Comparable<MyTask>{
    public MyTask(Runnable runnable, long time) {
        this.runnable = runnable;
        this.time = time;
    }

    //要执行的任务
    private Runnable runnable;
    //任务在执行的时候使用毫秒级时间戳
    private long time;

    //获取当前任务的时间
    public long getTime(){
        return time;
    }
    //执行任务
    public void run(){
        runnable.run();
    }

    @Override
    public int compareTo(MyTask o) {
        //返回小于、大于或等于0三种情况
        //this比o小 返回小于0
        //this比o大,返回大于0
        //this和o相同,返回等于0
        //当前需要实现的是队首元素是时间最小的
        return (int) (this.time-o.time);
    }
}

//咱们自己写一个简单的定时器
class MyTimer{
    //扫描线程
    private Thread t = null;
    //一个阻塞优先级队列来保存任务
    private PriorityBlockingQueue<MyTask> queue = new PriorityBlockingQueue<>();
    //构造方法里面创建线程
    public MyTimer(){
        t = new Thread(() ->{
            while(true){
                try {
                    //取出队首元素,检查队首元素是否到时间了
                    // 如果时间没到,塞回队列中
                    //如果时间到,就执行任务
                    synchronized (this) {
                        MyTask myTask = queue.take();
                        long curTime = System.currentTimeMillis();
                        if(curTime <myTask.getTime()){
                            //还没到点,先不执行
                            queue.put(myTask);
                            //在put之后进行一个wait
                                this.wait(myTask.getTime()-curTime);
                        }else{
                            //时间到了,执行任务
                            myTask.run();
                        }
                    }
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        t.start();
    }

    //指定两个参数
    //第一个参数是任务、内容
    //第二个参数是任务、在多少毫秒后执行
    public void schedule(Runnable runnable,long after){
        //注意这里的时间上的换算
        MyTask task = new MyTask(runnable,System.currentTimeMillis() +after);
        queue.put(task);
        synchronized (this) {
            this.notify();
        }
    }

}
public class ThreadDemo25 {
    public static void main(String[] args) {
        MyTimer myTimer = new MyTimer();
        myTimer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("任务1");
            }
        },1000);

        myTimer.schedule(new Runnable() {
            @Override
            public void run() {
                System.out.println("任务2");
            }
        },2000);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1006388.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

六、全局约束中的Alldifferen和Cumulative的实现

文章目录 1、Alldifferent的内部实现1.1 防御怪物问题1.2 Alldifferent 值域传播器 2、Cumulative的内部实现THE END 1、Alldifferent的内部实现 1.1 防御怪物问题 \qquad 假设有 N N N个村庄和 K K K支军队&#xff0c;且满足 K ≥ N K \geq N K≥N&#xff0c;每支军队都与某…

android studio platform使用体验分享(as无法跳转c/c++等native源码的福音,强烈推荐)

hi&#xff0c;粉丝朋友们&#xff1a; 大家好&#xff01;这些天粉丝朋友们分享了一下Android Studio for Platform 这个最新的google开发的阅读aosp源码的工具&#xff0c;特别适合做原生系统开发。具体官方介绍如下地址&#xff1a; 参考链接&#xff1a;https://developer.…

《动手学深度学习 Pytorch版》 4.6 暂退法

import torch from torch import nn from d2l import torch as d2l4.6.1 重新审视过拟合 整节理论&#xff0c;详见书本。 4.6.2 扰动的稳健性 整节理论&#xff0c;详见书本。 4.6.3 实践中的暂退法 整节理论&#xff0c;详见书本。 4.6.4 从零开始实现 def dropout_la…

《C和指针》笔记24: 指针和间接访问

本文主要讲指针和间接访问&#xff0c;标题对应《C和指针对应的章节》&#xff0c;引用的地方是自己写的一些注释、理解和总结。 指针、间接访问和左值 先回顾一下左值和右值 左值代表着一个位置。右值代表着一个值。赋值等号左边是个左值&#xff0c;赋值等号右边是一个右值…

fabirc 将图像绘制原点定为图形内部

添加元素时&#xff0c;设置属性&#xff1a; originX: center, originY: center, 如我定义两个矩形&#xff1a; addrect () { // 矩形1var rect new fabric.Rect({top: 50,left: 100,width: 100,height: 70,fill: #F56C6C,strokeUniform: true // 限制边框宽度缩放})canva…

vue2实现自定义主题webpack-theme-color-replacer

需求&#xff1a;根据element的自定义主题色&#xff0c;之后改变element的全局所有颜色&#xff0c;解决页面刷新后主题色失效问题&#xff0c;这个需要把颜色存入到浏览器的存储中&#xff0c;如果换个浏览器就得重新选择了哈&#xff0c;如果需要在不同的浏览器保持一致的主…

langchain主要模块(四):Memory

langchain2之Memory langchain1.概念2.主要模块模型输入/输出 (Model I/O)数据连接 (Data connection)链式组装 (Chains)代理 (Agents)内存 (Memory)回调 (Callbacks) 3.MemoryConversationBufferMemoryConversationBufferWindowMemoryConversationTokenBufferMemoryConversati…

Error: svn: E155004: Run ‘svn cleanup‘ to remove locks

解决办法如下&#xff1a;点击settings 点击清除缓存按钮&#xff0c;然后再使用svn进行提交更新操作&#xff0c;但是可能还会有其它的错误&#xff0c;比如svn: E230001: Server SSL certificate verification failed&#xff0c;解决这个错误请参考我另一篇文章&#xff1a;…

【LeetCode: 2596. 检查骑士巡视方案:深度优先搜索】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

力扣 -- 300. 最长递增子序列

解题步骤&#xff1a; 参考代码&#xff1a; class Solution { public:int lengthOfLIS(vector<int>& nums) {int nnums.size();vector<int> dp(n,1);int ret1;for(int i0;i<n;i){for(int j0;j<i;j){if(nums[j]<nums[i]){dp[i]max(dp[i],dp[j]1);}}r…

分布式系统第五讲:分布式事务及实现方案

分布式系统第五讲&#xff1a;分布式事务及实现方案 事务是一个程序执行单元&#xff0c;里面的所有操作要么全部执行成功&#xff0c;要么全部执行失败。而分布式事务是指事务的参与者、支持事务的服务器、资源服务器以及事务管理器分别位于不同的分布式系统的不同节点之上。本…

微服务07-认识MQ+RabbitMQ入门

1.前言 了解同步调用和异步调用 1.1.同步调用 比如这里的支付服务&#xff0c;需要等待订单服务、短信服务…执行完毕才能执行&#xff0c;这样支付整个流程完毕需要500ms 然后如果订单、仓储等其中一个服务挂掉了&#xff0c;那么支付服务请求请求不了&#xff0c;挂掉的服…

Python实现进制转换

1 问题 如何将一个数从十进制转换成二进制&#xff0c;这个数从键盘输入。 2 方法 &#xff08;1&#xff09;输入一个十进制的数; &#xff08;2&#xff09;将这个数除以2求商和余数 &#xff08;3&#xff09;再用2去除商&#xff0c;又会得到一个商和余数&#xff0c;如此进…

企业用设备管理系统有什么好处?的修大数据管理平台的功能强大吗?

在现代工业运营中&#xff0c;设备的高效管理与维护对于维持生产稳定及提升企业竞争力具有重大意义。而设备管理系统作为一项重要工具&#xff0c;可以极大地提高企业的生产效率与设备维护的准确性。 企业使用设备管理系统的原因如下&#xff1a;   提高设备利用率&#xff1…

小白自助发卡销售系统|支持自助选号批量购买Thinkphp6.0+Layui

小白自助发卡销售系统 本系统由小白独立开发thinkphp+layui 功能特色: 1.前台下单支持自助选号。 2.前台库存显示方案增加3种。 3.自定义商品图片。 4.支持邮件通知功能具体可到后台自行体验。 5.支持批量下单购买。 6.支持易支付接口(目前用的比较多) 7.支持批量导入卡密 8.P…

从0到1理解ChatGPT原理

目录 写在前面 1.Tansformer架构模型 2.ChatGPT原理 3.提示学习与大模型能力的涌现 3.1提示学习 3.2上下文学习 3.3思维链 4.行业参考建议 4.1拥抱变化 4.2定位清晰 4.3合规可控 4.4经验沉淀 机械工业出版社京东自购链接 写在前面 2022年11月30日&#xff0c;ChatG…

【数据结构】二叉树链式结构的实现(三)

目录 一&#xff0c;二叉树的链式结构 二&#xff0c;二叉链的接口实现 1&#xff0c;二叉链的创建 2&#xff0c;接口函数 3&#xff0c;动态创立新结点 4&#xff0c;创建二叉树 5&#xff0c;前序遍历 6&#xff0c;中序遍历 7&#xff0c;后序遍历 三&#xff0c;结点个…

嵌入式学习笔记(25)串口通信的基本原理

三根通信线&#xff1a;Tx Rx GND &#xff08;1&#xff09;任何通信都要有信息作为传输载体&#xff0c;或者有线的或则无线的。 &#xff08;2&#xff09;串口通信时有线通信&#xff0c;是通过串口线来通信的。 &#xff08;3&#xff09;串口通信最少需要2根&#xff…

“熊猫杯” | 赛宁网安获网络安全优秀创新成果大赛优胜奖

9月11日&#xff0c;四川省2023年国家网络安全宣传周正式启动。由四川省委网信办指导&#xff0c;中国网络安全产业联盟&#xff08;CCIA&#xff09;主办&#xff0c;成都信息工程大学、四川省网络空间安全协会承办的“2023年网络安全优秀创新成果大赛—成都分站赛(暨四川省‘…

Linux HTTP协议

目录 1.浏览器与服务器通信过程2.HTTP请求报头&#xff08;1&#xff09;HTTP的请求报头结构&#xff08;2&#xff09;HTTP的请求方法 3.HTTP应答报头&#xff08;1&#xff09;HTTP的应答报头结构&#xff08;2&#xff09; HTTP的应答状态 1.浏览器与服务器通信过程 浏览器…