多线程理解

1:什么是多线程?

讲到多线程,就不可避免要提到程序和进程了,他们之间的大小顺序是: 程序>进程>线程 ,在我的理解中一个程序最少会包含一个进程,一个进程最少会包含一个线程,而超过一个线程的话就叫做“多线程”,就比如我点开了某易云,这个时候就等于计算机运行了一个程序,而当我点击播放的时候,就相当于开启了一个进程,而这个进程中有播放,暂停,上一曲,下一曲,评论等等…行为状态,当我点击上一曲/下一曲的时候线程是同步 的,因为业务只需要同时播放一首歌,而线程之间默认是异步的,所以我可以边看MV边听歌边写评论,这个 行为就叫做多线程

2:线程从创建、运行到结束总是处于下面五个状态之一:新建状态、就绪状态、运行状态、阻塞状态及死亡状态。

  • 新建状态(New)

    当用new操作符创建一个线程时, 例如new Thread®,线程还没有开始运行,此时线程处在新建状态。 当一个线程处于新生状态时,程序还没有开始运行线程中的代码

  • 就绪状态(Runnable)

    一个新创建的线程并不自动开始运行,要执行线程,必须调用线程的start()方法。当线程对象调用start()方法即启动了线程,start()方法创建线程运行的系统资源,并调度线程运行run()方法。当start()方法返回后,线程就处于就绪状态。
    处于就绪状态的线程并不一定立即运行run()方法,线程还必须同其他线程竞争CPU时间,只有获得CPU时间才可以运行线程。因为在单CPU的计算机系统中,不可能同时运行多个线程,一个时刻仅有一个线程处于运行状态。因此此时可能有多个线程处于就绪状态。对多个处于就绪状态的线程是由Java运行时系统的线程调度程序(thread scheduler)来调度的。

  • 运行状态(Running)

    当线程获得CPU时间后,它才进入运行状态,真正开始执行run()方法.

  • 阻塞状态(Blocked)

    ​ 线程运行过程中,可能由于各种原因进入阻塞状态:

    ​ 1>线程通过调用sleep方法进入睡眠状态

    ​ 2>线程调用一个在I/O上被阻塞的操作,即该操作在输入输出操作完成之前不会返回到它的调用者

    ​ 3>线程试图得到一个锁,而该锁正被其他线程持有;

    ​ 4>线程在等待某个触发条件;
    ​ …

    ​ 所谓阻塞状态是正在运行的线程没有运行结束,暂时让出CPU,这时其他处于就绪状态的线程就可以获得CPU时间,进入运行状态。

  • 死亡状态(Dead)

    有两个原因会导致线程死亡:

    1. run方法正常退出而自然死亡,
    2. 一个未捕获的异常终止了run方法而使线程猝死。
      为了确定线程在当前是否存活着(就是要么是可运行的,要么是被阻塞了),需要使用isAlive方法。如果是可运行或被阻塞,这个方法返回true; 如果线程仍旧是new状态且不是可运行的, 或者线程死亡了,则返回false.

3:多线程的三种实现

  • 继承Thread类:重写run()方法

     public class MyThread extends Thread {
    	
    	public void start(){
    		System.out.println("执行start");
    	}
    	
    	public void run(){
    		System.out.println("执行run");
    	}
     
    	public static void main(String[] args) {
    		MyThread m = new MyThread();//调用MyThread
    		m.start();//启动线程
    	}
     
    } 
    
  • 实现Runnable接口,实现run()方法

    package io;
    
    public class MyRunnable implements Runnable{
    	@Override
    	public void run() {
    		System.out.println("启动线程");
    	}
    }
    
  • 实现Callable接口

    public class MyCallable implements Callable<Integer>{
        public static void main(String[] args) throws ExecutionException, InterruptedException {
            Test_2_Callable t=new Test_2_Callable();
            //FutureTask类,接收返回结果
            FutureTask<Integer> res=new FutureTask<>(t);
            //启动线程,FutureTask实现了Runnable接口
            new Thread(res).start();
            //接收返回结果
            int i=res.get();
            System.out.println(i);
        }
    
        @Override
        public Integer call() throws Exception {
            //继承Callable接口,重写call()方法,该方法返回<V>或者抛出异常
            int sum=0;
            for (int i = 1; i <=100; i++) {
                sum+=i;
            }
            return sum;
        }
    }
    
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小狐狸崽子OvO

你的鼓励将是我创造最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值