• JAVA之多线程


    进程与线程

    1. 进程是指运行中的应用程序,每一个进程都有自己独立的内存空间;
    2. 线程是指进程中的一个执行流程,有时也称为执行情景;
    3. 一个进程可以由多个线程组成,即在一个进程中可以同时运行多个不同的线程,它们分别执行不同的任务;
    4. 当进程内的多个线程同时运行,这种运行方式称为并发运行;

    多线程的生命周期

    在这里插入图片描述

    1. 就绪状态:就绪状态的线程又叫做可运行状态,表示当前线程具有抢夺CPU时间片的权力(CPU时间片就是执行权)。当一个线程抢夺到CPU时间片之后,就开始执行run方法,run方法的开始执行标志着线程进入运行状态。

    2. 运行状态:run方法的开始执行标志着这个线程进入运行状态,当之前占有的CPU时间片用完之后,会重新回到就绪状态继续抢夺CPU时间片,当再次抢到CPU时间之后,会重新进入run方法接着上一次的代码继续往下执行。

    3. 阻塞状态:当一个线程遇到阻塞事件,例如接收用户键盘输入,或者sleep方法等,此时线程会进入阻塞状态,阻塞状态的线程会放弃之前占有的CPU时间片。之前的时间片没了需要再次回到就绪状态抢夺CPU时间片。

    4. 锁池:在这里找共享对象的对象锁线程进入锁池找共享对象的对象锁的时候,会释放之前占有CPU时间片,有可能找到了,有可能没找到,没找到则在锁池中等待,如果找到了会进入就绪状态继续抢夺CPU时间片。(这个进入锁池,可以理解为一种阻塞状态)

    多线程的实现方式

    1. 继承java.lang.Thread类 :extends
      Thread类代表线程类,它的最主要的两个方法是:
      run():包含线程运行时所执行的代码。
      start():用于启动线程。
    package extendth;
    public class Machine extends Thread{
      public void run(){
        for(int a=0;a<50;a++)
          System.out.println(a);
      }
      
      public static void main(String args[]){
        Machine machine=new Machine();
        machine.start();  //启动machine线程
      }	
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    1. 实现Runnable接口:implements
    package runimpl;
    public class Machine implements Runnable{
      private int a=0;
      public void run(){
        for(a=0;a<50;a++){
          System.out.println(Thread.currentThread().getName()+":"+a);
          try{
            Thread.sleep(100);
          }catch(InterruptedException e){throw new RuntimeException(e);}
        }
      }
      public static void main(String args[]){
        Machine machine=new Machine();
        Thread t1=new Thread(machine);
        Thread t2=new Thread(machine);
        t1.start();
        t2.start();
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    1. 线程池实现
      先创建一个线程池工具类,再调用工具类方法实现。
    import java.util.concurrent.LinkedBlockingQueue;
    import java.util.concurrent.ThreadFactory;
    import java.util.concurrent.ThreadPoolExecutor;
    import java.util.concurrent.TimeUnit;
    import java.util.concurrent.atomic.AtomicInteger;
    
    /*
     * 线程池工具类
     * 使用的话 : 获取线程池: ThreadUtils.getInstance().getExecutor();
     *           执行任务:  ThreadUtils.getInstance().execute(Runnable runnable);
     *           杀死线程:  ThreadUtils.getInstance().remove(Runnable runnable);
     *
     * */
    public class ThreadPoolUtil {
    
        /*
         * volatile 保证变量操作的可见性,也防止指令重排序   应用就是双锁检测的单例模式
         * */
        private static volatile ThreadPoolUtil mInstance;
    
        /**
         * 核心线程池的数量,同时能够执行的线程数量
         */
        private int corePoolSize;
        /**
         * 最大线程池数量,表示当缓冲队列满的时候能继续容纳的等待任务的数量
         */
        private int maxPoolSize;
        /**
         * 存活时间
         */
        private long keepAliveTime = 1;
    
        private ThreadPoolExecutor executor;
    
        public ThreadPoolExecutor getExecutor() {
            return executor;
        }
    
        private ThreadPoolUtil() {
            //给corePoolSize赋值:当前设备可用处理器核心数*2 + 1,能够让cpu的效率得到最大程度执行(有研究论证的)
            corePoolSize = Runtime.getRuntime().availableProcessors() * 2 + 1;
            maxPoolSize = corePoolSize;
            TimeUnit unit = TimeUnit.HOURS;
            executor = new ThreadPoolExecutor(
                    //当某个核心任务执行完毕,会依次从缓冲队列中取出等待任务
                    corePoolSize,
                    // 然后new LinkedBlockingQueue(),然后maximumPoolSize,但是它的数量是包含了corePoolSize的
                    maxPoolSize,
                    //表示的是maximumPoolSize当中等待任务的存活时间
                    keepAliveTime,
                    unit,
                    //缓冲队列,用于存放等待任务,Linked的先进先出
                    new LinkedBlockingQueue<>(),
                    new DefaultThreadFactory(Thread.NORM_PRIORITY, "thread-pool-"),
                    new ThreadPoolExecutor.AbortPolicy()
            );
        }
    
        /**
         * 获取线程池单例
         */
        public static ThreadPoolUtil getInstance() {
            if (mInstance == null) {
                synchronized (ThreadPoolUtil.class) {
                    if (mInstance == null) {
                        mInstance = new ThreadPoolUtil();
                    }
                }
            }
            return mInstance;
        }
    
        /**
         * 执行任务
         */
        public void execute(Runnable runnable) {
            if (executor == null) {
                executor = new ThreadPoolExecutor(
                        corePoolSize,
                        maxPoolSize,
                        keepAliveTime,
                        TimeUnit.SECONDS,
                        new LinkedBlockingQueue<>(),
                        new DefaultThreadFactory(Thread.NORM_PRIORITY, "thread-pool-"),
                        new ThreadPoolExecutor.AbortPolicy());
            }
            if (runnable != null) {
                executor.execute(runnable);
            }
        }
    
        /**
         * 移除任务
         */
        public void remove(Runnable runnable) {
            if (runnable != null) {
                executor.remove(runnable);
            }
        }
    
        private static class DefaultThreadFactory implements ThreadFactory {
            //线程池的计数
            private static final AtomicInteger POOL_NUMBER = new AtomicInteger(1);
            //线程的计数
            private final AtomicInteger threadNumber = new AtomicInteger(1);
            private final ThreadGroup group;
            private final String namePrefix;
            private final int threadPriority;
    
            DefaultThreadFactory(int threadPriority, String threadNamePrefix) {
                this.threadPriority = threadPriority;
                this.group = Thread.currentThread().getThreadGroup();
                this.namePrefix = threadNamePrefix + POOL_NUMBER.getAndIncrement() + "-thread-";
            }
    
            @Override
            public Thread newThread(Runnable r) {
                Thread thread = new Thread(group, r, namePrefix + threadNumber.getAndIncrement(), 0);
                // 返回True该线程就是守护线程,守护线程应该永远不去访问固有资源,如:数据库、文件等。因为它会在任何时候甚至在一个操作的中间发生中断。
                if (thread.isDaemon()) {
                    thread.setDaemon(false);
                }
                thread.setPriority(threadPriority);
                return thread;
            }
        }
    
    	//测试
     	public static void main(String[] args) {
            ThreadPoolExecutor newThreadPoolExecutor=getInstance().getExecutor();
            for (int i = 0; i < 100; i++) {
                int n = i;
                newThreadPoolExecutor.execute(new Runnable() {
                    @Override
                    public void run() {
                        System.out.println(Thread.currentThread().getName() +"正在执行任务");
                    }
                });
            }
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142

    使用方法:
    获取线程池: ThreadUtils.getInstance().getExecutor();
    执行任务: ThreadUtils.getInstance().execute(Runnable runnable);
    杀死线程: ThreadUtils.getInstance().remove(Runnable runnable);

  • 相关阅读:
    AUTOSAR配置工具开发教程 – 改造篇(方法创建)
    matplotlib 笔记: contourf & contour
    机器学习--循环神经网络(RNN)3
    Hadoop运行模式、本地运行模式(官方WordCount)、完全分布式运行模式(开发重点)、scp、rsync 远程同步工具、xsync集群分发脚本
    【软件测试及质量保证】小结
    AB实验总结
    企业知识管理难?选对系统可解决90%的问题
    高效数据管理:Java助力实现Excel数据验证
    AOP+反射 批量参数校验
    目前比较好用的LabVIEW架构及其选择
  • 原文地址:https://blog.csdn.net/qq_44614878/article/details/133811065