찾다
Javajava지도 시간Java LinkedBlockingQueue를 마스터하는 방법

Java LinkedBlockingQueue를 마스터하는 방법

Apr 18, 2023 pm 04:04 PM
javalinkedblockingqueue

    병원에서 돈을 지불하기 위해 줄을 서야 하고, 핵산 검사를 하기 위해 줄을 서야 하고, 자동차 신호등에도 줄을 서야 합니다.

    Java LinkedBlockingQueue를 마스터하는 방법

    큐는 선착순 큐가 앞에 배치되고, 두번째 선착순 큐가 뒤에 배치되는 데이터 구조입니다. 배열과 연결리스트를 사용해 구현했습니다. 일반적으로 작업 실행 및 데이터 교환을 조정하는 데 사용됩니다.

    Introduction

    LinkedBlockingQueue는 선택적인 제한된 차단 대기열입니다. Bounded는 대기열에 최대 용량이 있다는 것을 의미합니다. 즉, 대기열이 가득 차고 계속해서 대기열에 요소를 추가하려는 경우 이 작업이 일시 중지되고 대기열에 공간이 생길 때까지 추가 작업은 계속되지 않습니다. 큐가 이미 비어 있고 큐에서 요소를 가져오려는 경우 큐에 요소가 있을 때까지 작업이 일시 중지되어 획득 작업을 계속합니다.

    구현 원칙

    LinkedBlockingQueue는 내부적으로 연결 목록을 요소의 저장 구조로 사용합니다. 저장 작업과 검색 작업을 위해 각각 내부적으로 두 개의 잠금이 사용됩니다.

    액세스 작업을 수행할 때 LinkedBlockingQueue가 스레드로부터 안전한지 확인하기 위해 액세스 작업을 수행하기 전에 먼저 잠금을 획득해야 합니다.

    LinkedBlockingQueue는 두 개의 조건 조건 대기열(notFull 조건 하나와 notEmpty 조건 하나)을 전달합니다. 큐에 요소를 삽입할 때 현재 큐가 꽉 찼다고 판단되면 다른 스레드가 해당 스레드에 큐에 요소를 계속 삽입할 수 있음을 알릴 때까지 notFull 조건을 통해 해당 스레드를 차단합니다. 큐에서 요소를 제거할 때 현재 큐가 비어 있다고 판단되면 다른 스레드가 이 스레드를 통해 요소를 계속 얻을 수 있을 때까지 해당 스레드는 notEmpty 조건을 통해 차단됩니다.

    이렇게 하면 스레드의 액세스 작업에 오류가 발생하지 않습니다. 큐가 가득 찼을 때 삽입된 요소를 삭제하지 마세요. 또한 큐가 비어 있을 때 null 값을 가져오는 것도 방지하세요.

    Constructor

    public LinkedBlockingQueue() {
        this(Integer.MAX_VALUE);
    }
    
    public LinkedBlockingQueue(int capacity) {
        if (capacity <= 0) throw new IllegalArgumentException();
        this.capacity = capacity;
        last = head = new Node<E>(null);
    }

    매개변수 없는 생성자에서는 Integer.MAX_VALUE가 기본적으로 대기열의 최대 용량으로 사용됩니다.

    매개변수화된 생성자에서 대기열의 최대 용량을 직접 지정하고 헤드 노드와 테일 노드를 생성할 수 있습니다. 그런 다음 LinkedBlockingQueue는 단방향 연결 목록을 사용합니다.

    private final int capacity;
    
    /** Current number of elements */
    private final AtomicInteger count = new AtomicInteger();
    
    transient Node<E> head;
    
    private transient Node<E> last;
    
    // 取锁
    private final ReentrantLock takeLock = new ReentrantLock();
    
    private final Condition notEmpty = takeLock.newCondition();
    
    // 存锁
    private final ReentrantLock putLock = new ReentrantLock();
    
    private final Condition notFull = putLock.newCondition();

    그리고 객체가 초기화되면 저장 작업과 검색 작업에 각각 사용되는 두 개의 잠금이 생성됩니다. 비어 있는 대기열 조건과 꽉 찬 대기열 조건에 대해 각각 두 개의 조건부 대기열이 생성됩니다.

    Insert 함수

    public void put(E e) throws InterruptedException {
        if (e == null) throw new NullPointerException();
        final int c;
        final Node<E> node = new Node<E>(e);
        final ReentrantLock putLock = this.putLock;
        final AtomicInteger count = this.count;
        putLock.lockInterruptibly();
        try {
            while (count.get() == capacity) {
                notFull.await();
            }
            enqueue(node);
            c = count.getAndIncrement();
            if (c + 1 < capacity)
                notFull.signal();
        } finally {
            putLock.unlock();
        }
        if (c == 0)
            signalNotEmpty();
    }

    1. 잠금 획득

    2. 현재 대기열이 가득 찼는지 확인

    • 대기열이 가득 차면 notFull 조건부 대기열의 wait() 메서드를 호출하여 스레드를 차단하고 일시 중지합니다. 스레드 삽입 작업. 내부 오버플로 문제를 피하세요.

    • 가득 차 있지 않으면 enqueue 함수 enqueue를 직접 호출하여 대기열 끝에 삽입합니다.

    3. 이때 큐가 꽉 찼는지 확인하세요.

    꽉 차지 않았다면 notFull 조건 큐의 signal() 메서드를 호출하여 notFull 조건 큐에 차단된 스레드를 깨웁니다.

    4. 잠금 해제

    • 요소를 삽입하기 전의 대기열 요소 수가 0

    • 인지 확인한 다음 notEmpty 조건부 대기열의 signal() 메서드를 호출하여 이제 대기열은 비어 있지 않으며 차단 스레드를 깨울 수 있습니다. 요소를 가져옵니다.

    Java LinkedBlockingQueue를 마스터하는 방법

    왜 notFull 조건 대기열의 signal() 메서드를 호출해야 합니까? 대기열 검색 작업과 저장 작업에 사용되는 잠금이 다르기 때문에 한 스레드가 입금 작업을 수행하면 다른 스레드가 검색 작업을 수행할 수 있음을 의미합니다. 다음 예를 살펴보겠습니다.

    Java LinkedBlockingQueue를 마스터하는 방법

    • 큐의 총 용량은 5이고 현재 요소 수는 5입니다. 스레드 A는 잠금을 획득하고 요소를 삽입하려고 합니다. 하지만 대기열 용량이 가득 차서 잠금이 해제되고 조건 대기열에 추가되어 깨어나기를 기다립니다.

    • 스레드 B가 잠금을 획득하고 요소를 삽입하려고 합니다. 하지만 대기열 용량이 가득 차서 잠금이 해제되고 조건 대기열에 추가되어 깨어나기를 기다립니다.

    • 스레드 C가 잠금을 획득하고 요소 1을 꺼냈습니다. 그리고 notFull이라는 신호 방식을 통해 조건 대기열에 있는 차단된 스레드 A를 깨웁니다. 스레드 A는 깨어난 후 동기화 대기열에 참가하지만 현재로서는 아직 잠금을 위해 경쟁하지 않았습니다.

    • 스레드 D가 잠금 장치를 획득하고 요소 2를 꺼냈습니다. 그러나 차단된 스레드를 깨우는 코드는 아직 실행되지 않았습니다.

    • 스레드 A는 잠금을 놓고 경쟁하고 요소 삽입을 시작합니다. 요소를 삽입한 후 대기열 요소의 개수가 대기열의 전체 용량보다 적은 4개인지 확인하므로 notFull 의 시그널 메서드를 실행하여 조건부 대기열에 차단된 스레드 B를 깨웁니다. 스레드 B가 깨어난 후 동기화 대기열에 합류하고 잠금 경쟁을 시작합니다.

    • 스레드 B가 잠금을 놓고 경쟁하고 요소 삽입을 시작합니다. 요소를 삽입한 후 대기열 요소의 개수가 5인지 확인하고 wake-up 작업을 수행하지 않습니다.

    这样做的目的是尽快的唤醒阻塞线程,可以更快的完成插入元素操作。因为线程存和取的操作相互之间并不是互斥的,而是独立运行的,提高吞吐量。

    获取函数

    public E take() throws InterruptedException {
        final E x;
        final int c;
        final AtomicInteger count = this.count;
        final ReentrantLock takeLock = this.takeLock;
        takeLock.lockInterruptibly();
        try {
            while (count.get() == 0) {
                notEmpty.await();
            }
            x = dequeue();
            c = count.getAndDecrement();
            if (c > 1)
                notEmpty.signal();
        } finally {
            takeLock.unlock();
        }
        if (c == capacity)
            signalNotFull();
        return x;
    }

    1.获得取锁

    2.判断当前队列是否为空

    • 如果队列没有元素,调用 notEmpty 条件队列的 await() 方法,将该线程阻塞,暂停该线程的获取操作。避免获取元素出错。

    • 如果不为空,则直接调用出队函数 dequeue 移除队列第一个元素,并返回给客户端。

    3.检查此时队列是否为空

    如果不为空,则调用 notEmpty 条件队列的 signal() 方法,唤醒被阻塞在 notEmpty 条件队列的线程。

    4.释放锁

    5.检查获取元素前的队列元素数量是否等于最大容量

    等于最大容量,因为此时已经取出一个元素,因此队列处于未满的状态,可以唤醒阻塞在 notFull 条件的线程,让线程继续插入元素。

    Java LinkedBlockingQueue를 마스터하는 방법

    步骤 3 的目的是尽快的唤醒阻塞线程,可以更快的完成取元素操作。提高吞吐量。可以尝试自己画出流程图。

    入队函数

    private void enqueue(Node<E> node) {
        last = last.next = node;
    }

    入队函数极其简单,只要将最后一个元素的 next 指针指向当前元素即完成了插入操作。

    Java LinkedBlockingQueue를 마스터하는 방법

    出队函数

    private E dequeue() {
        // assert takeLock.isHeldByCurrentThread();
        // assert head.item == null;
        Node<E> h = head;
        Node<E> first = h.next;
        h.next = h; // help GC
        head = first;
        E x = first.item;
        first.item = null;
        return x;
    }

    我们前面说 LinkedBlockingQueue 使用的是有头链表。头节点只是作为一个标志,实际上并不是一个真正的元素。当获取元素时,将头节点的下一个节点作为头节点,将原来的头节点取消引用,被垃圾回收即可。

    Java LinkedBlockingQueue를 마스터하는 방법

    应用场景

    适用场景

    LinkedBlockingQueue 和 ArrayBlockingQueue 一样适用于多个线程之间需要共享数据、协调任务执行的场景。因此可以总结出以下几个应用场景:

    线程池:线程池是一个常见的并发编程模型,它通过线程池中的线程执行任务。并且可以重复使用这些线程。在线程池中,可以使用 LinkedBlockingQueue 来存储需要执行的任务,以此控制任务数量和执行顺序。当线程池中的线程执行完任务之后,可以从 LinkedBlockingQueue 中取出下一个任务执行。

    生产者-消费者:在生产者-消费者模型中,生产者负责生产数据,消费者负责对数据进行处理。在这种模式下,LinkedBlockingQueue 可以作为生产者与消费者之间的数据通道,保证线程安全和数据正确。

    实际应用场景

    • Nacos: Nacos 是一个动态服务发现、配置和服务管理平台,它使用 LinkedBlockingQueue 来实现内部的任务队列。

    • Tomcat:从 Tomcat 7 开始,请求队列默认使用了 LinkedBlockingQueue 实现。

    • Hystrix: 一个流行的容错框架,其默认使用 LinkedBlockingQueue 作为请求队列。

    위 내용은 Java LinkedBlockingQueue를 마스터하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

    성명
    이 기사는 亿速云에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제
    고급 Java 프로젝트 관리, 구축 자동화 및 종속성 해상도에 Maven 또는 Gradle을 어떻게 사용합니까?고급 Java 프로젝트 관리, 구축 자동화 및 종속성 해상도에 Maven 또는 Gradle을 어떻게 사용합니까?Mar 17, 2025 pm 05:46 PM

    이 기사에서는 Java 프로젝트 관리, 구축 자동화 및 종속성 해상도에 Maven 및 Gradle을 사용하여 접근 방식과 최적화 전략을 비교합니다.

    적절한 버전 및 종속성 관리로 Custom Java 라이브러리 (JAR Files)를 작성하고 사용하려면 어떻게해야합니까?적절한 버전 및 종속성 관리로 Custom Java 라이브러리 (JAR Files)를 작성하고 사용하려면 어떻게해야합니까?Mar 17, 2025 pm 05:45 PM

    이 기사에서는 Maven 및 Gradle과 같은 도구를 사용하여 적절한 버전 및 종속성 관리로 사용자 정의 Java 라이브러리 (JAR Files)를 작성하고 사용하는 것에 대해 설명합니다.

    카페인 또는 구아바 캐시와 같은 라이브러리를 사용하여 자바 애플리케이션에서 다단계 캐싱을 구현하려면 어떻게해야합니까?카페인 또는 구아바 캐시와 같은 라이브러리를 사용하여 자바 애플리케이션에서 다단계 캐싱을 구현하려면 어떻게해야합니까?Mar 17, 2025 pm 05:44 PM

    이 기사는 카페인 및 구아바 캐시를 사용하여 자바에서 다단계 캐싱을 구현하여 응용 프로그램 성능을 향상시키는 것에 대해 설명합니다. 구성 및 퇴거 정책 관리 Best Pra와 함께 설정, 통합 및 성능 이점을 다룹니다.

    캐싱 및 게으른 하중과 같은 고급 기능을 사용하여 객체 관계 매핑에 JPA (Java Persistence API)를 어떻게 사용하려면 어떻게해야합니까?캐싱 및 게으른 하중과 같은 고급 기능을 사용하여 객체 관계 매핑에 JPA (Java Persistence API)를 어떻게 사용하려면 어떻게해야합니까?Mar 17, 2025 pm 05:43 PM

    이 기사는 캐싱 및 게으른 하중과 같은 고급 기능을 사용하여 객체 관계 매핑에 JPA를 사용하는 것에 대해 설명합니다. 잠재적 인 함정을 강조하면서 성능을 최적화하기위한 설정, 엔티티 매핑 및 모범 사례를 다룹니다. [159 문자]

    Java의 클래스로드 메커니즘은 다른 클래스 로더 및 대표 모델을 포함하여 어떻게 작동합니까?Java의 클래스로드 메커니즘은 다른 클래스 로더 및 대표 모델을 포함하여 어떻게 작동합니까?Mar 17, 2025 pm 05:35 PM

    Java의 클래스 로딩에는 부트 스트랩, 확장 및 응용 프로그램 클래스 로더가있는 계층 적 시스템을 사용하여 클래스로드, 링크 및 초기화 클래스가 포함됩니다. 학부모 위임 모델은 핵심 클래스가 먼저로드되어 사용자 정의 클래스 LOA에 영향을 미치도록합니다.

    See all articles

    핫 AI 도구

    Undresser.AI Undress

    Undresser.AI Undress

    사실적인 누드 사진을 만들기 위한 AI 기반 앱

    AI Clothes Remover

    AI Clothes Remover

    사진에서 옷을 제거하는 온라인 AI 도구입니다.

    Undress AI Tool

    Undress AI Tool

    무료로 이미지를 벗다

    Clothoff.io

    Clothoff.io

    AI 옷 제거제

    AI Hentai Generator

    AI Hentai Generator

    AI Hentai를 무료로 생성하십시오.

    인기 기사

    R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
    3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O. 최고의 그래픽 설정
    3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
    3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
    WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
    4 몇 주 전By尊渡假赌尊渡假赌尊渡假赌

    뜨거운 도구

    SublimeText3 Linux 새 버전

    SublimeText3 Linux 새 버전

    SublimeText3 Linux 최신 버전

    Dreamweaver Mac版

    Dreamweaver Mac版

    시각적 웹 개발 도구

    스튜디오 13.0.1 보내기

    스튜디오 13.0.1 보내기

    강력한 PHP 통합 개발 환경

    mPDF

    mPDF

    mPDF는 UTF-8로 인코딩된 HTML에서 PDF 파일을 생성할 수 있는 PHP 라이브러리입니다. 원저자인 Ian Back은 자신의 웹 사이트에서 "즉시" PDF 파일을 출력하고 다양한 언어를 처리하기 위해 mPDF를 작성했습니다. HTML2FPDF와 같은 원본 스크립트보다 유니코드 글꼴을 사용할 때 속도가 느리고 더 큰 파일을 생성하지만 CSS 스타일 등을 지원하고 많은 개선 사항이 있습니다. RTL(아랍어, 히브리어), CJK(중국어, 일본어, 한국어)를 포함한 거의 모든 언어를 지원합니다. 중첩된 블록 수준 요소(예: P, DIV)를 지원합니다.

    VSCode Windows 64비트 다운로드

    VSCode Windows 64비트 다운로드

    Microsoft에서 출시한 강력한 무료 IDE 편집기