首页 > 编程语言 >夯实Java基础,一篇文章全解析线程问题

夯实Java基础,一篇文章全解析线程问题

时间:2022-10-07 10:00:18浏览次数:52  
标签:Java Thread myThread 夯实 MyThread 线程 new public

1\. 线程是什么

操作系统支持多个应用程序并发执行,每个应用程序至少对应一个进程 ,彼此之间的操作和数据不受干扰,彼此通信一般采用管道通信、消息队列、共享内存等方式。当一个进程需要磁盘IO的时候,CPU就切换到另外的进程,提高了CPU利用率。

有了进程,为什么还要线程?因为进程的成本太高了。

启动新的进程必须分配独立的内存空间,建立数据表维护它的代码段、堆栈段和数据段,这是昂贵的多任务工作方式。线程可以看作轻量化的进程。线程之间使用相同的地址空间,切换线程的时间远小于切换进程的时间。

进程是资源分配的最小单位,而线程是CPU调度的最小单位。每一个进程中至少有一个线程,同一进程的所有线程共享该进程的所有资源,多个线程可以完成多个不同的任务,也就是我们常说的并发多线程。

2\. 怎样创建线程

创建线程常用的有四种方式,分别是:

  1. 继承Thread类
  2. 实现Runnable接口
  3. 实现Callable接口
  4. 使用线程池创建

分别看一下怎么具体怎么使用代码创建的?

2.1 继承Thread类

public class ThreadDemo {

public static void main(String[] args) {
Thread thread = new MyThread();
thread.start(); // 启动线程
}
}

class MyThread extends Thread {
@Override
public void run() {
System.out.println("hello,word");
}
}

输出结果:

hello,word

start方法用来启动线程,只能被调用一次。

run方法是线程的核心方法,业务逻辑都写在run方法中。

2.2 实现Runnable接口

public class ThreadDemo {

public static void main(String[] args) {
MyRunnable myRunnable = new MyRunnable();
Thread thread1 = new Thread(myRunnable, "线程1");
Thread thread2 = new Thread(myRunnable, "线程2");
thread1.start(); // 启动线程1
thread2.start(); // 启动线程2
}
}

class MyRunnable implements Runnable {
private int count = 5;

@Override
public void run() {
while (count > 0) {
System.out.println(Thread.currentThread().getName()
+ ",hello,word," + count--);
}
}
}

输出结果:

线程2,hello,word,4
线程1,hello,word,5
线程1,hello,word,2
线程1,hello,word,1
线程2,hello,word,3

需要把Runnable实例放到Thread类中,才能执行,Thread对象才是真正的线程对象。

使用实现Runnable接口创建线程方式,相比继承Thread类创建线程,优点是:

  1. 实现的方式没有类的单继承性的局限性
  2. 实现的方式更适合来处理多个线程有共享数据的情况

2.3 实现Callable接口

public class ThreadTest {
public static void main(String[] args) throws ExecutionException, InterruptedException {
MyCallable myCallable = new MyCallable();
FutureTask<String> futureTask = new FutureTask<String>(myCallable);
Thread thread = new Thread(futureTask);
thread.start();
System.out.println(futureTask.get());
}
}

class MyCallable implements Callable {
@Override
public String call() throws Exception {
return "hello,word";
}
}

输出结果:

hello,word

实现Callable接口的线程实例对象,配合FutureTask使用,可以接收返回值。

2.4 使用线程池创建

public class ThreadDemo {

public static void main(String[] args) {
ExecutorService executorService = Executors.newFixedThreadPool(10);
executorService.execute(() -> System.out.println("hello,word"));
}
}

输出结果:

hello,word

使用线程池创建线程是工作开发中最常用的方式,优点是:

  1. 线程池帮忙管理对象的创建与销毁,减轻开发者工作量
  2. 线程池帮忙管理任务的调用,资源的创建与分配
  3. 复用线程和对象,提高使用效率

3\. 线程的状态

线程共有6种状态,分别是NEW(初始化)、RUNNABLE(可运行)、WAITING(等待)、TIMED_WAITING(超时等待)、BLOCKED(阻塞)、TERMINATED(终止)。

  • NEW(初始化)
    表示创建线程对象之后,还没有调用start方法。
  • RUNNABLE(可运行)
    表示调用start方法之后,等待CPU调度。为了便于理解,通常又把RUNNABLE分别RUNNING(运行中)和READY(就绪)。处在RUNNING(运行中)状态的线程可以调用yield方法,让出CPU时间片,然后跟其他处于READY(就绪)一起等待被调度。
  • WAITING(等待)
    处于RUNNABLE状态的线程调用wait方法之后,就处于等待状态,需要其他线程显示地唤醒。
  • TIMED_WAITING(超时等待)
    处于RUNNABLE状态的线程调用wait(long)方法之后,就处于等待状态,需要其他线程显示地唤醒。
  • BLOCKED(阻塞)
    等待进入synchronized方法/代码块,处于阻塞状态。
  • TERMINATED(终止)
    表示线程已经执行结束。

