Primitif serentak Java termasuk kelas yang disegerakkan, tidak menentu, atom, CAS dan kunci kunci, yang merupakan asas untuk membina aplikasi konvensional yang tinggi. 1. Disegerakkan memastikan atomik dan penglihatan melalui kunci monitor, dan menggunakan halangan memori untuk mengelakkan pengajaran semula; 2. Tidak menentu memastikan penglihatan yang berubah -ubah dan melarang pengajaran semula, sesuai untuk bendera status dan mod singleton; 3. 4. Locksupport menyediakan sokongan asas untuk penggantungan benang dan bangun, yang lebih fleksibel daripada menunggu/memberitahu dan tidak memerlukan kunci. Memahami primitif ini dapat membantu meningkatkan keupayaan pengaturcaraan serentak dan menyelesaikan masalah praktikal.
Pengaturcaraan serentak di Java adalah sekeping kandungan yang banyak pemaju tidak dapat dielakkan, terutama ketika membina aplikasi berprestasi tinggi dan sangat serentak. Ramai orang telah menggunakan synchronized
dan volatile
, tetapi tidak banyak orang yang benar -benar memahami primitif concurrency Java. Artikel ini tidak bercakap tentang pembungkusan lanjutan seperti Pool Thread dan CompletableFuture, tetapi bermula dari lapisan bawah dan bercakap tentang beberapa primitif keseragaman teras yang biasanya kita gunakan.

Apakah primitif serentak?
Primitif serentak merujuk kepada mekanisme penyegerakan peringkat yang paling asas dan paling rendah yang disediakan oleh sistem operasi atau tahap bahasa. Di Jawa, primitif ini termasuk, tetapi tidak terhad kepada:

- Kata kunci
synchronized
- pembolehubah
volatile
- Kelas atom di bawah
java.util.concurrent.atomic
Package -
LockSupport.park()
/unpark()
- Cas (bandingkan dan bertukar)
Mekanisme ini bukan kelas alat serentak (seperti reentrantlock, countdownlatch), tetapi asas untuk membina alat ini. Mereka biasanya berkait rapat dengan model memori JVM (JMM), yang mempengaruhi penglihatan berubah, perintah pelaksanaan dan tingkah laku lain.
Apa sebenarnya yang diselaraskan?
synchronized
adalah salah satu kata kunci terawal yang Java menyokong kesesuaian. Ia bukan sahaja menjamin atom, tetapi juga mengawal akses ke zon kritikal melalui kunci monitor terbina dalam.

Anda mungkin tahu bahawa ia boleh mengunci, tetapi anda tidak semestinya memahami mekanisme penghalang memori di belakangnya:
- Masukkan halangan beban/kedai apabila mengunci untuk memastikan data terkini dibaca.
- Masukkan penghalang kedai sebelum membuka kunci untuk memastikan pengubahsuaian dapat dilihat pada benang lain.
- Ia juga menghalang arahan daripada menyusun semula.
Sebagai contoh: pelbagai benang memanggil kaedah synchronized
pada masa yang sama, hanya satu benang yang boleh masuk, dan selebihnya benang blok dan tunggu. Di sebalik proses ini sebenarnya perubahan keadaan tanda kata dalam tajuk objek di tempat kerja.
Walau bagaimanapun, perlu diperhatikan bahawa prestasi synchronized
memang lebih buruk dalam versi awal, tetapi selepas memperkenalkan pengoptimuman seperti kunci berat sebelah dan kunci ringan dari JDK6, prestasinya sangat baik.
Tidak menentu: Bukan hanya "penglihatan"
volatile
sering digunakan untuk menyelesaikan masalah penglihatan yang berubah -ubah. Sebagai contoh, jika bendera Boolean tidak ditambah dalam persekitaran pelbagai threaded, satu benang mungkin tidak dapat melihat benang lain jika ia diubah.
Tetapi sebenarnya, ia mempunyai dua ciri penting:
- Dilarang untuk menyusun semula arahan
- Tulis operasi berlaku sebelum membaca operasi
Ini menjadikan volatile
bukan sahaja sesuai untuk bendera status mudah, tetapi juga untuk reka bentuk dengan struktur bebas kunci, seperti penguncian cek berganda dalam mod singleton.
kelas awam singleton { contoh singleton swasta swasta; singleton statik awam getInstance () { jika (contoh == null) { disegerakkan (singleton.class) { jika (contoh == null) { instance = singleton baru (); } } } Contoh kembali; } }
volatile
di sini adalah untuk mengelakkan pengajaran menyusun semula objek baru daripada menyebabkan benang lain untuk mendapatkan objek yang tidak terbentuk.
Kelas atom dan CAS: tanpa kunci boleh selamat
Di sebalik AtomicInteger
, AtomicReference
dan kelas-kelas lain adalah operasi CAS, iaitu, membandingkan-dan-swap. Ia adalah mekanisme penguncian yang optimis yang sesuai untuk senario dengan kurang konflik.
Cas mempunyai tiga pengendali: Alamat Memori V, Nilai yang dijangkakan E, dan Nilai Baru U. Nilai ini dikemas kini kepada anda hanya jika nilai dalam memori sama dengan E.
CAS di Java dilaksanakan melalui kelas Unsafe
, dan asas Unsafe
bergantung pada arahan CPU (seperti cmpxchg
pada x86).
Kelebihan CAS adalah bahawa tidak ada penyekatan benang, dan kelemahannya adalah bahawa masalah ABA mungkin berlaku. Anda boleh menggunakan AtomicStampedReference
untuk menaip cap waktu untuk mengelakkan salah faham.
Penggunaan yang disyorkan:
- Sesuai untuk senario di mana lebih banyak bacaan dan kurang persaingan tidak sengit
- Perhatikan bahawa terlalu banyak kitaran boleh menyebabkan penggunaan CPU yang tinggi
- Elakkan semula tak terhingga di bawah jumlah yang besar, yang boleh digabungkan dengan mengehadkan spin.
Locksupport: Cara penjadualan benang asal
Jika anda telah melihat kod sumber AbstractQueuedSynchronizer
(AQS), anda akan mendapati bahawa ia menggunakan LockSupport.park()
dan LockSupport.unpark()
secara dalaman untuk menggantung dan bangun benang.
Kesan kedua -dua kaedah ini sangat langsung:
-
park()
: Biarkan benang semasa tidur sehingga tidak ada atau terganggu -
unpark(Thread)
: Bangun benang yang ditentukan
Berbanding dengan Object.wait()
dan notify()
, ia lebih fleksibel, boleh dipanggil tanpa memegang kunci, dan tidak membuang pengecualian.
Tetapi ia juga lebih terdedah kepada kesilapan. Sebagai contoh, memanggil unpark
dua kali hanya akan membangunkan benang sekali, dan park
panggilan berikutnya tidak akan kembali dengan segera.
Pada dasarnya itu sahaja. Walaupun primitif serentak Java kelihatan mudah, masing -masing mempunyai skop aplikasi dan syarat sempadan. Memahami mekanisme kerja mereka bukan sahaja membantu menulis kod serentak yang lebih cekap, tetapi juga membantu menyelesaikan masalah kebuntuan, penglihatan memori dan isu -isu lain.
Atas ialah kandungan terperinci Menyelam jauh ke dalam primitif bersamaan Jawa. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undress AI Tool
Gambar buka pakaian secara percuma

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Clothoff.io
Penyingkiran pakaian AI

