Rumah  >  Artikel  >  Java  >  Jangan Biarkan Bujang Anda Berpisah! Inilah Cara Menjadikannya % Selamat Benang di Java

Jangan Biarkan Bujang Anda Berpisah! Inilah Cara Menjadikannya % Selamat Benang di Java

Patricia Arquette
Patricia Arquetteasal
2024-11-02 09:55:02277semak imbas

Don’t Let Your Singleton Break! Here’s How to Make It % Thread-Safe in Java

Dalam siaran ini, kami akan meneroka beberapa cara untuk melaksanakan singleton selamat-benang di Java, termasuk pemulaan yang tidak sabar-sabar, penguncian yang disemak dua kali, dan pendekatan kelas statik dalaman. Kami juga akan membincangkan sebab kata kunci terakhir bermanfaat dalam memastikan integriti satu tunggal.

Mengapa Menggunakan Singleton?

Singleton berguna apabila anda memerlukan tepat satu contoh kelas sepanjang aplikasi. Kes penggunaan biasa termasuk mengurus sumber yang dikongsi seperti pengelogan, konfigurasi atau kumpulan sambungan. Singleton memastikan bahawa berbilang permintaan untuk mengakses kelas berkongsi contoh yang sama, dan bukannya membuat yang baharu.

1. Permulaan Bersemangat: Singleton Paling Mudah

Corak permulaan yang bersemangat mencipta tika tunggal apabila kelas dimuatkan. Ini adalah mudah dan memastikan keselamatan rangkaian kerana kejadian dibuat apabila kelas dimuatkan oleh JVM.

public final class Singleton {
    // Instance is created at class loading time
    private static final Singleton INSTANCE = new Singleton();

    // Private constructor prevents instantiation from other classes
    private Singleton() {}

    public static Singleton getInstance() {
        return INSTANCE;
    }
}

Kebaikan dan Keburukan Permulaan Bersemangat

Kebaikan:

  • Mudah dan selamat untuk benang secara lalai, kerana JVM menjamin permulaan kelas adalah selamat untuk benang.
  • Tidak memerlukan penyegerakan atau kerumitan tambahan.

Keburukan:

  • Instance dibuat sama ada ia digunakan atau tidak, yang boleh menyebabkan sumber terbuang jika singleton tidak diperlukan.

Bila Perlu Digunakan: Pemmulaan yang bersemangat adalah yang terbaik apabila kelas tunggal adalah ringan dan anda pasti ia akan digunakan semasa masa jalan aplikasi.


2. Permulaan Malas dengan Penguncian Disemak Dua Kali

Dalam kes di mana anda ingin menangguhkan penciptaan singleton sehingga ia diperlukan (dikenali sebagai pemulaan malas), penguncian yang disemak dua kali menyediakan penyelesaian selamat benang. Ia menggunakan penyegerakan yang minimum dan memastikan tika itu dibuat hanya apabila ia diakses buat kali pertama.

public final class Singleton {  // Marked as final to prevent subclassing

    // volatile ensures visibility and prevents instruction reordering
    private static volatile Singleton instance;

    // Private constructor prevents instantiation from other classes
    private Singleton() {}

