- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试理解 join() 的代码流。
public class Multi extends Thread {
public void run() {
for (int i = 0; i < 5; i++) {
System.out.println(Thread.currentThread().getName());
}
}
public static void main(String[] args) {
Thread t1 = new Multi();
Thread t2 = new Multi();
Thread t3 = new Multi();
Thread t4 = new Multi();
t1.start();
try {
t1.join();
} catch (Exception e) {
}
t2.start();
t3.start();
try {
t3.join();
} catch (Exception e) {
}
t4.start();
System.out.println("........" + Thread.currentThread().getName());
t1.setName("A");
t2.setName("B");
t3.setName("C");
t4.setName("D");
}
}
正如我多次运行程序后观察到的那样,输出始终是 thread t1
首先执行,并且它将完成它的执行
,而无需任何上下文切换,并且每当 t3 将开始,它将完成它的执行
。我的理解清楚吗?
我观察到,如果不使用连接,main()
会在线程执行的任何地方执行,这意味着我看到 .......main
输出在我程序的输出之间,但在 join()
之后它总是在线程 t3 之后执行。这是我的疑问,因为 main()
在 join()
语法之前开始,所以它不应该遵循 t3/t1 线程完成?它有意义还是我遗漏了什么?
最佳答案
How does
join()
work in java?
我同意你 the javadoc for join()
有点不清楚,因为第一次阅读 this
指的是谁并不明显。
这意味着调用t.join()
的线程会阻塞,直到线程t
完成执行。如果当前线程调用t.join()
时t
已经结束,则当前线程不会停止,继续运行。文档中的单词 this
在这里指的是 t
,而不是调用该方法的线程。
Does it guarantee the execution before
main()
?[...] if no join is used,
main()
executes anywhere b/w the execution of threads [...]
您不应将 main()
视为一个整体。 main()
的一部分在其他线程之前执行,一部分并行执行,一部分在其他线程之后执行。这实际上是 start()
和 join()
控制的。下面我来解释一下。
main()
中发生了什么这是关于 t1.start()
和 t1.join()
的事件序列。对于 t3
,您显然可以采用相同的方式思考。
执行t1.start()
之前的main()
指令
t1.start()
启动线程 t1
(t1.run()
可能不会立即启动。)
t1.start()
和t1.join()
之间的main()
指令并行执行( *) t1.run()
中的那些。
注意:您的示例中没有任何指令,因此此时仅执行 t1.run() 指令。
t1.join()
:
t1.run()
已经完成,什么也不会发生,main()
继续运行t1.run()
尚未完成,主线程将停止并等待直到 t1.run()
完成。然后 t1.run()
完成,然后 main()
恢复。t1.join()
执行后main()
的指令
在这里你可以看到:
t1.start()
之前的 main()
部分保证在 t1.run() 之前执行
t1.join()
之后的 main()
部分保证在 t1.run() 之后执行
(*) 请参阅下面有关并行性的部分
假设您在 2 个线程 A 和 B 中执行了这 2 组指令:
// Thread A | // Thread B
|
System.out.println("A1"); | System.out.println("B1");
System.out.println("A2"); | System.out.println("B2");
System.out.println("A3"); | System.out.println("B3");
如果这 2 个线程“并行执行”,这意味着 3 件事:
保证线程A的指令执行顺序:
A1 在 A2 之前执行,A2 在 A3 之前执行。
保证线程B的指令执行顺序:
B1 在 B2 之前执行,B2 在 B3 之前执行。
但是,A 和 B 的指令可以交错,这意味着以下所有情况都是可能的(甚至更多):
A1, B1, A2, B2, B3, A3
B1, B2, A1, B3, A2, A3
A1, A2, A3, B1, B2, B3 // special case where A's are all executed before B's
B1, B2, B3, A1, A2, A3 // special case where B's are all executed before A's
注意:本节将并行处理作为操作系统创建的一种错觉,让用户感觉事物同时运行,而实际上只是一个核心按顺序执行指令,从一个进程/线程跳转到另一个。
事实上,A 指令和 B 指令可以在 2 个独立的内核上同时执行(真正的并行性)。无论如何,上面的 3 个要点仍然有效。正如@jameslarge 指出的那样,通常我们使用一系列事件对并发进行建模,即使对于多核也是如此。这忽略了 2 个事件同时发生的概念,它不会带来任何有用的东西,只会带来复杂性。
关于java - join() 在 Java 中如何工作?能保证在main()之前执行吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23422970/
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
有没有办法保证您的系统托盘图标被删除? 添加系统托盘图标: Shell_NotifyIcon(NIM_ADD, &m_tnd); 删除系统托盘图标: Shell_NotifyIcon(NIM_DELE
是否保证(-x) % m,其中x和m在c++中为正standard (c++0x) 为负数,等于 -(x % m)? 我知道它在我知道的所有机器上都是正确的。 最佳答案 除了Luchian的回答,这是
可能还有其他方法可以作为示例,但这不是我要问的重点。 我正在这样做: (future (clojure.java.shell/sh "sleep" "3" :dir "/tmp")) 启动对Shell
可以使用 XREAD(或者可能是另一个命令)以原子方式检测数据是否写入 Redis 流? 进一步来说: 假设您在一个进程中将一些数据添加到 Redis 流中,并看到数据已通过某个自动生成的 key 成
Kotlin 协程是否提供任何“发生之前”保证? 例如,在这种情况下,写入 mutableVar 和随后在(可能)其他线程上读取之间是否存在“发生之前”保证: suspend fun doSometh
我正在开发一个跟踪行程的应用程序。在搜索了这件事之后,我得出结论,实现这一点(持续跟踪用户的位置)的最好方法是使用前台服务。在某些情况下工作得很好,但在其他一些情况下(即使关闭 DOZE),我得到一些
我正在使用 ORM (sqlalchemy) 从 PG 数据库中获取数据。我想避免在我手工编写的 SQL 语句中指定所有表列名称*。 到目前为止,我的假设是返回的列按照用于创建数据库表的 DDL 语句
在 setState 的文档中这样说: setState() does not immediately mutate this.state but creates a pending state tr
我有一个与不同硬件接口(interface)的简单应用程序。对于每个硬件,我针对一个独特的监视器函数生成了一个 pthread_t,总共有 6 个线程:1 个管理线程和 5 个工作线程。 每个线程都有
目前,我有 private ThreadLocal shortDateFormat = new ThreadLocal() { @Override protected DateFormat i
我有一个使用 SolrCloud 将文档写入 Solr 的 Java 作业。输入数据被转换为不同实体的映射,然后将每个实体写入与其实体类型对应的 Solr 集合。 我的代码如下: public voi
我们使用嵌入式设备通过串行到以太网转换器将数据包从串行端口发送到服务器。我们使用的一家制造商 Moxa 将始终以与构建它们相同的方式发送数据包。意思是,如果我们构建一个大小为 255 的数据包,它将始
我是从 C++ 转到 Java 的。在 C++ 世界中,我们关注异常安全,并注意到变元器可以在变元器本身或其委托(delegate)的方法抛出异常时提供不同的保证(最小、强、不抛出)。实现具有强异常保
我想将来自 SAAJ 的 SOAPConnectionFactory 和 MessageFactory 类与多个线程一起使用,但事实证明我不能假设它们是线程安全的。一些相关的帖子: javax.xml
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 5 年前。 Improve
关于正确性,我找不到以下代码片段没有设计缺陷的证据/反证据。 template class MyDirtyPool { public: template std::size_t ad
对于这个问题,我找到了不同的答案,我知道一定有一个确定的答案。 C 中四种主要数据类型的最小分配内存大小是多少? int , double , float , 和 char是我在想什么。做 signe
我正在使用 Kafka Producer,我的应用程序将具有相同键的各个 ProducerRecords 发送到单个分区中,然后这些 ProducerRecords 在发送到代理之前进行批处理(使用
您好,我是服务器端编程 (java) 的新手,正在阅读 SendRedirect 与 Forward 之间的区别。来自 Post-redirect-get pattern它解释说这将阻止通过点击刷新按
我是一名优秀的程序员,十分优秀!