Rumah  >  Artikel  >  Java  >  Cara Membetulkan: Ralat Konkurensi Java: Mengelak Kebuntuan

Cara Membetulkan: Ralat Konkurensi Java: Mengelak Kebuntuan

PHPz
PHPzasal
2023-08-19 13:54:321395semak imbas

Cara Membetulkan: Ralat Konkurensi Java: Mengelak Kebuntuan

Cara menyelesaikan: Ralat Java Concurrency: Pengelakan Deadlock

Pengenalan:

Dalam pembangunan program Java, concurrency berbilang benang adalah penting. Walau bagaimanapun, pengaturcaraan serentak juga membawa beberapa masalah, salah satu masalah yang paling biasa dan berpotensi serius ialah kebuntuan. Kebuntuan merujuk kepada situasi di mana dua atau lebih utas memegang sumber yang diperlukan oleh satu sama lain, tetapi tidak dapat meneruskan pelaksanaan kerana pihak lain tidak melepaskan sumber. Artikel ini akan meneroka cara menyelesaikan isu kebuntuan dalam ralat konkurensi dalam Java dan memberikan beberapa contoh kod.

1. Fahami punca kebuntuan:

Sebelum selesaikan masalah kebuntuan, anda perlu faham dahulu punca kebuntuan. Kebuntuan biasanya berlaku apabila beberapa utas bersaing untuk berbilang sumber pada masa yang sama. Kebuntuan berlaku apabila dua atau lebih utas menunggu antara satu sama lain untuk mengeluarkan sumber yang diperlukan. Berikut ialah kod contoh mudah:

class Resource {
    private String name;

    public Resource(String name) {
        this.name = name;
    }

    public synchronized void doSomething() {
        System.out.println(name + " is doing something.");
    }

    public synchronized void doAnotherthing(Resource otherResource) {
        System.out.println(name + " is doing anotherthing.");
        otherResource.doSomething();
    }
}

public class DeadlockExample {
    public static void main(String[] args) {
        Resource resource1 = new Resource("Resource1");
        Resource resource2 = new Resource("Resource2");

        Thread t1 = new Thread(() -> {
            resource1.doAnotherthing(resource2);
        });

        Thread t2 = new Thread(() -> {
            resource2.doAnotherthing(resource1);
        });

        t1.start();
        t2.start();
    }
}

Dalam contoh di atas, terdapat dua sumber resource1 dan resource2. Dua utas t1 dan t2 dicipta dalam kaedah utama dan kaedah doAnotherthing sumber dipanggil masing-masing . Dalam urutan t1, ia memanggil kaedah doAnotherthing resource1 dan menghantar resource2 sebagai parameter. Dalam urutan t2, ia memanggil kaedah doAnotherthing resource2 dan menghantar resource1 sebagai parameter. resource1resource2。在main方法中创建了两个线程t1t2,并分别调用资源的doAnotherthing方法。在t1线程中,它调用resource1doAnotherthing方法,并传入resource2作为参数。在t2线程中,它调用resource2doAnotherthing方法,并传入resource1作为参数。

由于这两个线程互相等待对方释放所需的资源,所以会发生死锁。当然,这只是一个简单的示例,实际场景中可能包含更多资源和线程。

二、解决死锁问题:

  1. 预防死锁:

要预防死锁,首先需要了解死锁发生的原因。在上面的示例代码中,死锁是由于线程对资源的获取顺序不一致导致的。因此,我们可以通过规定线程获取资源的顺序来预防死锁。修改示例代码如下:

public class DeadlockExample {
    public static void main(String[] args) {
        Resource resource1 = new Resource("Resource1");
        Resource resource2 = new Resource("Resource2");

        Thread t1 = new Thread(() -> {
            synchronized (resource1) {
                System.out.println("Thread 1 acquired resource 1.");
                synchronized (resource2) {
                    System.out.println("Thread 1 acquired resource 2.");
                }
            }
        });

        Thread t2 = new Thread(() -> {
            synchronized (resource1) {
                System.out.println("Thread 2 acquired resource 1.");
                synchronized (resource2) {
                    System.out.println("Thread 2 acquired resource 2.");
                }
            }
        });

        t1.start();
        t2.start();
    }
}

通过对资源的获取顺序进行规定,确保不会出现互相等待对方所需的资源的情况,从而避免了死锁的发生。

  1. 死锁检测和恢复:

除了预防死锁,还可以通过死锁检测和恢复来解决死锁问题。Java提供了ThreadMXBean接口用于监测和管理线程的状态。以下是一个示例代码:

