Java多线程并发

  • 前言
  • 一、Java并发知识库
  • 二、Java线程实现方式
  • 2.1. 继承Thread类
  • 2.2. 实现Runnable接口
  • 2.3. 有返回值线程
  • 2.4. 基于线程池方式
  • 三、线程池
  • 3.1. newCachedThreadPool
  • 3.2. newFixedThreadPool
  • 3.3. newScheduledThreadPool
  • 3.4. newSingleThreadExecutor
  • 四、线程生命周期
  • 4.1. 新建状态(NEW)
  • 4.2. 就绪状态(RUNNABLE)
  • 4.3. 运行状态(RUNNING)
  • 4.4. 阻塞状态(BLOCKED)
  • 4.5. 死亡状态
  • 五、线程终止的方式
  • 5.1. 正常运行结束
  • 5.2. 使用退出标志退出线程
  • 5.3. Interrupt方法结束线程
  • 5.4. stop方法终止线程
  • 六、sleep()和wait()区别
  • 七、run()和start()区别
  • 八、Java后台线程
  • 总结

前言

本篇记录学习Java多线程的知识,主要是线程的实现和使用方式。

一、Java并发知识库

java 多线程并行 性能 java 多线程 并发_Java

二、Java线程实现方式

2.1. 继承Thread类

Thread本质是实现了Runnable接口的一个实例,代表一个线程的实例。启动线程的唯一方法就是调用start()方法。start()方法是一个native方法,它将启动一个新的线程,并执行run方法。

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

@Test
public void testThread() {
    Thread thread = new MyThread();
    thread.start();
}

2.2. 实现Runnable接口

如果一个类已经实现了其他类,无法再继承Thead类,可以实现Runnable接口。

public class MyRunnable implements Runnable {

    public void run() {
        System.out.println("start MyRunnable");
    }
}

@Test
public void testRunnable() {
    MyRunnable myRunnable = new MyRunnable();
    Thread thread = new Thread(myRunnable);
    thread.start();
}

2.3. 有返回值线程

有返回值的任务必须实现Callable接口,类似的,无返回值的任务必须实现Runnable接口。执行Callable任务后,可以获取一个Future的对象,在该对象上调用get就可以获取到Callable任务返回的Object了,再结合线程池接口ExecutorService就可以实现有返回值的多线程了。

public class MyCallable implements Callable {

    public Object call() throws Exception {
        return "start MyCallable";
    }
}

@Test
public void testCallable() throws ExecutionException, InterruptedException {
    ExecutorService es = Executors.newFixedThreadPool(10);
    List<Future> list = new ArrayList<Future>();

    for (int i = 0; i < 10; i++) {
        MyCallable callable = new MyCallable();
        Future future = es.submit(callable);
        list.add(future);
    }

    for (Future future : list) {
        System.out.println("res: " + future.get().toString());
    }
}

2.4. 基于线程池方式

线程的创建和销毁是非常耗费资源的,为了避免不必要的浪费,可以采用缓存策略,就是线程池。

三、线程池

Java里面线程池的顶级接口是Executor,但严格意义上Executor不是一个线程池,而是一个执行线程的工具。真正的线程池接口是ExecutorService。

3.1. newCachedThreadPool

创建一个可根据需要创建新线程的线程池,但是在以前构造的线程可用时将重用它们。对于执行很多短期异步任务的程序而言,这些线程通常可提高程序性能。调用execute将重用以前构造的线程(如果程序可用)。如果现有线程没有可用的,则创建一个新线程并添加到池中。终止并从缓存中移除那些已有60秒未被使用的线程。因此,长时间保持空闲的线程池不会使用任何资源。

3.2. newFixedThreadPool

创建一个可重用固定线程数的线程池,以共享的无界队列方式来运行这些线程。在任意点,在大多少nThread线程会处于处理任务的活动状态。如果在所有线程处于活跃状态时提交附加任务,则在有可用线程之前,附加任务将在队列中等待。如果在关闭前的执行期间由于失败而导致任何线程终止,那么一个新线程将代替它执行后续的任务。在某个线程被显式地关闭之前,池中的线程将一直存在。

3.3. newScheduledThreadPool

创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。

3.4. newSingleThreadExecutor

Executors.newSingleThreadExecutor()返回一个线程池(这个线程池只有一个),这个线程池可以在线程死后(或发生异常时)重新启动一个线程来替代原来的线程继续执行下去。

四、线程生命周期

当线程被创建并启动以后,它即不是一启动就进入了执行状态,也不是一直处于执行状态。在线程的生命周期中,它要经过新建、就绪、运行、阻塞和死亡5种状态。尤其是当线程启动以后,它不可能一直“霸占”着CPU独自运行,所以CPU需要在多条线程之间切换,于是线程状态也会多次在运行、阻塞之间切换。

4.1. 新建状态(NEW)

当程序使用new关键字创建了一个线程之后,该线程就处于新建状态,此时仅由JVM为其分配内存,并初始化其成员变量的值

4.2. 就绪状态(RUNNABLE)

当线程对象调用了start()方法之后,该线程处于就绪状态。Java虚拟机会为其创建方法调用栈和程序计数器,等待调度运行。

4.3. 运行状态(RUNNING)

如果处于就绪状态的线程获得了CPU,开始执行run()方法的线程执行体,则该线程处于运行状态。

4.4. 阻塞状态(BLOCKED)