Video Face Swap
Tukar muka dalam mana-mana video dengan mudah menggunakan alat tukar muka AI percuma kami!

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Java menyokong pengaturcaraan asynchronous termasuk penggunaan aliran yang boleh diselesaikan, aliran responsif (seperti ProjectReactor), dan benang maya di Java19. 1.CompletableFuture meningkatkan kebolehbacaan dan penyelenggaraan kod melalui panggilan rantai, dan menyokong orkestrasi tugas dan pengendalian pengecualian; 2. ProjectReactor menyediakan jenis mono dan fluks untuk melaksanakan pengaturcaraan responsif, dengan mekanisme tekanan belakang dan pengendali yang kaya; 3. Thread maya mengurangkan kos konvensional, sesuai untuk tugas I/O-intensif, dan lebih ringan dan lebih mudah untuk berkembang daripada benang platform tradisional. Setiap kaedah mempunyai senario yang berkenaan, dan alat yang sesuai harus dipilih mengikut keperluan anda dan model campuran harus dielakkan untuk mengekalkan kesederhanaan

Di Java, enums sesuai untuk mewakili set tetap tetap. Amalan terbaik termasuk: 1. Gunakan enum untuk mewakili keadaan tetap atau pilihan untuk meningkatkan keselamatan jenis dan kebolehbacaan; 2. Tambah sifat dan kaedah untuk meningkatkan fleksibiliti, seperti menentukan bidang, pembina, kaedah penolong, dan lain -lain; 3. Gunakan enummap dan enumset untuk meningkatkan prestasi dan jenis keselamatan kerana mereka lebih cekap berdasarkan tatasusunan; 4. Elakkan penyalahgunaan enum, seperti nilai dinamik, perubahan kerap atau senario logik kompleks, yang harus digantikan dengan kaedah lain. Penggunaan enum yang betul boleh meningkatkan kualiti kod dan mengurangkan kesilapan, tetapi anda perlu memberi perhatian kepada sempadannya yang berkenaan.

Javanio adalah IOAPI baru yang diperkenalkan oleh Java 1.4. 1) bertujuan untuk penampan dan saluran, 2) mengandungi komponen teras penampan, saluran dan pemilih, 3) menyokong mod tidak menyekat, dan 4) mengendalikan sambungan serentak lebih cekap daripada IO tradisional. Kelebihannya dicerminkan dalam: 1) IO yang tidak menyekat mengurangkan overhead thread, 2) Buffer meningkatkan kecekapan penghantaran data, 3) pemilih menyedari multiplexing, dan 4) memori pemetaan memori sehingga membaca dan menulis fail. Nota Apabila menggunakan: 1) Operasi flip/jelas penampan mudah dikelirukan, 2) Data yang tidak lengkap perlu diproses secara manual tanpa menyekat, 3) Pendaftaran pemilih mesti dibatalkan dalam masa, 4) NIO tidak sesuai untuk semua senario.

