Java多线程详解

一、区分多任务与多线程

多任务 : 同一时间处理多个任务

多线程 : 相当于开设了 多条通道,提高了CPU使用的效率

  • 例子: 多个人同时登陆游戏

    ​ 主线程-main方法 和 其他方法-子线程 同时运行

    • 普通方法调用:

      Java多线程详解

特点:只有主线程一条执行路径

    • 多线程:

      Java多线程详解

特点:多条执行路径,主线程和子线程并行交替执行

二、区分程序、进程、线程

进程 Process

  • 在OS中运行的程序=>进程 e.g.QQ 播放器 游戏...

  • 一个进程可以有多个线程!如同时听声音、看图像、看弹幕...

  • 进程是执行程序的一次执行过程,动态的,

  • 系统资源分配的单位

  • 一个进程包含至少一个线程

线程 Thread :

  • CPU调度和执行的单位(CPU时间片)
  • 独立的执行路径
  • main()称为主线程,作为系统的入口,用于执行整个程序
  • 在程序运行时,即使没有自己创建线程,后台也会有多个线程,如主线程、gc线程
  • 多线程情况下,线程的运行需要调度器(CPU)安排调度,先后顺序无法人为干预!
  • 对同一份资源操作时,会出现资源抢夺的问题,需要加入并发控制
  • 线程会带来额外的开销:CPU调度时间+并发控制开销
  • 每个线程在自己的工作内存交互,内存控制不当会造成数据不一致!

三、线程创建的方式

主要有三种 :Thread、Runnable、Callable

Thread => class 继承Thread类

Runnable => 接口 实现Runnable接口

Callable => 接口 实现Callable接口 (了解)

main() : 用户线程

gc线程: 守护线程

开启线程的方法:

  • 1、Thread

    • 自定义线程类 继承 (extends)Thread类
    • 重写run()方法,编写线程执行体
    • 创建线程对象,调用start()方法!

    注意:线程不一定立即执行,而是由CPU安排调度

    不建议使用:避免OOP单继承局限性

  • 2、实现Runnable接口

  • 定义MyRunnable类实现Runnable接口

  • 实现run()方法,编写线程执行体

  • 创建线程对象,调用start()方法启动线程

    • 推荐使用Runnable对象,因为Java单继承的局限性(接口可以多次实现) 灵活方便,方便同一个对象被对多个线程使用
public class StartThread implements Runnable{
    
    @Override
    public void run(){
        // 线程体
        for(int i = 0; i <20; i++){
            System.out.println("我在学习Java多线程!");
        }
    }
    
    public static void main(String[] args){
        // 创建实现类对象
        StartThread st = new StartThread();
        // 创建代理类对象
        Thread thread = new Thread(st);
        
        // 启动线程
        thread.start();              
    }
}
  • 3、实现Callable接口 (了解)

    (1). 实现Callable接口,需要返回值类型(Boolean String....) 相当于返回一条消息

    (2). 重写call方法,需要抛出异常

    (3). 创建目标对象

    (4). 创建执行服务

    ExecutorService ser = Executors.newFixedThreadPool(1);
    

    (5). 提交执行

    Future<Boolean>result1 = ser.submit(t1);
    

    (6). 获取结果

    boolean r1 = result1.get();
    

    (7). 关闭服务

    ser.shutdownNow();
    

静态代理? vs Thread

即 Static Proxy

静态代理模式:

  • 真实对象和代理对象都要实现同一个接口
  • 代理对象要 代理 真实对象,即
// 静态代理 You ==>真实对象 WeddingCompany代理对象
        WeddingCompany weddingCompany = new WeddingCompany(new You());
        weddingCompany.marry();

new WeddingCompany(new You()).marry();
new Thread(()-> System.out.println("这里是目标对象!")).start();

好处:

  • 代理对象可以做很多真实对象做不了的事情
  • 真实对象可以专注做自己的事情

lambda表达式

  • λ 希腊字母表中排序第十一位的字母,英文名称为Lambda

  • 作用:避免匿名内部类定义过多

  • 实质:函数式编程

    new Thread(()-> System.out.println("多线程学习中!"));
    

(params) -> expression [表达式] or

(params) -> statement[语句] or

(params) -> { statements}

为什么要使用Lambda表达式?

  • 避免匿名内部类定义过多
  • 让代码看起来很简洁
  • 去掉没有意义的代码,只留下核心逻辑

函数式接口 ?

即 Functional Interface

  • Java8 更新了lambda表达式

定义:

  • 任何接口,如果只包含唯一一个抽象方法,即为一个函数式接口

  • 对于函数式接口。可以通过Lambda表达式来创建该接口!

    public interface Runnable {
        public abstract void run();
    }
    

    总结:

    • lambda表达式只能在只有一行代码的情况下才能简化成为一行

      如果有多行代码,需要用代码块包裹(前提:接口为函数式接口且只有一个)

    • lambda表达式中的多个参数也可以去掉参数类型,要去掉就全部去掉,多个参数需要用括号包起来

线程状态 (五大状态)

  • 创建状态
  • 就绪状态
  • 阻塞状态
  • 运行状态
  • 死亡状态

创建状态 -- 启动线程 --> 就绪状态

阻塞状态 -- 阻塞解除 --> 就绪状态

就绪状态 --获得CPU资源 --> 运行状态

就绪状态 <--释放CPU资源 -- 运行状态

运行状态 -- 等待用户输入/线程休眠--> 阻塞状态

运行状态 --线程自然执行完毕/外部干涉终止线程--> 死亡状态

Java多线程详解

线程常用方法:

  • setPriority(int newPriority) 更改线程的优先级
  • static void sleep(long millis) 在指定的毫秒数内让当前正在执行的线程休眠
  • void join() 等待该线程终止(新线程此时无法进入就绪状态)
  • static void yield() 暂停当前正在执行的线程对象,并执行其他线程
  • void interrupt() 中断线程,不推荐使用
  • boolean isAlive() 测试线程是否处于活动状态

停止线程

  • 不推荐使用JDK提供的stop()、destory()方法 --> 已废弃

  • 推荐让线程自己停止下来

  • 建议使用一个标志位作为终止变量

    当flag = false 则终止线程运行

上一篇:多线程按照顺序执行


下一篇:自定义线程池代码