java 多线程 Thread & Runnable 使用与区别

简介: 1.线程和进程的定义进程:是执行中一段程序,即一旦程序被载入到内存中并准备执行,它就是一个进程。进程是表示资源分配的的基本概念,又是调度运行的基本单位,是系统中的并发执行的单位。线程:单个进程中执行中每个任务就是一个线程。线程是进程中执行运算的最小单位

多线程

1.线程和进程的定义

  • 进程:是执行中一段程序,即一旦程序被载入到内存中并准备执行,它就是一个进程。进程是表示资源分配的的基本概念,又是调度运行的基本单位,是系统中的并发执行的单位。
  • 线程:单个进程中执行中每个任务就是一个线程。线程是进程中执行运算的最小单位
  • 1.2.线程进程的区别体现在几个方面:
  • 因为进程拥有独立的堆栈空间和数据段,所以每当启动一个新的进程必须分配给它独立的地址空间,建立众多的数据表来维护它的代码段、堆栈段和数据段,这对于多进程来说十分“奢侈”,系统开销比较大,而线程不一样,线程拥有独立的堆栈空间,但是共享数据段,它们彼此之间使用相同的地址空间,共享大部分数据,比进程更节俭,开销比较小,切换速度也比进程快,效率高,但是正由于进程之间独立的特点,使得进程安全性比较高,也因为进程有独立的地址空间,一个进程崩溃后,在保护模式下不会对其它进程产生影响,而线程只是一个进程中的不同执行路径。一个线程死掉就等于整个进程死掉。


  • 体现在通信机制上面,正因为进程之间互不干扰,相互独立,进程的通信机制相对很复杂,譬如管道,信号,消息队列,共享内存,套接字等通信机制,而线程由于共享数据段所以通信机制很方便。


  • 属于同一个进程的所有线程共享该进程的所有资源,包括文件描述符。而不同的进程相互独立。


  • 线程又称为轻量级进程,进程有进程控制块,线程有线程控制块;


  • 线程必定也只能属于一个进程,而进程可以拥有多个线程而且至少拥有一个线程;


  • 简而言之:开启酷狗音乐只是一个进程,同时可以听歌和下载歌曲(2个线程)

1.3.我们的理解:

  • 进程是指在系统中正在运行的一个应用程序;程序一旦运行就是进程,或者更专业化来说:进程是指程序执行时的一个实例。
  • 线程是进程的一个实体。
  • 进程——资源分配的最小单位,线程——程序执行的最小单位

2.多线程的2种或者4种实现方法

ps:网上说有4种实现方法,但是也有一部分人是持但对意见的,那么这4种方法究竟如何呢?


2.1.Java多线程实现的方式有四种

  • 继承Thread类,重写run方法
  • 实现Runnable接口,重写run方法,实现Runnable接口的实现类的实例对象作为Thread构造函数的target
  • 通过Callable和Future/FutureTask创建多线程
  • 通过线程池创建多线程

补充:

前面两种可以归结为一类:无返回值,原因很简单,通过重写run方法,run方式的返回值是void,所以没有办法返回结果。

后面两种可以归结成一类:有返回值,通过Callable接口,就要实现call方法,这个方法的返回值是Object,所以返回的结果可以放在Object对象中

3.多线程的创建之Thread的步骤

  1. 定义一个类继承Thread
  2. 重写run方法
  3. 创建子类对象,就是创建线程对象
  4. 调用start方法,开启多线程并执行,同时还会告诉jvm去调用run方法
package com.Li.xc01;
/**
 * @Description:通过继承thread实现多线程
 * @auther:Li Ya Hui
 * @Time:2021年4月20日下午7:39:10
 */
public class JiSuanQi extends Thread {
  @Override
  public void run() {
//      System.out.println("计算器:run()");
//      System.out.println("当前线程的名字:"+Thread.currentThread().getName());
    for (int i = 0; i < 10; i++) {
      System.out.println("主函数的名字\t"+getName());
            //Thread.currentThread().getName()可以缩写,但是为了语义化,一般都要写
    }
  }
  //此构造期为更改线程名字
  public JiSuanQi(String name) 
  {
    super(name);
  }
}
/**
 * @Description:  测试类 通过继承thread类实现多线程             多线程 之 Thread的步骤
 * @auther:Li Ya Hui
 * @Time:2021年4月20日下午7:17:50
 */
public class Test {
  public static void main(String[] args) {
    for (int i = 0; i < 10; i++) {
      System.out.println("主函数的名字\t"+Thread.currentThread().getName());
    }
    System.out.println("主函数001");
    System.out.println("主函数002");
    System.out.println("主函数003");
    //主函数的线程名字:main
    System.out.println("当前线程的名字"+Thread.currentThread().getName());
    //开启多线程,创建子类对象
    JiSuanQi jiSuanQi = new JiSuanQi("计算器线程");
    //当调用start方法的时候,jvm会自动执行run方法,
    //其他非主函数的线程的名字:Thread-0...... 但是可以通过有参数构造器进行修改
    jiSuanQi.start();
  }
}