HashMap melaksanakan penyimpanan pasangan nilai utama melalui jadual hash di Java, dan terasnya terletak di lokasi data yang cepat. 1. Mula -mula gunakan kaedah hashcode () kunci untuk menghasilkan nilai hash dan mengubahnya menjadi indeks array melalui operasi bit; 2 Objek yang berbeza boleh menghasilkan nilai hash yang sama, mengakibatkan konflik. Pada masa ini, nod dipasang dalam bentuk senarai yang dipautkan. Selepas JDK8, senarai yang dipautkan terlalu panjang (panjang lalai 8) dan ia akan ditukar kepada pokok merah dan hitam untuk meningkatkan kecekapan; 3. Apabila menggunakan kelas tersuai sebagai kunci, sama () dan kaedah hashcode () mesti ditulis semula; 4. HashMap secara dinamik mengembangkan kapasiti. Apabila bilangan elemen melebihi kapasiti dan multiplies oleh faktor beban (lalai 0.75), mengembangkan dan mengembalikan; 5. hashmap tidak selamat benang, dan concu harus digunakan dalam multithreaded

Penghitungan Java bukan sahaja mewakili pemalar, tetapi juga boleh merangkum tingkah laku, membawa data, dan melaksanakan antara muka. 1. Penghitungan adalah kelas yang digunakan untuk menentukan contoh tetap, seperti minggu dan negeri, yang lebih selamat daripada rentetan atau bilangan bulat; 2. Ia boleh membawa data dan kaedah, seperti nilai lulus melalui pembina dan menyediakan kaedah akses; 3. Ia boleh menggunakan suis untuk mengendalikan logik yang berbeza, dengan struktur yang jelas; 4. Ia boleh melaksanakan antara muka atau kaedah abstrak untuk membuat tingkah laku yang berbeza dari nilai penghitungan yang berbeza; 5. Beri perhatian untuk mengelakkan penyalahgunaan, perbandingan kod keras, pergantungan pada nilai ordinal, dan penamaan dan bersiri yang munasabah.

Corak reka bentuk singleton di Java memastikan bahawa kelas hanya mempunyai satu contoh dan menyediakan titik akses global melalui pembina swasta dan kaedah statik, yang sesuai untuk mengawal akses kepada sumber yang dikongsi. Kaedah pelaksanaan termasuk: 1. 2. Pemprosesan Safe Thread, memastikan bahawa hanya satu contoh yang dibuat dalam persekitaran berbilang threaded melalui kaedah penyegerakan atau penguncian cek berganda, dan mengurangkan kesan prestasi; 3. 4. Pelaksanaan penghitungan, menggunakan penghitungan Java untuk secara semulajadi menyokong serialisasi, keselamatan thread dan mencegah serangan reflektif, adalah kaedah ringkas dan boleh dipercayai. Kaedah pelaksanaan yang berbeza boleh dipilih mengikut keperluan khusus

Pilihan dapat jelas menyatakan niat dan mengurangkan bunyi kod untuk penghakiman null. 1. Pilihan.Ofnullable adalah cara biasa untuk menangani objek null. Sebagai contoh, apabila mengambil nilai dari peta, Orelse boleh digunakan untuk memberikan nilai lalai, supaya logik lebih jelas dan ringkas; 2. Gunakan panggilan rantaian peta untuk mencapai nilai bersarang untuk menghindari NPE dengan selamat, dan secara automatik menamatkan jika ada pautan adalah null dan mengembalikan nilai lalai; 3. Penapis boleh digunakan untuk penapisan bersyarat, dan operasi seterusnya akan terus dilakukan hanya jika syarat -syarat dipenuhi, jika tidak, ia akan melompat terus ke Orelse, yang sesuai untuk penghakiman perniagaan ringan; 4. Ia tidak disyorkan untuk menggunakan terlalu banyak pilihan, seperti jenis asas atau logik mudah, yang akan meningkatkan kerumitan, dan beberapa senario akan terus kembali ke NU.

Penyelesaian teras untuk menghadapi java.io.notserializableException adalah untuk memastikan bahawa semua kelas yang perlu bersiri melaksanakan antara muka berseri dan periksa sokongan serialisasi objek bersarang. 1. Tambah implementsSerializable ke kelas utama; 2. Pastikan kelas medan tersuai yang sepadan di dalam kelas juga melaksanakan bersiri; 3. Gunakan sementara untuk menandakan medan yang tidak perlu bersiri; 4. Periksa jenis yang tidak berseri dalam koleksi atau objek bersarang; 5. Semak kelas mana yang tidak melaksanakan antara muka; 6. Pertimbangkan reka bentuk pengganti untuk kelas yang tidak dapat diubah suai, seperti menyimpan data utama atau menggunakan struktur pertengahan berseri; 7. Pertimbangkan untuk mengubah suai
