【JAVAEE学习】探究Java中多线程的使用和重点及考点

简介: 【JAVAEE学习】探究Java中多线程的使用和重点及考点

一.线程

 1.什么是线程

线程(Thread)是程序中执行的最小单元,是操作系统能线程是计算机科学中的基本概念,指的是在同一进程中执行的单一执行流。线程是操作系统能够进行运算调度的最小单位。在多线程编程中,多个线程可以同时执行,共享进程的资源,但每个线程有自己的执行流程和栈空间。线程的引入可以提高程序的并发性和响应性,允许程序在同一时间处理多个任务。

在计算机系统中,线程是由操作系统调度和管理的,可以实现不同线程之间的切换和协同工作。线程之间可以共享进程的资源,如内存空间、文件描述符等,但也需要注意线程之间的同步和互斥,以避免竞态条件和数据不一致问题。

在编程中,线程可以用来实现并发编程,允许程序同时执行多个任务,提高程序的性能和效率。常见的多线程编程模型包括线程池、生产者-消费者模型等

2.进程和线程的区别

这个先解释一下线程和进程的区别,这个两个词虽然看起来相差不大,但意思却相差很大,当对于初学者来说还是比较容易混淆的

1.在这里先解释一下最基础的定义

   进程:在计算机操作系统中,进程是对运行中程序的一个抽象,它是系统进行资源分配和调度的基本单位。每个进程都有自己独立的地址空间,包含了程序代码、数据、打开的文件描述符等资源。当一个程序开始执行时,操作系统会为其创建一个新的进程,使得程序可以占用系统资源并独立运行

    线程:线程是进程内的一个执行路径,也是CPU调度的最小单位一个进程中可以有一个或多个线程(每个进程都有最基本的主线程),这些线程共享进程的相同地址空间(包括代码、全局变量等资源)。每个线程都维护有自己的程序计数器、寄存器集合和栈,这样它们就可以在进程的上下文中并发执行各自的任务。相较于进程,线程之间的切换成本更低,而且由于资源共享的特性,线程间的通信和同步更为简便。

2.举个我们日常生活中的小例子

用日常生活的例子来解释进程和线程的区别:

进程比喻: 想象一个餐厅,每个餐厅就是一个进程,它有自己独立的厨房(资源)、服务员(线程)、菜单(程序)、餐桌(内存空间)和客户账单(数据)。如果有两家不同的餐厅(两个进程),它们各自运营,拥有各自的食材和设备,互不影响,也不能直接共享对方的东西。

线程比喻: 回到同一个餐厅内,如果餐厅有多个服务员(线程),他们共享餐厅的所有资源(如厨房、菜单等),并且能在同一餐厅的不同区域同时服务多个顾客。比如,一个服务员负责点菜(处理请求),另一个负责上菜(执行任务),还有一个负责结账(清理资源)。虽然他们在同一片工作区(地址空间)内同时活动,但会通过合理的协调(例如,加锁机制)来避免冲突。

总结来说,在这个比喻中:

  • 进程就像是独立运作的餐厅,每个餐厅有一套完整的设施和人员;
  • 线程则是同一餐厅内的不同服务员,他们共享餐厅资源并在其中执行各自的任务,可以同时服务于不同的顾客,实现并发操作。

3.线程和进程的区别以及线程相比于进程的优点:(这个面试中也算是比较高频的问题)

  1. 资源分配
  • 进程:进程是操作系统进行资源分配和保护的基本单位,每个进程都有独立的内存空间,其中包括代码段、数据段、堆和栈。这意味着不同进程之间无法直接访问彼此的内存空间,从而保证了进程间的隔离性。
  • 线程:线程是进程内部的执行实体,是系统调度和分配CPU的基本单位。同一进程内的所有线程共享相同的地址空间(包括全局变量、文件描述符等资源),也就是说,线程间可以直接读写同一进程内的内存,无需通过IPC(进程间通信)机制。
  1. 创建和切换开销
  • 进程创建新进程需要分配独立的地址空间和其他相关资源,因此开销较大。进程间的切换除了保存和恢复CPU上下文外,还可能涉及虚拟内存、页表等映射的切换,开销相对较高。
  • 线程创建线程的成本比进程低得多,因为它不需要额外分配地址空间。线程间的切换只需要保存和恢复少量寄存器状态(如程序计数器、栈指针等),因此线程切换的开销较小。
  1. 并发性和并行性
  • 进程:进程提供了并发执行的能力,即在单个处理器上通过时间片轮转实现看似同时运行的效果,而在多处理器环境下,则可以真正地并行执行不同的进程。
  • 线程:线程提供了更加细粒度的并发执行,一个进程中的多个线程可以在单个处理器上通过时间片轮转并发执行,也可以在多核处理器上真正并行执行。
  1. 通信与同步
  • 进程:进程间的通信通常需要使用IPC机制,如管道、信号量、消息队列、共享内存等。
  • 线程:由于同一进程内的线程共享内存空间,它们之间的通信和同步可以通过更简单的机制实现,如锁、条件变量等。
  1. 管理复杂性
  • 进程:进程管理相对复杂,需要考虑资源的独立性和安全性。
  • 线程:线程管理更加灵活,但由于线程间的资源共享特性,可能导致竞态条件死锁等(涉及到线程安全问题)问题,因此线程同步和互斥问题较为复杂。