    public static Singleton getInstance() {
        if (instance == null) {               // First check (no locking)
            synchronized (Singleton.class) {   // Locking
                if (instance == null) {        // Second check (with locking)
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

Mengapa Penguncian Semakan Dua Kali Berfungsi

  1. Semakan Pertama: Semakan if (instance == null) di luar blok yang disegerakkan membolehkan kami mengelak daripada mengunci setiap kali getInstance() dipanggil. Ini meningkatkan prestasi dengan memintas blok disegerakkan untuk panggilan akan datang selepas permulaan.

  2. Blok Disegerakkan: Sebaik sahaja contoh adalah batal, memasukkan blok yang disegerakkan memastikan hanya satu urutan mencipta tikas Singleton. Urutan lain yang mencapai tahap ini mesti menunggu, menghalang keadaan perlumbaan.

  3. Semakan Kedua: Di dalam blok disegerakkan, kami menyemak contoh sekali lagi untuk memastikan tiada utas lain telah memulakannya semasa utas semasa sedang menunggu. Semakan dua kali ini memastikan hanya satu tika Singleton dibuat.

Mengapa tidak menentu Diperlukan

Kata kunci yang tidak menentu adalah penting dalam corak penguncian yang disemak dua kali untuk mengelakkan penyusunan semula arahan. Tanpa itu, contoh pernyataan = new Singleton(); mungkin kelihatan lengkap pada urutan lain sebelum ia dimulakan sepenuhnya, yang membawa kepada contoh yang sebahagiannya dibina dikembalikan. jaminan yang tidak menentu bahawa apabila contoh bukan nol, ia dibina sepenuhnya dan kelihatan kepada semua urutan.

Mengapa akhir ialah Amalan Baik

Kata kunci terakhir digunakan di sini untuk menghalang subkelas. Menandakan kelas Singleton sebagai akhir mempunyai dua faedah utama:

  1. Menghalang Subkelas: Dengan menjadikan kelas akhir, kami menghalang kelas lain daripada melanjutkannya. Ini memastikan bahawa hanya satu tika kelas Singleton boleh wujud kerana subkelas boleh membawa kepada kejadian tambahan, yang akan memecahkan corak tunggal.

  2. Ketidakbolehubah Isyarat: muktamad berfungsi sebagai penunjuk yang jelas kepada pembangun lain bahawa kelas tunggal bertujuan untuk tidak berubah dan tidak boleh dilanjutkan. Ini menjadikan kod lebih mudah difahami dan diselenggara.

Ringkasnya, akhir mengukuhkan integriti singleton dan membantu mengelakkan tingkah laku yang tidak dijangka daripada subkelas.

Kebaikan dan Keburukan Penguncian Disemak Dua Kali

Kebaikan:

  • Pengamalan yang malas menjimatkan sumber dengan menangguhkan penciptaan sehingga diperlukan.
  • overhed penyegerakan minimum disebabkan oleh semakan dua kali.

Keburukan:

  • Sedikit lebih kompleks dan memerlukan tidak menentu untuk keselamatan.
  • Boleh berlebihan untuk keperluan tunggal yang lebih ringkas di mana permulaan yang bersemangat adalah mencukupi.

Bila Menggunakannya: Corak ini berguna apabila kelas tunggal adalah intensif sumber dan mungkin tidak selalu diperlukan, atau apabila prestasi dalam persekitaran berbilang benang membimbangkan.


3. Kelas Statik Dalaman: Alternatif Permulaan Malas yang Lebih Bersih

Pendekatan selamat benang alternatif untuk permulaan malas ialah corak kelas statik dalaman. Ini memanfaatkan mekanisme pemuatan kelas Java untuk memulakan instance tunggal hanya apabila ia diperlukan, tanpa penyegerakan yang jelas.

public final class Singleton {
    // Instance is created at class loading time
    private static final Singleton INSTANCE = new Singleton();

    // Private constructor prevents instantiation from other classes
    private Singleton() {}

    public static Singleton getInstance() {
        return INSTANCE;
    }
}

Bagaimana Kelas Statik Dalaman Berfungsi

Dalam corak ini, kelas SingletonHelper hanya dimuatkan apabila getInstance() dipanggil buat kali pertama. Ini mencetuskan permulaan INSTANCE, memastikan pemuatan malas tanpa memerlukan sekatan yang disegerakkan.

Kebaikan dan Keburukan Kelas Statik Dalaman

Kebaikan:

  • Selamat untuk benang tanpa memerlukan sekatan yang tidak menentu atau disegerakkan.
  • Mudah dan bersih, dengan reka bentuk permulaan yang malas.

Keburukan:

  • Sedikit kurang intuitif untuk pembangun baharu yang tidak biasa dengan mekanik pemuatan kelas Java.

Bila Perlu Digunakan: Gunakan corak kelas statik dalaman apabila anda mahukan permulaan malas dengan kod yang bersih dan boleh diselenggara. Ini selalunya menjadi pilihan pilihan kerana kesederhanaan dan keselamatan benang.


Membungkus

Kami telah melihat tiga cara popular untuk melaksanakan singleton selamat benang di Java:

  1. Permulaan Bersemangat: Terbaik untuk kes mudah di mana singleton ringan dan akan sentiasa digunakan.
  2. Penguncian Bersemak Dua Kali: Bagus untuk persekitaran berbilang benang yang sensitif prestasi dan memerlukan pengamulaan yang malas.
  3. Kelas Statik Dalaman: Pendekatan yang bersih dan selamat dari benang untuk permulaan malas menggunakan gelagat pemuatan kelas Java.

Setiap pendekatan mempunyai kekuatannya dan sesuai untuk senario yang berbeza. Cuba mereka dalam projek anda sendiri dan lihat mana yang paling sesuai untuk anda! Beritahu saya dalam ulasan jika anda mempunyai pendekatan pilihan atau sebarang pertanyaan.

Selamat pengekodan! ?‍???‍?

Atas ialah kandungan terperinci Jangan Biarkan Bujang Anda Berpisah! Inilah Cara Menjadikannya % Selamat Benang di Java. 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