我写了一个启动两个线程的代码片段;一个线程打印所有奇数,而另一个线程打印所有偶数。我使用了内在锁和线程通信命令的组合来实现两个线程的正确交叉。这是我的代码,
public class threadEvenOdd implements Runnable
{
static Boolean isOdd=true;
int count = 10;
Boolean value;
static int c=1;
static Object lock = new Object();
threadEvenOdd(Boolean temp)
{
value = temp;
}
public void run()
{
if(value)
{
printOdd(count);
}
if(!value)
{
printEven(count);
}
}
void printOdd(int count)
{
try
{
for(int i=0;i<count/2;i++)
{
//System.out.println("odd enters lock");
synchronized(lock)
{
if(!isOdd)
{
//System.out.println("odd in barrier");
lock.wait();
}
System.out.println(c);
c++;
isOdd = false;
//System.out.println("odd notifies");
lock.notify();
}
}
}
catch(Exception e)
{
System.out.println(e);
}
}
void printEven(int count)
{
try
{
for(int i=0;i<count/2;i++)
{
//System.out.println("even enters lock");
synchronized(lock)
{
if(isOdd)
{
//System.out.println("even in barrier");
lock.wait();
}
System.out.println(c);
c++;
isOdd = true;
//System.out.println("even notifies");
lock.notify();
}
}
}
catch(Exception e)
{
System.out.println(e);
}
}
public static void main (String args[])
{
threadEvenOdd th1 = new threadEvenOdd(true);
threadEvenOdd th2 = new threadEvenOdd(false);
Thread t1 = new Thread(th1);
t1.setName("odd");
Thread t2 = new Thread(th2);
t2.setName("even");
//System.out.println(t1.getName() + " starts");
t1.start();
//System.out.println(t2.getName() + " starts");
t2.start();
}
}
以下是我的问题:
>
奇数线程在printOdd()函数中执行,而偶数线程在print偶数()函数中执行。我对两个线程都使用一个内在锁;我不明白两个线程怎么能同时存在于各自的同步块中,因为使用了相同的锁。
我从代码中删除了线程通信语句(通知,等待),但仍然获得了所需的输出。我现在想知道我的代码是否真的需要线程通信语句。
我想我仍然需要努力理解多线程概念,因为我正在努力理解我自己的代码:p谁能解释一下是否有更好的方法来做到这一点,只使用我使用的html" target="_blank">多线程概念?
>
每个线程都有自己的代码执行路径。即使两个线程运行完全相同的代码,它们仍然有两个不同的执行点,通过代码执行代码。当线程到达同步语句时,它会等待锁可用 - 仅当由同一锁保护的同步块内没有其他线程时,它才会进入同步块。
尽管删除了notify/wait语句,但仍会得到相同的输出。您是否尝试使用相对较大的count
字段值来执行此操作?
现在很难回答这个问题,因为你没有具体说明你期望这个程序产生什么输出。“1,3,5,7,9,2,4,6,8”是有效输出吗?是“1,3,2,4,6,5,7,9,8”?还是说“1,2,3,4,5,6,7,8,9”是唯一有效的输出?也就是说,这里有几个要点:
使用 notifyAll() 代替通知
最小化线程间共享的状态。在这种情况下,您共享< code>isOdd和< code>c。请注意,前者可以通过< code>c % 2 == 1从后者计算得出。因此,您可以让线程计算异常,而不是将其作为共享数据来维护。
不是通过静态字段共享,而是创建一个对象(带有实例字段)并将此对象传递给每个线程的构造函数。然后您可以将对象本身用作锁。
下面是它的外观:
class SharedData {
int c;
boolean isOdd;
}
class ThreadEvenOdd {
SharedData sharedData;
public ThreadEvenOdd(SharedData sd) { this.sharedData = sd }
// ...
void printOdd(int count) {
try {
for(int i=0;i<count/2;i++) {
synchronized(sharedData) {
if(!sharedData.isOdd) { ... }
System.out.println(sharedData.c);
sharedData.c++;
sharedData.isOdd = false;
lock.notify();
}
}
}
catch(Exception e) {
System.out.println(e);
}
}
}
这样做的好处是,您可以开始在 sharedData 上定义真正的方法(例如:一种增加 c 并根据 c
的值将 isOdd
设置为适当值的方法,从而进一步简化了线程类中的代码 - 并使同步/通知与数据处理的交错更少, 这使得代码更具可读性,不易出错。
下面是
如果我没有错的话,Thread-1和thread-3正在进入synchronized方法,因为它有两个不同的目标对象。但是为什么线程2进入同步块呢? 请帮助我理解这一点。提前谢了。
我试图制作一个时钟,如果 如何同时运行警告语音和时钟,使时钟在播放警告语音()时不会停止? 语音代码: 时钟代码:
问题内容: 我使用的不是可重入的库(用C编写)(即库中没有函数可重入)。假设我已经通过System.load加载了库以获取句柄“ v”。由于重入问题(尝试过但无意义的结果),我无法在两个线程中使用v。我可以使用锁,但这会破坏我本可以获得的任何并行性。 我想做的是启动两个线程,然后在每个线程中加载库以获取两个不同的句柄(因此,加载的库有两个副本)。 这在Java中可行吗?问候Saptarshi 问题
问题内容: 我想知道如果在同一个对象上同步两次,在Java中是否会出现任何奇怪的行为? 场景如下 两种方法都使用该对象并对其进行同步。当第一个方法调用第二个方法时,它会被锁定而停止吗? 我不这么认为,因为它是同一个线程,但是我不确定是否可能会出现其他任何奇怪的结果。 问题答案: 同步块使用 可重入 锁,这意味着如果线程已经持有该锁,则它可以重新获取它而不会出现问题。因此,您的代码将按预期工作。 请
问题内容: 如何在同一“行”上的两个div块居中? 第一格: 第二个div: 问题答案: CSS: HTML 另外,您不应将原始内容放入中,而应使用诸如或的适当标签。