通过测试发现thread里的run方法不会因为main方法的结束而受到影响

4.通过实现runnable接口实现多线程

  1. 定义类实现Runnable接口
  2. 覆盖/实现接口中的run方法
  3. 创建Thread类的对象
  4. 将Runnable接口的实现类对象作为参数传递给Thread类的构造函数
  5. 调用Thread类的start方法开启线程
//Runable 接口  的实现类
/**
 * @Description: 通过Runable实现类  给Thread传参数实现 多线程
 * @auther:Li Ya Hui
 * @Time:2021年4月20日下午8:34:21
 */
public class Prims implements Runnable{
  @Override
  public void run() {
    while (true)
    {
      try {
        //线程睡眠 1000毫秒
        Thread.sleep(1000);
        System.out.println("当前线程的内容:"+Thread.currentThread().getName());
      } catch (InterruptedException e) {//线程中断异常
        //异常处理
        System.out.println("当前"+Thread.currentThread().getName()+"线程终端异常");
      }
    }
  }
}
//测试类
package com.Li.xc02;
/**
 * @Description: 测试  runnable  给Thread传参数实现 多线程
 * @auther:Li Ya Hui
 * @Time:2021年4月21日下午3:05:40
 */
public class Test {
  public static void main(String[] args) throws InterruptedException {
    System.out.println("主线程执行的内容001");    
    //实例化runnable,
    Prims a = new Prims();
    //参数形式传递给Thread
    Thread TH = new Thread(a);
    //主线程睡眠
    Thread.sleep(2000);
    System.out.println("主线程执行的内容002");    
    //子线程开启
    TH.start();   
    System.out.println("主线程执行的内容003");
  }
}

4.2.解释说明

Thread类中包括构造函数Thread(Runnable target)和Thread(Runnable target,String name),可以床底Runnable接口,说明构造函数支持传入一个Runnable接口的对象。

构造函数Thread(Runnable target)不光可以传入Runnable接口的对象,还可以传入一个Thread类的对象。这样的好处是可以完全将Thread对象中的run()方法交由其他线程进行调用。


4.3.比较

如果一个类继承Thread,则不适合资源共享。但是如果实现了Runable接口的话,则很容易的实现资源共享。

main函数,实例化线程对象也有所不同,

extends Thread :t.start();

implements Runnable : new Thread(t).start();

使用Runnable,增加程序的健壮性,代码可以被多个线程共享,代码和数据独立

线程池只能放入实现Runable或callable类线程,不能直接放入继承Thread的类


4.4.实现Runnable的原理

  • 为什么需要定一个类去实现Runnable接口呢?继承Thread类和实现Runnable接口有啥区别呢?


  • 实现Runnable接口,避免了继承Thread类的单继承局限性。覆盖/实现Runnable接口中的run方法,将线程任务代码定义到run方法中。


  • 创建Thread类的对象,只有创建Thread类的对象才可以创建线程。线程任务已被封装到Runnable接口的run方法中,而这个run方法所属于Runnable接口的实现对象,所以将这个实现类对象作为参数传递给Thread的构造函数,这样,线程对象创建时就可以明确的得到要运行的线程的任务。


  • 简而言之:实现Runable接口进而实现接口中的run方法,然后创建实例化Thread类,将线程任务所在的对象/类以参数的形式传递给Thread的构造函数最终实现多线程


4.5.实现Runnable的好处

  • 第二种方式,实现Runnable接口避免了单继承的局限性,所以较为常用。实现Runnable接口的方式,更加的符合面向对象,线程分为两部分,一部分线程对象,一部分线程任务。继承Thread类,线程对象和线程任务耦合在一起。一旦创建Thread类的子类对象,既是线程对象,又有线程任务。实现runnable接口,将线程任务单独分离出来封装成对象,类型就是Runnable接口类型。Runnable接口对线程对象和线程任务进行解耦。
  • 简而言之:避免了单继承的局限性,安全,耦合度低

5.利用匿名内部类+runnable接口实现多线程

package com.Li.xc02;
/**
 * @Description: 利用匿名内部类+runnable接口实现多线程
 * @auther:Li Ya Hui
 * @Time:2021年4月21日下午6:49:12
 */
public class Test02 {
  public static void main(String[] args) {
    System.out.println("主函数线程打印:001");
    Thread thread = new Thread(new Runnable() 
    {
      @Override
      public void run()
      {
        System.out.println("当前线程的名字为:"+Thread.currentThread().getName());
      }
    });
    thread.start();
    System.out.println("主函数线程打印:002");
  }
}

