六、守护线程

目录

守护线程

 定时器

生产者和消费者模式


守护线程

java语言中线程分为两大类:
一类是:用户线程
一类是:守护线程(后台线程)
其中具有代表性的就是:垃圾回收线程(守护线程)。

​ 守护线程的特点:
​ 一般守护线程是一个死循环,所有的用户线程只要结束,守护线程自动结束。

​ 注意:主线程main方法是一个用户线程。

​ 守护线程用在什么地方呢?
​ 每天00:00的时候系统数据自动备份。这个需要使用到定时器,并且我们可以将定时器设置为守护线程。一直在那里看着,每到00:00的时候就备份一次,所有的用户线程
​ 如果结束了,守护线程自动退出,没有必要进行数据备份了。
 

package 多线程.守护线程;

/*
守护线程
 */
public class Test14 {
    public static void main(String[] args) {
        Thread t= new BakDtaThread();
        t.setName("T1");
        //启动线程之前,将线程设置为守护线程
        t.setDaemon(true);
        t.start();

        //主线程
        for (int i =0;i<10;i++){
            System.out.println(Thread.currentThread().getName() + "-->" + i);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
class BakDtaThread extends Thread{
    public void run(){
        int i = 0;
        while(true){
            System.out.println(Thread.currentThread().getName() + " --- >" + (++i));
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

启动线程之前,将线程设置为守护线程 t.setDaemon(true);

可以看到T1线程设置为守护线程之后,他会在所有线程执行完成之后执行守护线程

 定时器

构造方法

Timer()创建一个定时器
Timer(boolean isDaemon)isDaemon为true为守护线程定时器
Timer(String name)创建一个定时器,其线程名字为name
Timer(String name, boolean isDaemon)有参构造
void schedule(TimerTask task, Date firstTime, long period)安排指定的任务在指定的时间开始进行重复的固定延迟执行
void cancel()终止定时器
package 多线程.守护线程;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

class TimerTest02{
    public static void main(String[] args) {
        Timer timer = new Timer();
        String firstTimeStr = "2022-12-04 19:26:00";
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        try {
            Date firstTime = sdf.parse(firstTimeStr);
            timer.schedule(new TimerTask() {
                @Override
                public void run() {
                    Date d = new Date();
                    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                    String time = sdf.format(d);
                    System.out.println(time + ":备份日志一次!");
                }
            }, firstTime, 1000 * 5);
        } catch (ParseException e) {
            e.printStackTrace();
        }
    }
}

    从初始时间开始打印定时任务,没5s打印一次

生产者和消费者模式

方法名作用
void wait()让活动在当前对象的线程无限等待(释放之前占有的锁)
void notify()唤醒当前对象正在等待的线程(只提示唤醒,不会释放锁)
void notifyAll()唤醒当前对象全部正在等待的线程(只提示唤醒,不会释放锁)
  •  wait和notify方法不是线程对象的方法,是java中任何一个java对象都有的方法,因为这两个方法是 Object类中自带 的。
  • wait方法和notify方法不是通过线程对象调用不是这样的:t.wait(),也不是这样的:t.notify()…不对。而是 Object o = new Object();  o.wait();
  • o.wait()表示:
    ​ 让正在o对象上活动的线程进入等待状态,无期限等待,直到被唤醒为止。
    ​ o.wait();方法的调用,会让“当前线程(正在o对象上活动的线程)”进入等待状态。
  • o.notify();表示:
    ​ 唤醒正在o对象上等待的线程。还有一个notifyAll()方法:这个方法是唤醒o对象上处于等待的所有线程。

模拟生产与消费者需求

仓库采用list集合list集合中假设只能存储一个元素 1个元素就代表仓库满了 如果list集合中的元素为0,就表示仓库空了 保证list集合永远都是最多存储一个元素 必须做到这种效果:生成一个消费一个

package 多线程.守护线程;


import java.util.ArrayList;
import java.util.List;

public class ProductAndComsumer {
    public static void main(String[] args) {
        //创建仓库对象
        List list  = new ArrayList();
        //创建两个线程对象

        //一个是生产者线程
        Thread thread1 = new Thread(new Procuder(list));
        //一个是消费者线程
        Thread thread2 = new Thread(new Consumer(list));

        thread1.setName("生产者线程");
        thread2.setName("消费者线程");
        thread1.start();
        thread2.start();
    }
}
//生产线程
class Procuder implements Runnable{
    //共享的仓库
    private List list;

    public Procuder() {
    }

    public Procuder(List list) {
        this.list = list;
    }

    @Override
    public void run() {
        //一直生产
        while(true){
            synchronized (list){
                if(list.size() > 0){
                    //当前线程进入等待状态
                    try {
                        list.wait();//当前线程进入等待状态,并且释放procuder之前占有list集合的锁
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                //程序执行到这里说明,仓库是空的可以生产
                Object obj = new Object();
                list.add(obj);
                System.out.println(Thread.currentThread().getName() + "--->" + obj);
                //唤醒消费者进行消费
                list.notify();
            }
        }
    }
}
//消费线程
class Consumer implements Runnable{
    private List list;

    public Consumer() {
    }

    public Consumer(List list) {
        this.list = list;
    }

    @Override
    public void run() {
        //一直消费
        while(true){
            synchronized (list){
                if(list.size() == 0){
                    try {
                        list.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                //程序执行到这里说明仓库中有数据,进行消费
                Object obj = list.remove(0);
                System.out.println(Thread.currentThread().getName() + "---->" + obj);
                //唤醒生产者生产
                list.notify();
            }
        }
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值