当前位置: 首页 > 知识库问答 >
问题:

两个线程,两个同步块和一个内在锁

吴举
2023-03-14

我写了一个启动两个线程的代码片段;一个线程打印所有奇数,而另一个线程打印所有偶数。我使用了内在锁和线程通信命令的组合来实现两个线程的正确交叉。这是我的代码,

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">多线程概念?

  • 共有1个答案

    南门欣怡
    2023-03-14

    >

  • 每个线程都有自己的代码执行路径。即使两个线程运行完全相同的代码,它们仍然有两个不同的执行点,通过代码执行代码。当线程到达同步语句时,它会等待锁可用 - 仅当由同一锁保护的同步块内没有其他线程时,它才会进入同步块。

    尽管删除了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 另外,您不应将原始内容放入中,而应使用诸如或的适当标签。