夯实Java基础,一篇文章全解析线程问题_线程池

4\. 线程常用方法

说一下线程有哪些常用的方法。

方法定义

含义

使用方式

public synchronized void start() {……}

启动线程

MyThread myThread = new MyThread();

myThread.start();



public static native Thread currentThread();

获取当前线程实例对象

Thread thread = Thread.currentThread();

public static native void yield();

让出CPU时间片

Thread.yield();

public static native void sleep(long millis);

睡眠指定时间

Thread.sleep(1L);

public void interrupt() {……}

中断线程

MyThread myThread = new MyThread();

myThread.interrupt();



public static boolean interrupted() {……}

判断线程是否已中断

MyThread myThread = new MyThread();

boolean interrupted = myThread.isInterrupted();



public final native boolean isAlive();

判断线程是否是存活状态

MyThread myThread = new MyThread();

boolean alive = myThread.isAlive();



public final String getName() {……}

获取线程名称

MyThread myThread = new MyThread();

String name = myThread.getName();



public State getState() {……}

获取线程状态

MyThread myThread = new MyThread();

Thread.State state = myThread.getState();



public long getId() {……}

获取线程ID

MyThread myThread = new MyThread();

long id = myThread.getId();



public final void join() {……}

等待其他线程执行完再执行

MyThread myThread = new MyThread();

myThread.join();



标签:Java,Thread,myThread,夯实,MyThread,线程,new,public
From: https://blog.51cto.com/u_15773567/5734233

相关文章

  • Java方法
    方法方法的定义方法包含一个方法头和一个方法体,下面是一个方法的所有部分:修饰符:修饰符,可选,告诉编译器如何调用该方法。定义了该方法的访问类型。返回值类型∶方法......
  • Java 输出为乱码的解决方案之一
    最近在学习Java的过程中,写了一个查询天气的小程序,发现输出为乱码,尝试了几种方法,发现一个奇怪又简单的方法解决了这个问题,只需要编码一个字段即可,其它字段自动重新编码,理论......
  • 虚拟线程 - VirtualThread源码透视
    前提JDK19于2022-09-20发布GA版本,该版本提供了虚拟线程的预览功能。下载JDK19之后翻看了一下有关虚拟线程的一些源码,跟早些时候的Loom项目构建版本基本并没有很大出入,也跟......
  • 我整理了50道经典Java算法题,直接进了字节跳动!!
    写在前面最近,很多小伙伴都想进入字节跳动这个快速发展的公司,而字节跳动对于算法的要求比较高。于是乎,有些小伙伴问我能否整理一些基础的算法题,帮助他们提升下基础算法能......
  • 2022.10.7Java方法详解
    Java方法详解System,out,println()是输出语句,也是方法Java方法是语句的集合,它们在一起执行一个功能方法是解决一类问题步骤的有序组合方法是包含类或对象中......
  • java类加载器 defineClass和loadClass的区别
    在阅读《自己动手写java虚拟机》时,通过观察P133~P135的代码classLoader会将加载过的class保存起来(包括defineclass的类),而只有在要加载一个类时 没有从已加载的类找到......
  • Java 面试题 10 - 海量数据处理算法
    大数据处理中的分治思想哈希映射:如果数据太大,不能全部放入内存中,就可以利用映射函数将每条数据映射到一个小文件中,例如%1000可以将大文件映射成1000个小文件。相同的......
  • Java 面试题 11 - 分布式系统常见问题
    分布式ID的实现分布式ID需要满足哪些需求?基本需求:全局唯一高性能:生成速度快,对本地资源消耗小。高可用:生成分布式ID的服务要保证高可用性。方便易用:使用方便......
  • Java 面试题 09 - 计算机网络
    TCP&UDPTCP和UDP的区别有什么?TCP面向连接,UDP无连接。TCP提供可靠的传输,在传递数据之前,需要通过三次握手建立连接,在传递数据时,有确认、窗口、重传、拥塞机......
  • Java 面试题 08 - 计算机网络
    进程什么是系统调用?根据进程访问资源的特点,可以把进程的运行状态分为两个级别:用户态:只能读取用户程序的数据;内核态:可以访问几乎一切资源。用户程序基本都运行在用户......