综上所述,进程是操作系统中资源分配和保护的基本边界,而线程则是提供更高效并发执行能力的基础,并且在线程之间更容易共享信息和协同工作。

二.编写多线程代码

1.定义线程类

class MyThread extends Thread {
    @Override
    public void run() {
        while (true) {
            System.out.println("hello Thread");
            try {
                Thread.sleep(1000);//使线程休眠
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
public class Demo1 {
 
    }
    public static void main(String[] args)  {
        Thread thread = new MyThread();
        thread.start();//创建线程
        while (true) {
            System.out.println("hello main");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

运行结果:

我们可以看到主线程和Thread线程在交替打印各自的信息需要注意的是,实际运行时,由于线程调度机制,线程间的执行并不是严格按照交替顺序进行的,而是随机交错的。

同时我们还可以通过命令行的方式查看线程运行的状态

WIN + R 后输入cmd打开命令行后输入jconsole

点击我们刚才创建好的线程

看到这里提醒不安全连接我们不用理会直接点击

此时我们就清楚的看见线程的状态以及其它信息

1.说明

我们可以看到如果使单线程运行的化,此时陷入了死循环,应该无限循环不断地输出"hello Thread",但就是因为多线程操作并发操作程序实现了两个并发执行的线程:主线程和MyThread子线程。这两个线程交替打印各自的信息,同时还可以发现,打印时并不是严格按照交替顺序进行,而是随机交错但由于线程调度的不确定性,(抢占执行)实际输出的结果可能会出现交错现象,比如连续输出多次"hello Thread"后再输出几次"hello main",或者是反过来的情况,这就是线程随机交错的现象。

同时在这里解释一下

1.首先定义了一个名为MyThread的类,该类继承自java.lang.Thread。在MyThread类中重写了run()方法,这个方法是线程需要执行的任务主体。在run()方法内部,有一个无限循环不断地输出"hello Thread",并且每次循环之间让当前线程睡眠1秒通过Thread.sleep(1000)实现并且实现这个方法需要抛异常

2.创建线程实例:在main方法中,创建了MyThread类的一个实例thread。启动线程:调用thread.start()方法来启动线程。这会让JVM找到这个线程对象的run()方法,并在一个新的线程上下文中执行它

为什么要调用thread.start() (这一点很重要)

在Java中,当你想要启动一个新的线程去执行特定任务时,你需要调用Thread对象的start()方法。这是因为start()方法的作用是让JVM创建一个新的线程,并在这个新线程中调用你之前重写的run()方法

具体来说:

  • 当你创建一个Thread对象实例时,只是在内存中构建了一个表示线程的对象,并没有真正开启一个新的执行流。
  • 而当你调用start()方法时,Java虚拟机会为此线程分配必要的系统资源(如内存),并在某个时刻将该线程放入可执行线程队列等待调度。
  • 线程调度器会选择合适的时机将该线程从就绪状态转为运行状态,这时run()方法才会在新创建的线程上下文中执行。

因此,如果你直接调用run()方法而不是start()方法,那么代码将在当前线程(通常是主线程)中同步执行,而非异步在新的线程中执行,也就失去了多线程的意义。

2.通过实现Runnadle接口创建线程

class MyRunnable implements Runnable {
    @Override
    public void run() {
        while (true) {
            System.out.println();
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
public class Demo2 {
    public static void main(String[] args)  {
        Thread t = new Thread(new MyThread());
        t.start();
        while (true) {
            System.out.println("hello main");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

定义Runnable接口的实现类:MyRunnable类实现了java.lang.Runnable接口,并在其内部重写了run()方法。这个run()方法同样是线程需要执行的任务主体,这里也是一个无限循环,每秒输出一行空白字符,并在每次循环间使当前线程睡眠1秒。

输出结果和第一种方法同样主线程和Thread线程在交替打印各自的信息,同样不是严格按照交替顺序进行的,而是随机交错

3.针对方法1使用匿名内部类

public class Demo3 {
    public static void main(String[] args) {
        Thread t = new Thread() {
            @Override
            public void run() {
                while (true) {
                    System.out.println("hello thread");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        };
 
        t.start();
        while (true) {
            System.out.println("hello main");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
 
}

就是针对方法1的一个变种方法,结果都一样,都是实现了多线程

4.针对方法2使用匿名内部类

public class Demo4 {
    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    System.out.println("hello thread");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        });
        t.start();
        while (true) {
            System.out.println("hello main");
            Thread.sleep(1000);
        }
 
    }
}

就是针对方法2的一个变种方法,结果都一样,都是实现了多线程

5.使用 lambda表达式

public class Demo5 {
    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(()->{
           while (true) {
               System.out.println("hello Thread");
               try {
                   Thread.sleep(1000);
               } catch (InterruptedException e) {
                   throw new RuntimeException(e);
               }
           }
        });
        t.start();
        while (true) {
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }
}

Lambda表达式是Java 8及其后续版本引入的一种简洁的函数式编程风格的特性,用于创建匿名函数或闭包。Lambda表达式使得开发者能够更加方便地处理函数式接口(即只有一个抽象方法的接口),无需显式声明新的类来实现这些接口。

基本结构: Lambda表达式的通用格式如下:

Java

1(parameters) -> {body}
  • 参数部分(parameters):可以为空,也可以包含零个或多个参数。每个参数都有一个类型或可以通过类型推断得出。如果只有一个参数,可以省略小括号;如果有多个参数,则需用逗号分隔。
  • 箭头符号(->):标志着参数列表的结束和函数体的开始。
  • 函数体(body):可以是一个表达式或一个代码块。如果函数体只包含一条表达式且能隐式转换为方法的返回类型,可以省略大括号;否则,需要使用大括号包围多条语句形成代码块。

结果如下:

这5种方法博主这里建议大家不说你都会写,(当然这5种方法多写几遍,应该差不多就可以掌握了)但你至少都得看的懂,并且熟练掌握几种.其实这5种方法本质上就是

1.把线程具体要实现的业务写出来即(重写run()方法)

2.通过调用start()方法创建/启动线程.

三.Thread类

1.构造方法

Thread(String name)这里可以为你创建的线程,命名以便在后续这个名字对于理解和追踪多线程应用程序中的各个线程很有帮助,尤其是在调试和日志记录的过程中。

public class Demo23 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()-> {
            while (true) {
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                System.out.println("hello Thread");
            }
        },"text1");
        t1.start();
        while (true) {
            Thread.sleep(1000);
            System.out.println("hello main");
        }
    }
}

通过命令行输入jconsole打开Java的监视控制台就可以查看到,此时我们创建的线程的状态,非常便于识别和管理各个线程特别是在后期我们创建很多线程时,优势就体现出来了

2.其他的方法

1.获取Id getid()

public class Demo23 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()-> {
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                System.out.println("hello thread");
 
        },"text1");
        t1.start();
        System.out.println(t1.getId());//获取线程的ID
        Thread.sleep(1000);
        System.out.println("hello main");
 
    }
}

结果为:

 2.获取名称 getName()

3.获取状态 getState()

4.是否为后台线程 isDaemon()

这里我们介绍一下后台线程和前台线程的区别

  1. 前台线程(Foreground Thread)
  • 常指那些负责处理用户交互、执行主业务逻辑或执行关键服务的线程,这类线程通常决定了整个应用程序的生命周期。例如,GUI应用程序的事件循环线程就是一个典型的前台线程,只要这个线程还在运行,应用程序就不会退出。另外,Web服务器中的请求处理线程也是前台线程,它们必须保持活跃以响应客户端请求。
  1. 后台线程(Background Thread)
  • 也称为守护线程(Daemon Thread)。在Java中,通过调用 Thread.setDaemon(true) 方法可以将一个线程设置为守护线程。守护线程主要用于执行支持性的任务,比如清理工作、资源监控、定时任务等。当所有的非守护线程(也就是所谓的前台线程)都结束后,即使还有守护线程在运行,Java虚拟机也会退出。也就是说,守护线程依赖于非守护线程的存在,非守护线程全部结束后,守护线程也随之结束,不再单独维持应用程序的运行。

总结来说,前台线程与应用程序的主要功能和生命周期紧密相关,而后台线程则更多是服务于前台线程,不直接影响应用程序的关闭与否

为了通俗易懂点博主这里举个小例子

  1. 前台线程(重要主线任务)
  • 假设你正在厨房做饭(这是你的主要任务,类似于前台线程),你正在炒菜(主业务流程),这个过程中你需要不断地翻炒、调味等操作(前台线程的工作)。如果不做这些,饭就无法完成,这就是至关重要的前台任务。
  1. 后台线程(辅助支持任务)
  • 同时,厨房里的洗碗机正在运行清洗餐具(这是一个后台守护任务,类似于后台线程)。虽然洗碗很重要,但如果炒菜任务完成了(所有非守护线程结束),你可以离开厨房,即使洗碗机还没洗完(后台线程仍在运行),你也不会留在那里等待它结束。洗碗机就是在后台默默地支持你的主要烹饪任务。

通过这个例子,可以看出前台线程(炒菜)主导着整个活动的进程和结束,而后台线程(洗碗机)虽然重要,但它的运行并不影响整个活动(做饭)的基本结束条件。在计算机程序中,后台线程往往用来处理一些辅助性、长期运行或维护性的工作,而不直接影响程序的主流程和退出。

再通过具体的代码例子来说明后台线程和前台线程

这个属于两个都是前台线程的情况

public class Demo25 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()->{
           for(int i = 0; i <5;i++) {
               try {
                   Thread.sleep(1000);
               } catch (InterruptedException e) {
                   throw new RuntimeException(e);
               }
               System.out.println("hello Thread");
           }
            System.out.println("t1结束");
        });
        t1.start();
        for (int i = 0;i < 3;i++) {
            Thread.sleep(1000);
            System.out.println("hello main");
        }
        System.out.println("main结束");
    }
}

可以看到,main线程以及结束了,可是t1线程没有结束,此时进程就没有结束,直到t1结束后,进程才结束,这就是只有所有前台线程结束后,进程才结束

我们现在把t1线程设置为后台线程再看看结果如何

这里要注意在创建线程之前(在start()方法之前),就要先设置好,否则此操作无效

public class Demo25 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()->{
           for(int i = 0; i <5;i++) {
               try {
                   Thread.sleep(1000);
               } catch (InterruptedException e) {
                   throw new RuntimeException(e);
               }
               System.out.println("hello Thread");
           }
            System.out.println("t1结束");
        });
        t1.setDaemon(true);//要在线程创建前,将线程设置为后台线程
        t1.start();
        for (int i = 0;i < 3;i++) {
            Thread.sleep(1000);
            System.out.println("hello main");
        }
        System.out.println("main结束");
    }
}

此时,main线程为前台线程,t1为后台线程,我们可以看到只要main线程结束了,无论t1是否结束,进程都结束了.以及如果t1结束,但前台线程main线程没有结束,进程也不会结束

public class Demo25 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()->{
           for(int i = 0; i <3;i++) {
               try {
                   Thread.sleep(1000);
               } catch (InterruptedException e) {
                   throw new RuntimeException(e);
               }
               System.out.println("hello Thread");
           }
            System.out.println("t1结束");
        });
        t1.setDaemon(true);//要在线程创建前,将线程设置为后台线程
        t1.start();
        for (int i = 0;i < 5;i++) {
            Thread.sleep(1000);
            System.out.println("hello main");
        }
        System.out.println("main结束");
    }
}