import java.lang.management.ManagementFactory;
import java.lang.management.ThreadMXBean;

public class DeadlockExample {
    public static void main(String[] args) {
        Resource resource1 = new Resource("Resource1");
        Resource resource2 = new Resource("Resource2");

        Thread t1 = new Thread(() -> {
            synchronized (resource1) {
                System.out.println("Thread 1 acquired resource 1.");

                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                synchronized (resource2) {
                    System.out.println("Thread 1 acquired resource 2.");
                }
            }
        });

        Thread t2 = new Thread(() -> {
            synchronized (resource2) {
                System.out.println("Thread 2 acquired resource 2.");

                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                synchronized (resource1) {
                    System.out.println("Thread 2 acquired resource 1.");
                }
            }
        });

        t1.start();
        t2.start();

        ThreadMXBean threadMxBean = ManagementFactory.getThreadMXBean();
        long[] deadlockedThreadIds = threadMxBean.findDeadlockedThreads();

        if (deadlockedThreadIds != null) {
            ThreadInfo[] threadInfos = threadMxBean.getThreadInfo(deadlockedThreadIds);
            for (ThreadInfo threadInfo : threadInfos) {
                System.out.println(threadInfo.getThreadName() + " is deadlocked.");
                // 恢复死锁线程的执行,或者进行其他操作
            }
        }
    }
}

在上面的示例代码中,我们通过ThreadMXBeanfindDeadlockedThreads

Kebuntuan berlaku kerana kedua-dua utas ini menunggu antara satu sama lain untuk mengeluarkan sumber yang diperlukan. Sudah tentu, ini hanyalah contoh mudah, dan senario sebenar mungkin termasuk lebih banyak sumber dan rangkaian.

2. Selesaikan masalah kebuntuan:

Cegah kebuntuan:

  1. Untuk mengelakkan kebuntuan, anda perlu memahami punca kebuntuan terlebih dahulu. Dalam kod contoh di atas, kebuntuan disebabkan oleh urutan memperoleh sumber dalam susunan yang tidak konsisten. Oleh itu, kita boleh mengelakkan kebuntuan dengan menyatakan susunan urutan memperoleh sumber. Ubah suai kod sampel seperti berikut:
  2. rrreee
  3. Dengan menetapkan susunan sumber diperoleh, ia memastikan satu sama lain tidak menunggu sumber yang diperlukan oleh pihak yang satu lagi, dengan itu mengelakkan berlakunya kebuntuan.
    1. Pengesanan dan pemulihan jalan buntu: 🎜🎜🎜Selain mengelakkan kebuntuan, masalah kebuntuan juga boleh diselesaikan melalui pengesanan dan pemulihan kebuntuan. Java menyediakan antara muka ThreadMXBean untuk memantau dan mengurus status urutan. Berikut ialah kod sampel: 🎜rrreee🎜Dalam kod sampel di atas, kami menemui utas di mana kebuntuan berlaku melalui kaedah findDeadlockedThreads ThreadMXBean dan mengendalikannya dengan sewajarnya. Anda boleh menyambung semula pelaksanaan benang yang buntu, atau melakukan operasi lain. 🎜🎜Kesimpulan: 🎜🎜Kebuntuan adalah salah satu masalah biasa dalam pengaturcaraan serentak berbilang benang Jika tidak diselesaikan, ia boleh menyebabkan program ranap atau gagal meneruskan pelaksanaan. Artikel ini memperkenalkan dua kaedah untuk menyelesaikan masalah kebuntuan iaitu pencegahan kebuntuan dan pengesanan dan pemulihan kebuntuan. Sudah tentu, ini hanyalah beberapa penyelesaian asas, dan strategi yang lebih kompleks mungkin diperlukan untuk menyelesaikan masalah kebuntuan dalam aplikasi sebenar. Pembangun harus memberi perhatian untuk mengelakkan kebuntuan semasa menulis program serentak berbilang benang dan mengendalikannya dengan sewajarnya untuk memastikan kestabilan dan kebolehpercayaan program. 🎜🎜Bahan rujukan: 🎜🎜🎜[Pengaturcaraan serentak Java: pemahaman mendalam tentang disegerakkan](https://www.jianshu.com/p/6d293a1a412c)🎜🎜[Analisis dan penyelesaian masalah kebuntuan benang Java](https: //blog .csdn.net/coslay/article/details/78387673)🎜🎜

    Atas ialah kandungan terperinci Cara Membetulkan: Ralat Konkurensi Java: Mengelak Kebuntuan. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

    Kenyataan:
    Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn