一、使用方式。
join是Thread类的一个方法,启动线程后直接调用,例如:
1
|
Thread t = new AThread(); t.start(); t.join();
|
二、为什么要用join()方法
在很多情况下,主线程生成并起动了子线程,如果子线程里要进行大量的耗时的运算,主线程往往将于子线程之前结束,但是如果主线程处理完其他的事务后,需要用到子线程的处理结果,也就是主线程需要等待子线程执行完成之后再结束,这个时候就要用到join()方法了。
三、join方法的作用
在JDk的API里对于join()方法是:
join
public final void join() throws InterruptedException Waits for this thread to die. Throws: InterruptedException - if any thread has interrupted the current thread. The interrupted status of the current thread is cleared when this exception is thrown.
即join()的作用是:“等待该线程终止”,这里需要理解的就是该线程是指的主线程等待子线程的终止。也就是在子线程调用了join()方法后面的代码,只有等到子线程结束了才能执行。
四、用实例来理解
写一个简单的例子来看一下join()的用法:
1.AThread 类
-
BThread类
-
TestDemo 类
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152class
BThread
extends
Thread {
public
BThread() {
super
(
"[BThread] Thread"
);
};
public
void
run() {
String threadName = Thread.currentThread().getName();
System.out.println(threadName +
" start."
);
try
{
for
(
int
i =
0
; i <
5
; i++) {
System.out.println(threadName +
" loop at "
+ i);
Thread.sleep(
1000
);
}
System.out.println(threadName +
" end."
);
}
catch
(Exception e) {
System.out.println(
"Exception from "
+ threadName +
".run"
);
}
}
}
class
AThread
extends
Thread {
BThread bt;
public
AThread(BThread bt) {
super
(
"[AThread] Thread"
);
this
.bt = bt;
}
public
void
run() {
String threadName = Thread.currentThread().getName();
System.out.println(threadName +
" start."
);
try
{
bt.join();
System.out.println(threadName +
" end."
);
}
catch
(Exception e) {
System.out.println(
"Exception from "
+ threadName +
".run"
);
}
}
}
public
class
TestDemo {
public
static
void
main(String[] args) {
String threadName = Thread.currentThread().getName();
System.out.println(threadName +
" start."
);
BThread bt =
new
BThread();
AThread at =
new
AThread(bt);
try
{
bt.start();
Thread.sleep(
2000
);
at.start();
at.join();
}
catch
(Exception e) {
System.out.println(
"Exception from main"
);
}
System.out.println(threadName +
" end!"
);
}
}
打印结果:
1234567891011main start.
//
主线程起动,因为调用了at.
join
(),要等到at结束了,此线程才能向下执行。
[BThread] Thread start.
[BThread] Thread loop at 0
[BThread] Thread loop at 1
[AThread] Thread start.
//
线程at启动,因为调用bt.
join
(),等到bt结束了才向下执行。
[BThread] Thread loop at 2
[BThread] Thread loop at 3
[BThread] Thread loop at 4
[BThread] Thread end.
[AThread] Thread end.
//
线程AThread在bt.
join
();阻塞处起动,向下继续执行的结果
main end!
//
线程AThread结束,此线程在at.
join
();阻塞处起动,向下继续执行的结果。
1234567891011121314151617public
class
TestDemo {
public
static
void
main(String[] args) {
String threadName = Thread.currentThread().getName();
System.out.println(threadName +
" start."
);
BThread bt =
new
BThread();
AThread at =
new
AThread(bt);
try
{
bt.start();
Thread.sleep(
2000
);
at.start();
//at.join(); //在此处注释掉对join()的调用
}
catch
(Exception e) {
System.out.println(
"Exception from main"
);
}
System.out.println(threadName +
" end!"
);
}
}
打印结果:
123456789101112main start.
//
主线程起动,因为Thread.
sleep
(2000),主线程没有马上结束;
[BThread] Thread start.
//
线程BThread起动
[BThread] Thread loop at 0
[BThread] Thread loop at 1
main end!
//
在
sleep
两秒后主线程结束,AThread执行的bt.
join
();并不会影响到主线程。
[AThread] Thread start.
//
线程at起动,因为调用了bt.
join
(),等到bt结束了,此线程才向下执行。
[BThread] Thread loop at 2
[BThread] Thread loop at 3
[BThread] Thread loop at 4
[BThread] Thread end.
//
线程BThread结束了
[AThread] Thread end.
//
线程AThread在bt.
join
();阻塞处起动,向下继续执行的结果
五、从源码看join()方法
在AThread的run方法里,执行了bt.join();,进入看一下它的JDK源码:
123public
final
void
join()
throws
InterruptedException {
join(0L);
}
123456789101112131415161718192021public
final
synchronized
void
join(
long
l)
throws
InterruptedException
{
long
l1 = System.currentTimeMillis();
long
l2 = 0L;
if
(l < 0L)
throw
new
IllegalArgumentException(
"timeout value is negative"
);
if
(l == 0L)
for
(; isAlive(); wait(0L));
else
do
{
if
(!isAlive())
break
;
long
l3 = l - l2;
if
(l3 <= 0L)
break
;
wait(l3);
l2 = System.currentTimeMillis() - l1;
}
while
(
true
);
}
单纯从代码上看: * 如果线程被生成了,但还未被起动,isAlive()将返回false,调用它的join()方法是没有作用的。将直接继续向下执行。 * 在AThread类中的run方法中,bt.join()是判断bt的active状态,如果bt的isActive()方法返回false,在bt.join(),这一点就不用阻塞了,可以继续向下进行了。从源码里看,wait方法中有参数,也就是不用唤醒谁,只是不再执行wait,向下继续执行而已。 * 在join()方法中,对于isAlive()和wait()方法的作用对象是个比较让人困惑的问题:
isAlive()方法的签名是:public final native boolean isAlive(),也就是说isAlive()是判断当前线程的状态,也就是bt的状态。
wait()方法在jdk文档中的解释如下:
Causes the current thread to wait until another thread invokes the notify() method or the notifyAll() method for this object. In other words, this method behaves exactly as if it simply performs the call wait(0).
The current thread must own this object's monitor. The thread releases ownership of this monitor and waits until another thread notifies threads waiting on this object's monitor to wake up either through a call to the notify method or the notifyAll method. The thread then waits until it can re-obtain ownership of the monitor and resumes execution.
在这里,当前线程指的是at。
相关推荐
在Java编程语言中,`Thread.join()`方法是一个非常重要的同步工具,它允许一个线程(调用者)等待另一个线程(被调用者)执行完成。`Thread.join()`的使用能够有效地解决多线程环境中的顺序执行问题,确保线程间的...
`join()`方法定义在`java.lang.Thread`类中,其主要功能是使调用该方法的线程(这里称为调用线程)暂停执行,直到目标线程(即`join()`方法所在的线程)执行完毕。这样,可以确保调用线程在目标线程执行完成后才...
在"JAVA多线程编程详解-详细操作例子.doc"和"Java多线程编程详解.doc"文档中,你应该能找到关于以上知识点的具体示例和深入解释,包括如何创建线程、线程间的通信(如wait/notify机制、Semaphore、CountDownLatch)...
以上是对"Java多线程详解"主题的详细阐述,涵盖了Java多线程的基本概念、实现方式、线程控制、线程池、并发集合、线程间通信以及并发编程中常见的问题和解决方案。学习和熟练掌握这些内容对于开发高效的多线程Java...
总之,《Java多线程编程详解》这份PDF文档全面覆盖了Java多线程编程的核心内容,无论你是初学者还是有经验的开发者,都能从中受益。通过学习,你可以更好地理解和掌握如何在Java中有效地利用多线程,提升程序的性能...
Java多线程编程是Java语言中的一个重要特性,它允许开发者编写能够同时执行多个任务的程序,提高了应用程序的效率和响应性。多线程是通过创建和管理多个执行流来实现的,每个流被称为一个线程,它们共享同一进程的...
Java中的`join()`方法是多线程编程中的一个重要特性,用于控制线程间的协作和同步。在Java中,多线程允许程序同时执行多个任务,但有时我们需要确保某些线程按照特定顺序完成,或者主线程需要等待其他线程执行完毕后...
Java多线程编程是Java开发中的重要组成部分,它允许程序同时执行多个任务,极大地提高了程序的效率和响应性。在Java中,多线程的实现...通过阅读《Java多线程编程详解.pdf》这样的资料,可以系统学习和掌握这些知识。
除了基础的创建和启动线程外,Java的`Thread`类还提供了许多用于线程控制和同步的方法,比如`join()`、`interrupt()`、`yield()`等,以及用于线程安全的数据结构如`ConcurrentHashMap`等。此外,Java的并发工具包`...
### JAVA多线程编程详解 #### 一、深入理解多线程 多线程机制是现代编程语言中处理并发操作的关键技术之一,尤其在资源密集型应用和提高系统响应速度方面发挥着重要作用。在Java中,多线程的实现不仅简化了并发...
### Java线程与并行详解 #### 一、Java线程基础 在Java中,**线程**是一种轻量级的进程,它允许一个程序同时执行多个任务,从而提高程序的执行效率。Java从1.0版本开始就支持多线程编程,并在后续版本中不断完善。...
Java多线程编程详解 Java 多线程编程是Java平台的一大特性,它允许程序同时执行多个任务,提高系统的效率和响应性。在现代软件设计中,多线程技术是不可或缺的一部分,尤其对于处理并发操作和高并发场景的应用至关...
thread1.join(); thread2.join(); // 清理线程局部变量,防止内存泄漏 threadLocal.remove(); } } ``` 在这个例子中,`threadLocal`变量被两个线程分别设置为10和20,每个线程看到的都是自己设置的值,不会...
5. **TIMED_WAITING(超时等待)**:线程处于有限期等待状态,等待另一个线程执行依赖于指定等待时间的操作,例如调用`Object.wait(long timeout)`、`Thread.join(long timeout)`或`Thread.sleep(long millis)`等...
此外,Java还提供了`Thread.sleep()`, `Thread.join()`, `Thread.yield()`等方法来控制线程的执行顺序和休眠。`ExecutorService`和`Future`接口则提供了线程池和异步计算的能力,可以有效地管理和控制线程的生命周期...
【Java线程详解】 在Java编程中,线程(Thread)是程序执行的最小单元,它与进程一起构成了操作系统的基础。Java提供了丰富的线程管理功能,使得开发者可以方便地进行多任务并行处理。本篇将深入探讨Java线程的核心...
总的来说,Java多线程提供了强大的并发处理能力,通过继承Thread类或实现Runnable接口创建线程,使用同步机制确保线程安全,以及通过中断机制控制线程的生命周期。理解并熟练掌握这些概念对于编写高效、健壮的多线程...
### CoreJava核心知识点详解 #### 一、Java的特点与运行原理 **1.1 Java特点** - **简单性**:Java的设计目标之一是使语言尽可能简单直观。 - **面向对象**:Java是一种纯面向对象的语言,支持封装、继承、多态等...