


1. Pourquoi la communication entre les threads est nécessaire
Les threads sont exécutés simultanément, ce qui semble être une exécution aléatoire des threads. Cependant, dans les applications réelles, nous avons des exigences concernant l'ordre d'exécution des threads, ce qui nécessite l'utilisation de threads. communication.
Pourquoi la communication des threads n'utilise-t-elle pas la priorité pour résoudre l'ordre de passage des threads ?
La priorité globale est déterminée par les informations de priorité dans la carte PCB du thread et le temps d'attente du thread, donc dans le développement général, la priorité n'est pas utilisée pour indiquer l'ordre d'exécution des threads
Regardez la scène suivante : boulangerie Exemple pour décrire le modèle producteur-consommateur
Il y a une boulangerie avec des boulangers et des clients, correspondant à nos producteurs et consommateurs, et la boulangerie a un inventaire pour stocker le pain Quand l'inventaire est plein, elle ne produira plus En même temps, des consommateurs. achètent également du pain. Lorsque le stock de pain est épuisé, les consommateurs doivent attendre que du nouveau pain soit produit avant de pouvoir continuer à acheter. Analyse : Quand arrêter la production et quand arrêter la consommation, la communication par fil doit être appliquée avec précision. déterminer Transmettre les informations de production et de consommation
2. Méthodes d'attente et de notification
wait() : Laissez l'objet se verrouiller détenu par la version de thread actuelle et attendez
wait(long timeout) : Le paramètre correspondant est le temps pendant lequel le thread attend
notify( ) : réveille le thread qui utilise le même objet pour appeler wait pour entrer dans le thread en attente et rivalise à nouveau pour le verrouillage de l'objet
notifyAll() : s'il y a plusieurs threads en attente, notifyAll réveille tous, notifie le réveil en créer un au hasard
Remarque :Ces plusieurs méthodes sont toutes des méthodes de la classe Object
Doivent être utilisées dans les blocs de code de synchronisation synchronisés/méthodes de synchronisation
Quel objet est verrouillé, c'est l'objet qui attend et est averti
Après l'appel notifier, il ne se réveille pas immédiatement, mais attend synchronisé Après la fin, réveillez-vous
1. Méthode wait()
Après avoir appelé la méthode wait :Faites attendre le thread exécutant le code actuel (le thread est placé dans la file d'attente)
Libérer le verrou actuel
Remplir certaines conditions Au réveil, essayez à nouveau d'acquérir le verrou
Conditions d'attente pour terminer :D'autres threads appellent la méthode notify de l'objet
wait wait time expire (paramètre timeout pour spécifier le temps d'attente)
D'autres threads appellent la méthode interrompue, ce qui provoque l'attente pour lancer InterruptedException
2 méthode notify()
Lorsque vous utilisez la méthode wait sans paramètres, vous devez utiliser. la méthode notify pour réveiller le thread en attente.
Cette méthode consiste à réveiller les threads en attente du verrouillage de l'objet, leur permettant de réacquérir le verrouillage de l'objet
S'il y a plusieurs threads en attente, le thread. le planificateur sélectionnera au hasard un thread en état d'attente (il n'y a pas de premier arrivé, premier servi)
Après la méthode notify(), le thread actuel ne sera pas libéré immédiatement. Le verrouillage de l'objet ne sera pas libéré. libéré jusqu'à ce que le thread exécutant la méthode notify() ait fini d'exécuter le programme, c'est-à-dire après avoir quitté le bloc de code synchronisé
3 Méthode notifyAll()
Cette méthode et la méthode notify() La fonction est la même, sauf que lorsque au réveil, tous les threads en attente sont réveillés
La méthode notify() réveille juste un thread au hasard
3 Utilisez wait et notify pour mettre en œuvre une activité de boulangerie
Instructions préalables :Il y a 2 Baker, le boulanger peut faire. deux pains à la fois
L'entrepôt peut stocker 100 pains
Il y a 10 consommateurs, chaque consommateur achète un pain à la fois
Remarque :La consommation et la production sont concurrentes en même temps Elle est réalisée en parallèle, pas une production et une consommation
Code d'implémentation :public class Bakery {
private static int total;//库存
public static void main(String[] args) {
Producer producer = new Producer();
for(int i = 0;i < 2;i++){
new Thread(producer,"面包师傅-"+(i-1)).start();
}
Consumer consumer = new Consumer();
for(int i = 0;i < 10;i++){
new Thread(consumer,"消费者-"+(i-1)).start();
}
}
private static class Producer implements Runnable{
private int num = 3; //生产者每次生产三个面包
@Override
public void run() {
try {
while(true){ //一直生产
synchronized (Bakery.class){
while((total+num)>100){ //仓库满了,生产者等待
Bakery.class.wait();
}
//等待解除
total += num;
System.out.println(Thread.currentThread().getName()+"生产面包,库存:"+total);
Thread.sleep(500);
Bakery.class.notifyAll(); //唤醒生产
}
Thread.sleep(500);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
private static class Consumer implements Runnable{
private int num = 1; //消费者每次消费1个面包
@Override
public void run() {
try {
while(true){ //一直消费
synchronized (Bakery.class){
while((total-num)<0){ //仓库空了,消费者等待
Bakery.class.wait();
}
//解除消费者等待
total -= num;
System.out.println(Thread.currentThread().getName()+"消费面包,库存:"+total);
Thread.sleep(500);
Bakery.class.notifyAll(); //唤醒消费
}
Thread.sleep(500);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
Résultats d'impression partiels :
4. File d'attente de blocage
La file d'attente de blocage est une file d'attente spéciale qui suit également le "premier entré, premier sorti " principe En principe, il s'agit d'une structure de file d'attente thread-safe
Caractéristiques : Modèle typique producteur-consommateur, généralement utilisé pour découpler les tâches et éliminer les pics
Lorsque la file d'attente est pleine, la file d'attente sera bloquée et attendue (production) jusqu'à il y a D'autres threads prennent des éléments de la file d'attente
Lorsque la file d'attente est vide, la file d'attente est bloquée et attendue (consommée) jusqu'à ce que d'autres threads insèrent des éléments dans la file d'attente
1 Modèle producteur-consommateur
Le modèle producteur-consommateur est transmis. Un conteneur pour résoudre le problème de couplage fort entre producteurs et consommateurs
Les producteurs et les consommateurs ne communiquent pas directement entre eux, mais communiquent via des files d'attente bloquantes, donc après que le producteur a produit les données, il attend que le consommateur les traite et les lance directement dans la file d'attente de blocage, le consommateur ne demande pas de données au producteur, mais les extrait directement de la file d'attente de blocage. La file d'attente de blocage est équivalente à un tampon, équilibrant les capacités de traitement du producteur et de la file d'attente de blocage. peut également permettre aux producteurs et au découplage entre les consommateurs
La mise en œuvre de l'activité de boulangerie mentionnée ci-dessus est un exemple du modèle producteur-consommateur
2 File d'attente de blocage dans la bibliothèque standard
BlockingQueue 是一个接口. 真正实现的类是 LinkedBlockingQueue
put 方法用于阻塞式的入队列, take 用于阻塞式的出队列
BlockingQueue 也有 offer, poll, peek 等方法, 但是这些方法不带有阻塞特性
BlockingDeque<String> queue = new LinkedBlockingDeque<>(); queue.put("hello"); //如果队列为空,直接出出队列就会阻塞 String ret = queue.take(); System.out.println(ret);
3. 阻塞队列的模拟实现
这里使用数组实现一个循环队列来模拟阻塞队列
当队列为空的时候,就不能取元素了,就进入wait等待,当有元素存放时,唤醒
当队列为满的时候,就不能存元素了,就进入wait等待,当铀元素取出时,唤醒
实现代码:
public class MyBlockingQueue { //使用数组实现一个循环队列,队列里面存放的是线程要执行的任务 private Runnable[] tasks; //队列中任务的数量,根据数量来判断是否可以存取 private int count; private int putIndex; //存放任务位置 private int takeIndex; //取出任务位置 //有参的构造方法,表示队列容量 public MyBlockingQueue(int size){ tasks = new Runnable[size]; } //存任务 public void put(Runnable task){ try { synchronized (MyBlockingQueue.class){ //如果队列容量满了,则存任务等待 while(count == tasks.length){ MyBlockingQueue.class.wait(); } tasks[putIndex] = task; //将任务放入数组 putIndex = (putIndex+1) % tasks.length; //更新存任务位置 count++; //更新存放数量 MyBlockingQueue.class.notifyAll(); //唤醒存任务 } } catch (InterruptedException e) { e.printStackTrace(); } } //取任务 public Runnable take(){ try { synchronized (MyBlockingQueue.class){ //如果队列任务为空,则取任务等待 while(count==0){ MyBlockingQueue.class.wait(); } //取任务 Runnable task = tasks[takeIndex]; takeIndex = (takeIndex+1) % tasks.length; //更新取任务位置 count--; //更新存放数量 MyBlockingQueue.class.notifyAll(); //唤醒取任务 return task; } } catch (InterruptedException e) { throw new RuntimeException("存放任务出错",e); } } }
五. wait和sleep的区别(面试题)
相同点:
都可以让线程放弃执行一段时间
不同点:
☘️wait用于线程通信,让线程在等待队列中等待
☘️sleep让线程阻塞一段时间,阻塞在阻塞队列中
☘️wait需要搭配synchronized使用,sleep不用搭配
☘️wait是Object类的方法,sleep是Thread的静态方法
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于结构化数据处理开源库SPL的相关问题,下面就一起来看一下java下理想的结构化数据处理类库,希望对大家有帮助。

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于PriorityQueue优先级队列的相关知识,Java集合框架中提供了PriorityQueue和PriorityBlockingQueue两种类型的优先级队列,PriorityQueue是线程不安全的,PriorityBlockingQueue是线程安全的,下面一起来看一下,希望对大家有帮助。

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于java锁的相关问题,包括了独占锁、悲观锁、乐观锁、共享锁等等内容,下面一起来看一下,希望对大家有帮助。

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于多线程的相关问题,包括了线程安装、线程加锁与线程不安全的原因、线程安全的标准类等等内容,希望对大家有帮助。

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于枚举的相关问题,包括了枚举的基本操作、集合类对枚举的支持等等内容,下面一起来看一下,希望对大家有帮助。

本篇文章给大家带来了关于Java的相关知识,其中主要介绍了关于关键字中this和super的相关问题,以及他们的一些区别,下面一起来看一下,希望对大家有帮助。

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于平衡二叉树(AVL树)的相关知识,AVL树本质上是带了平衡功能的二叉查找树,下面一起来看一下,希望对大家有帮助。

本篇文章给大家带来了关于Java的相关知识,其中主要整理了Stream流的概念和使用的相关问题,包括了Stream流的概念、Stream流的获取、Stream流的常用方法等等内容,下面一起来看一下,希望对大家有帮助。


Outils d'IA chauds

Undresser.AI Undress
Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover
Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool
Images de déshabillage gratuites

Clothoff.io
Dissolvant de vêtements AI

AI Hentai Generator
Générez AI Hentai gratuitement.

Article chaud

Outils chauds

SublimeText3 version chinoise
Version chinoise, très simple à utiliser

MinGW - GNU minimaliste pour Windows
Ce projet est en cours de migration vers osdn.net/projects/mingw, vous pouvez continuer à nous suivre là-bas. MinGW : un port Windows natif de GNU Compiler Collection (GCC), des bibliothèques d'importation et des fichiers d'en-tête librement distribuables pour la création d'applications Windows natives ; inclut des extensions du runtime MSVC pour prendre en charge la fonctionnalité C99. Tous les logiciels MinGW peuvent fonctionner sur les plates-formes Windows 64 bits.

Télécharger la version Mac de l'éditeur Atom
L'éditeur open source le plus populaire

Bloc-notes++7.3.1
Éditeur de code facile à utiliser et gratuit

mPDF
mPDF est une bibliothèque PHP qui peut générer des fichiers PDF à partir de HTML encodé en UTF-8. L'auteur original, Ian Back, a écrit mPDF pour générer des fichiers PDF « à la volée » depuis son site Web et gérer différentes langues. Il est plus lent et produit des fichiers plus volumineux lors de l'utilisation de polices Unicode que les scripts originaux comme HTML2FPDF, mais prend en charge les styles CSS, etc. et présente de nombreuses améliorations. Prend en charge presque toutes les langues, y compris RTL (arabe et hébreu) et CJK (chinois, japonais et coréen). Prend en charge les éléments imbriqués au niveau du bloc (tels que P, DIV),