阻塞状态是指线程因为某种原因放弃了cpu的使用权,也即让出了cpu timeslice,暂时停止运行。直到线程进入可运行状态,才有机会再次获得cpu timeslice转到运行状态。阻塞的状态分三种:

  • 等待阻塞:
    运行的线程执行obj.wait()方法,JVM会把该线程放入等待队列中。
  • 同步阻塞:
    运行的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池中。
  • 其他阻塞:
    运行的线程执行Thread.sleep(long ms)或t.join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入可运行状态。

4.5. 死亡状态

线程会以下面三种方式结束,结束后就是死亡状态。
正常结束
run()或call()方法执行完成,线程正常结束。
异常结束
线程抛出一个未捕获的Exception或Error。
调用stop
直接调用该线程的stop()方法来结束该线程(该方法通常容易造成死锁,不推荐使用)。

五、线程终止的方式

5.1. 正常运行结束

程序运行结束,线程自动结束。

5.2. 使用退出标志退出线程

一般run()方法执行完,线程就会正常结束,然而,常常有些线程是伺服线程。它们需要长时间的运行,只有在外部某些条件满足时,才能关闭这些线程。使用一个变量来控制循环。
示例代码:

public class ThreadSafe extend Thread {
    public volatile boolean exit = false;
    public void run() {
		while(!exit) {
			// do something
		}
	}
}

5.3. Interrupt方法结束线程

使用Interrupt()方法来中断线程有两种情况:

  • 线程处于阻塞状态:
    如使用了sleep,同步锁的wait,socket中的receiver,accept等方法时,会使线程处于阻塞状态。当调用线程的interrupt()方法时,会抛出InterruptException异常。阻塞中的那个方法抛出这个异常,通过代码捕获该异常,然后break跳出循环状态,从而让我们有机会结束这个线程的执行。通常很多人认为只要调用interrupt方法线程就会结束,实际上是错的,一定要捕获InterruptedException异常之后通过break跳出循环,才能正常结束run方法。
  • 线程未处于阻塞状态:
    使用isInterrupted()判断线程的中断标志来退出循环。当使用interrupt()方法时,中断标志就会置true,和使用自定义的标志来控制循环是一样的道理。

实例代码:

public class ThreadSafe extends Thread {
	public void run() {
		while(!isInterrupted()) {
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
				break;
			}
		}
	}
}

5.4. stop方法终止线程

程序中可以直接使用Thread.stop()来终止线程,但是stop是危险的,就像突然关闭计算机电源,而不是按正常程序关机一样,可能会产生不可预料的结果,不安全的原因是:thread.stop()调用以后,创建子线程的线程就会抛出ThreadDeatherror的错误,并会释放子线程锁持有的所有锁。一般任何进行加锁的代码块,都是为了保护数据的一致性,如果在调用thread.stop()导致了该线程所持有的所有锁的突然释放,那么被保护数据就有可能呈现不一致性,其他线程在使用这些被破坏的数据时,有可能导致一些很奇怪的应用程序错误。因此,并不推荐使用stop方法来终止线程。

六、sleep()和wait()区别

  1. 对于sleep()方法是属于Thread类中的,wait()方法则是属于Object类中。
  2. sleep()方法导致程序暂停执行指定的时间,让出cpu给其他线程,但是它的监控状态依然保持着,当指定的时间到了又会自动恢复运行状态。
  3. 在调用sleep()方法的过程中,线程不会释放对象锁。
  4. 而当调用wait()方法的时候,线程会放弃对象锁,进入等待此对象的等待锁定池,只有针对此对象调用notify()方法后本线程才进入对象锁定池准备获取对象锁进入运行状态。

七、run()和start()区别

  1. start()方法来启动线程,真正实现了多线程运行。这时无需等待run方法体代码体执行完毕,可以直接继续执行下面的代码。
  2. 通过调用Thread类的start()方法来启动一个线程,这时此线程是处于就绪状态,并没有运行。
  3. 方法run()称为线程体,它包含了要执行的这个线程的内容,线程就进入了运行状态,开始运行run函数当中的代码。Run方法运行结束,此线程终止。然后CPU再调度其他线程。

八、Java后台线程

  1. 定义:守护线程–“服务线程”,它是后台线程,他有一个特性,即为用户线程提供公共服务,在没有用户线程可服务时会自动离开。
  2. 优先级:守护线程的优先级比较低,用于为系统中的其他对象和线程提供服务。
  3. 设置:通过setDaemon(true)来设置线程为“守护线程”,将一个用户线程设置为守护线程的方式是在线程对象创建之前用线程对象的setDaemon方法。
  4. 在Daemon线程中产生的线程也是Daemon线程。
  5. 线程则是JVM级别的,以Tomcat为例,如果你在Web应用中启动一个线程,这个线程的生命周期并不会和Web应用程序保持同步。也就是说,即使你停止了Web应用,这个线程依然活跃。
  6. example:垃圾回收线程就是一个经典的守护线程,当我们的程序中不再有任何运行的Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于实时监控和管理系统中的可回收资源。
  7. 生命周期:守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或者等待处理某些发生的事件。也就是说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。当JVM中所有的线程都是守护线程的时候,JVM就可以退出了,如果还有一个或以上的非守护线程则JVM不会退出。

总结

提示:这里对文章进行总结:
例如:以上就是今天要讲的内容,本文仅仅简单介绍了pandas的使用,而pandas提供了大量能使我们快速便捷地处理数据的函数和方法。