优点:

  1. 线程任务类只是实现了Runnable接口,可以继续继承其他类,而且可以继续实现其他的功能
  2. 同一个线程任务对象可以被包装成多个线程对象
  3. 适合多个相同的程序代码的线程去共享同一个资源
  4. 实现解棍操作,代码可以被多个线程共享,线程任务代码和线程独立


6.Thread和Runnable的区别

6.1.通过继承Thread类模拟售票

package com.Li.xc03;
/**
 * 
 * @Description:买票类   测试通过继承Thread类模拟售票
 * @auther:Li Ya Hui
 * @Time:2021年4月21日下午9:04:36
 */
public class Mythread extends Thread{
  public int tickets = 5;
  public void run() {
    for (int i = 1; i < 10; i++) {
      System.out.println("当前线程为:"+getName()+":"+(tickets--));
    }
  }
  //线程添加名字  
  public Mythread(String name) {
    super(name);
    // TODO Auto-generated constructor stub
  }
}
package com.Li.xc03;
/**
 * @Description: 测试类 测试通过继承Thread类模拟售票
 * @auther:Li Ya Hui
 * @Time:2021年4月21日下午9:13:41
 */
public class Test {
  public static void main(String[] args) {
    Mythread myA = new Mythread("A窗口");
    Mythread myB = new Mythread("B窗口");
    myA.start();
    myB.start();
    //通过结果可以看成是卖了10张票但是一共就5张票,所以此时只能说明这种方式实现的时候是没有共享资源的
  }
}
  • 通过结果可以看成是卖了10张票但是一共就5张票,所以此时只能说明这种方式实现的时候是没有共享资源的 ( 没有共享tickets )

6.2.通过实现Runnable接口实现售票

可以实现共享资源的效果,但是会同时进行买票的问题,延伸出了互斥锁

synchronized (this)//同步锁    //  mutex互斥 在此处是锁的效果,即互斥锁
{
    if (tickets > 0) 
    {
      System.out.println("当前线程的名字:" + Thread.currentThread().getName() + ":" + (tickets--));
    }
}

互斥锁使得,多个线程不可同时访问一个资源

案例展示

package com.Li.xc04;
/**
 * @Description: 通过实现 Runnable接口  实现多线程-模拟售票
 * @auther:Li Ya Hui
 * @Time:2021年4月21日下午9:30:16 
 */
public class Mythread implements Runnable {
  //总票数
  public int tickets = 500;
  @Override
  public void run() {
    for (int i = 0 ;i<10000 ; i++)  {
      //this 指代当前类的意思  在此处表示为给当前的类加上互斥锁的效果
      synchronized (this)//同步锁    //  mutex互斥 在此处是锁的效果,即互斥锁
      {
        if (tickets > 0) {
          System.out.println("当前线程的名字:" + Thread.currentThread().getName() + ":" + (tickets--));
        }
      }
    }
  }
}
package com.Li.xc04;
/**
 * @Description: 测试类  测试Runnable的实现类运用多个线程来共享资源去买票,不会有数据读脏的可能
 * @auther:Li Ya Hui
 * @Time:2021年4月21日下午10:08:55
 */
public class Test {
  public static void main(String[] args) {
    //实例化我的售票类
    Mythread myRunThread1 = new Mythread();
    //实例两个线程  用来分工
    Thread thread1 = new Thread(myRunThread1, "1号窗口");
    Thread thread2 = new Thread(myRunThread1, "线程二");
    thread1.start();
    thread2.start();
    //通过结果我们发现,会有数据读脏的可能性
    //所以要使用互斥锁来避免同时读取数据问题
  }
}

Runnable总结:


  • 实现Runnable接口比继承Thread类所具有的优势:
  • 适合多个相同的程序代码的线程去处理同一个资源
  • 可以避免java中的单继承的限制
  • 增加程序的健壮性,代码可以被多个线程共享,代码和数据独立
  • 线程池只能放入实现Runable或callable类线程,不能直接放入继承Thread的类


目录
相关文章
|
20天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
82 17
|
30天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
16天前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
1月前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
|
1月前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。
|
30天前
|
Java 程序员 调度
【JavaEE】线程创建和终止,Thread类方法,变量捕获(7000字长文)
创建线程的五种方式,Thread常见方法(守护进程.setDaemon() ,isAlive),start和run方法的区别,如何提前终止一个线程,标志位,isinterrupted,变量捕获
|
30天前
|
安全 Java API
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程
|
1月前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
60 3
|
1月前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
177 2
|
1月前
|
安全 Java API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
55 6

热门文章

最新文章