Поток вошел в состояние взаимоблокировки при использовании анонимного потока и попытке печати нечетных, четных чисел двумя потоками - PullRequest
0 голосов
/ 26 апреля 2020

Я пытался печатать нечетные, четные числа двумя потоками, повторяя, используя wait и notify.

Я могу добиться этого путем реализации интерфейса Runnable

public class OddEven implements Runnable {
    public void run() {
        for (int i = 1; i <= 100; i++) {
            if (i % 2 == Integer.parseInt(Thread.currentThread().getName())) {
                synchronized (this) {
                    notifyAll();
                    System.out.println((Thread.currentThread().getName().equals("1") ? "odd : " : "even: ") + i);
                    try {
                        if (i != 100)
                            wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }

    public static void main(String[] args) {
        OddEven obj = new OddEven();
        Thread t1 = new Thread(obj, "1");
        Thread t2 = new Thread(obj, "0");
        t1.start();
        t2.start();
    }
}

    O/P is:

    Odd : 1
    Even: 2
    Odd : 3
    Even: 4
    Odd : 5
    .
    .
    .
    .
    Odd : 97
    Even: 98
    Odd : 99
    Even: 100

Затем я попробовал то же самое путем расширения класса Thread

public class Test {
    public static void main(String[] args) {
        Test t = new Test();
        new OddThraed(t).start();
        new EvenThraed(t).start();
    }
}

class OddThraed extends Thread {
    Test t;

    public OddThraed(Test t) {
        this.t = t;
    }

    public void run() {
        for (int i = 1; i <= 100; i++) {
            if (i % 2 == 1) {
                synchronized (t) {
                    t.notifyAll();
                    System.out.println("Odd : " + i);
                    try {                       
                        if (i != 100)
                            t.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }
}

class EvenThraed extends Thread {
    Test t;

    public EvenThraed(Test t) {
        this.t = t;
    }

    public void run() {
        for (int i = 1; i <= 100; i++) {
            if (i % 2 == 0) {
                synchronized (t) {
                    t.notifyAll();
                    System.out.println("Even: " + i);
                    try {                       
                        if (i != 100)
                            t.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }
}

O / P было ожидаемым

Тогда я думаю, что существует слишком много классов и пытался достичь этого с помощью анонимного потока

    public class TestOddEvenAnonymousThread {
        public static void main(String[] args) {
            (new Thread("Odd") {
                public void run() {
                    for (int i = 1; i <= 100; i++) {
                        if (i % 2 == 1) {
                            synchronized (this) {
                                notifyAll();
                                System.out.println("Odd : " + i);
                                try {
                                    if (i != 100)
                                        wait();
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                            }
                        }
                    }
                }
            }).start();

            (new Thread("Even") {
                public void run() {
                    for (int i = 1; i <= 100; i++) {
                        if (i % 2 == 0) {
                            synchronized (this) {
                                notifyAll();
                                System.out.println("Even: " + i);
                                try {
                                    if (i != 100)
                                        wait();
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                            }
                        }
                    }
                }
            }).start();

        }
    }

But this time O/P is

Odd : 1
Even: 2

и есть тупик, может кто-нибудь объяснить, почему последний не показывает соответствующий o / p, где я просто пытаюсь достичь этого с помощью анонимного потока, а не путем расширения отдельного класса потока (что я сделал во втором подходе выше)

Ответы [ 2 ]

1 голос
/ 26 апреля 2020
class TestOddEvenAnonymousThread {

    public static void main(String[] args) {
        TestOddEvenAnonymousThread t = new TestOddEvenAnonymousThread();
        Thread oddT = (new Thread("Odd") {
            public void run() {
                for (int i = 1; i <= 100; i++) {
                    if (i % 2 == 1) {
                        synchronized (t) {
                            t.notifyAll();
                            System.out.println("Odd : " + i);
                            try {
                                if (i != 100)
                                    t.wait();
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            }
        });
        Thread eventT = (new Thread("Even") {
            public void run() {
                for (int i = 1; i <= 100; i++) {
                    if (i % 2 == 0) {
                        synchronized (t) {
                            t.notifyAll();
                            System.out.println("Even: " + i);
                            try {
                                if (i != 100)
                                    t.wait();
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            }
        });
                oddT.start();
                eventT.start();
    }


}

Когда вы сослались на это в старом потоке событий и неосознанно, вы применили блок синхронизации к другим и вызвали notifyAll и wait к соответствующим объектам

0 голосов
/ 26 апреля 2020

Во второй версии вы дали потокам один и тот же объект блокировки

        Test t = new Test();
        new OddThraed(t).start();
        new EvenThraed(t).start();

и, например,

                synchronized (t) {
                    t.notifyAll();
                    System.out.println("Odd : " + i);
                    try {                       
                        if (i != 100)
                            t.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }

В третьей версии вы используете this в качестве объекта блокировки каждый

                            synchronized (this) {
                                notifyAll();
                                System.out.println("Odd : " + i);
                                try {
                                    if (i != 100)
                                        wait();
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                            }

Методы wait() и notify() / notifyAll() связаны с объектом для блокировки. Поток ожидает объекта блокировки и получает уведомление о том же объекте блокировки. Когда вы создаете два таких потока, каждый из них имеет свой собственный объект блокировки, который является самим потоком.

Застой в том, что поток A ожидает ObjectA, а поток B ожидает ObjectB , Когда A делает notifyAll(), то уведомляются только потоки, ожидающие ObjectA, а не те, которые ожидают ObjectB.

Если вам интересно, почему ваша первая версия работает: Да, есть this также используется для блокировки , но оба потока имеют одинаковые this.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...