5.是否存活 isAlive()

它用于检查一个线程是否仍然存活,即线程是否已经开始执行且还没有结束。

6.是否被中断 isInterrupted()

isInterrupted() 是Java中 Thread 类的一个方法,用于检查线程是否已被中断。中断是一种协作机制,允许一个线程通知另一个线程应该停止当前正在执行的操作。

语法如下:

Java

1boolean isInterrupted()

这个方法会返回一个布尔值:

  • 如果线程已被中断(调用了 interrupt() 方法),则返回 true
  • 如果线程未被中断,则返回 false

需要注意的是,调用 isInterrupted() 方法并不会清除中断状态。如果希望在检查中断状态的同时清除此状态,可以使用静态方法 Thread.interrupted()

举例说明:

Java

1Thread thread = new Thread(() -> {
2    while (!Thread.currentThread().isInterrupted()) {
3        // 执行任务...
4    }
5});
6
7thread.start();
8
9// 在某个时刻,决定中断线程
10thread.interrupt();
11
12// 另一线程或同一线程中的代码可以通过isInterrupted()检查中断状态
13if (thread.isInterrupted()) {
14    System.out.println("线程已被中断");
15}

在上面的代码中,线程在执行任务时会周期性检查自身的中断状态,一旦发现被中断,就会退出循环,从而达到协作式中断的目的

