Pengenalan:
Corak Pengeluar/Pengguna ialah corak reka bentuk konkurensi klasik yang melibatkan benang Pengeluar menghasilkan data dan meletakkannya dalam baris gilir, manakala utas Pengguna mendapatkan dan memproses data daripada baris gilir yang sama. Corak ini memastikan aliran data dan penyegerakan antara dua utas.
Pelaksanaan Baris Gilir:
Dalam contoh kod yang diberikan, kelas QueueHandler mewakili baris gilir yang menjadi pengantara pertukaran data antara benang Pengeluar dan Pengguna. Persoalan utama berkisar tentang pelaksanaan baris gilir mana yang lebih disukai.
Pendekatan 1: Contoh Gilir Statik
Dalam pendekatan pertama, kelas QueueHandler mempunyai tika Queue statik dinamakan readQ, diakses melalui kaedah enqueue() dan dequeue(). Walaupun pendekatan ini memastikan keselamatan benang, ia tidak mempunyai kefleksibelan kerana saiz baris gilir ditetapkan pada permulaan dan tidak boleh dilaraskan secara dinamik.
Pendekatan 2: Baris Berasaskan Contoh
Dalam pendekatan kedua, baris gilir diluluskan sebagai hujah kepada pembina Pengguna dan Pengeluar. Ini membolehkan setiap utas mempunyai contoh baris gilir sendiri, memberikan lebih fleksibiliti dan skalabiliti. Kelas QueueHandler dilanjutkan untuk mencipta contoh QueueHandler yang selamat untuk benang.
Pendekatan Optimum:
Dari sudut kebolehselenggaraan dan kebolehskalaan, pendekatan kedua dengan berasaskan contoh beratur adalah lebih diingini. Ia membenarkan pengurusan baris gilir dinamik, memenuhi keperluan beban kerja yang berbeza-beza dan membolehkan utas beroperasi secara bebas.
Menggunakan Alat Java Concurrency:
Alternatif untuk mengurus baris gilir secara manual ialah untuk menggunakan alat konkurensi terbina dalam Java, seperti ExecutorServices dan BlockingQueues. Pendekatan ini memudahkan pelaksanaan dan memberikan lebih fleksibiliti dalam mengurus kumpulan benang dan pemindahan data.
Contoh Disemak Menggunakan ExecutorServices:
import java.util.concurrent.BlockingQueue; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class ProducerConsumerUsingExecutorService { private static final BlockingQueue<Object> queue = new LinkedBlockingQueue<>(); private static final ExecutorService producers = Executors.newFixedThreadPool(100); private static final ExecutorService consumers = Executors.newFixedThreadPool(100); public static void main(String[] args) { // Submit producers to theExecutorService for (int i = 0; i < 100; i++) { producers.submit(new Producer(queue)); } // Submit consumers to the ExecutorService for (int i = 0; i < 100; i++) { consumers.submit(new Consumer(queue)); } // Shutdown and await completion of producers and consumers producers.shutdown(); producers.awaitTermination(Long.MAX_VALUE, TimeUnit.NANOSECONDS); consumers.shutdown(); consumers.awaitTermination(Long.MAX_VALUE, TimeUnit.NANOSECONDS); } private static class Producer implements Runnable { private final BlockingQueue<Object> queue; public Producer(BlockingQueue<Object> queue) { this.queue = queue; } @Override public void run() { while (true) { // Add objects to the queue try { queue.put(new Object()); } catch (InterruptedException e) { e.printStackTrace(); } } } } private static class Consumer implements Runnable { private final BlockingQueue<Object> queue; public Consumer(BlockingQueue<Object> queue) { this.queue = queue; } @Override public void run() { while (true) { // Get and process objects from the queue try { Object object = queue.take(); // Process object } catch (InterruptedException e) { e.printStackTrace(); } } } } }
Pendekatan ini menyediakan yang lebih berskala dan fleksibel pelaksanaan corak Pengeluar/Pengguna, memanfaatkan kuasa konkurensi terbina dalam Java alatan.
Atas ialah kandungan terperinci Baris Berasaskan Statik lwn.: Manakah yang Lebih Diutamakan dalam Benang Pengeluar/Pengguna?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!