同时需要注意的是,只能起到提醒作用,并不可以真正的中断,这能该线程自己中断自己

例如,线程可能在循环、等待IO操作、或者在调用 sleep()wait()join() 等方法时被中断。在这些情况下,被中断的线程通常会抛出一个 InterruptedException 异常,然后可以根据程序的需求选择恢复执行、清理资源后退出,或者直接结束线程。

总的来说,interrupt() 方法和 isInterrupted() 方法提供了线程间的通信手段,允许一个线程请求另一个线程停止其当前的操作,但具体的中断逻辑需要由被中断线程自行实现,体现了Java中线程中断的协作性和灵活性。

以上就是博主关于Java多线程学习的一点点部分,后续还有很多内容,例如重点的:线程安全问题

这里就不过多的赘述了,如果感兴趣的话,可以关注博主.查看线程安全的解释



相关文章
|
7天前
|
安全 Java API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
37 6
|
20天前
|
存储 监控 小程序
Java中的线程池优化实践####
本文深入探讨了Java中线程池的工作原理,分析了常见的线程池类型及其适用场景,并通过实际案例展示了如何根据应用需求进行线程池的优化配置。文章首先介绍了线程池的基本概念和核心参数,随后详细阐述了几种常见的线程池实现(如FixedThreadPool、CachedThreadPool、ScheduledThreadPool等)的特点及使用场景。接着,通过一个电商系统订单处理的实际案例,分析了线程池参数设置不当导致的性能问题,并提出了相应的优化策略。最终,总结了线程池优化的最佳实践,旨在帮助开发者更好地利用Java线程池提升应用性能和稳定性。 ####
|
16天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
16天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
40 3
|
17天前
|
监控 Java 开发者
深入理解Java中的线程池实现原理及其性能优化####
本文旨在揭示Java中线程池的核心工作机制,通过剖析其背后的设计思想与实现细节,为读者提供一份详尽的线程池性能优化指南。不同于传统的技术教程,本文将采用一种互动式探索的方式,带领大家从理论到实践,逐步揭开线程池高效管理线程资源的奥秘。无论你是Java并发编程的初学者,还是寻求性能调优技巧的资深开发者,都能在本文中找到有价值的内容。 ####
|
22天前
|
缓存 Java 开发者
Java多线程并发编程:同步机制与实践应用
本文深入探讨Java多线程中的同步机制,分析了多线程并发带来的数据不一致等问题,详细介绍了`synchronized`关键字、`ReentrantLock`显式锁及`ReentrantReadWriteLock`读写锁的应用,结合代码示例展示了如何有效解决竞态条件,提升程序性能与稳定性。
81 6
|
20天前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
28 2
|
20天前
|
监控 Java 开发者
Java线程管理:守护线程与本地线程的深入剖析
在Java编程语言中,线程是程序执行的最小单元,它们可以并行执行以提高程序的效率和响应性。Java提供了两种特殊的线程类型:守护线程和本地线程。本文将深入探讨这两种线程的区别,并探讨它们在实际开发中的应用。
27 1
|
22天前
|
安全 Java 开发者
Java中的多线程编程:从基础到实践
本文深入探讨了Java多线程编程的核心概念和实践技巧,旨在帮助读者理解多线程的工作原理,掌握线程的创建、管理和同步机制。通过具体示例和最佳实践,本文展示了如何在Java应用中有效地利用多线程技术,提高程序性能和响应速度。
54 1
|
7月前
|
存储 安全 Java
深入理解Java并发编程:线程安全与锁机制
【5月更文挑战第31天】在Java并发编程中,线程安全和锁机制是两个核心概念。本文将深入探讨这两个概念,包括它们的定义、实现方式以及在实际开发中的应用。通过对线程安全和锁机制的深入理解,可以帮助我们更好地解决并发编程中的问题,提高程序的性能和稳定性。
下一篇
DataWorks