Java 다중 스레드 프로그래밍에서 LockSupport 클래스의 스레드 차단 사용에 대한 자세한 설명
LockSupport는 잠금 및 기타 동기화 클래스를 생성하는 데 사용되는 기본 스레드 차단 기본 요소입니다.
LockSupport의 park() 및 unpark() 기능은 각각 스레드를 차단하고 스레드 차단을 해제하는 것이며, park() 및 unpark()에서는 "Thread.suspens 및 Thread.resume"으로 인해 발생할 수 있는 교착 상태가 발생하지 않습니다. "질문.
park() 및 unpark()에는 권한이 있으므로 park()를 호출하는 스레드와 unpark()를 시도하는 다른 스레드 간의 경쟁은 활성 상태로 유지됩니다.
기본 사용법
LockSupport는 바이너리 세마포어와 매우 유사합니다(라이센스가 1개만 사용 가능). 라이센스가 사용되지 않은 경우 현재 스레드가 라이센스를 획득하고 라이센스가 있으면 계속 실행됩니다. 점유된 경우 현재 스레드가 차단되어 권한을 얻기를 기다리고 있습니다.
public static void main(String[] args) { LockSupport.park(); System.out.println("block."); }
이 코드를 실행하면 메인 스레드가 항상 차단되어 있는 것을 확인할 수 있습니다. 라이센스는 기본적으로 점유되어 있기 때문에 park() 호출시 라이센스를 획득할 수 없어 차단 상태에 들어간다.
다음 코드: 먼저 권한을 해제한 후 권한을 얻으면 메인 스레드가 정상적으로 종료될 수 있습니다. LockSupport 권한 획득과 해제는 일반적으로 동일합니다. 여러 번 unpark하면 하나의 park만 사용해도 문제가 없습니다.
public static void main(String[] args) { Thread thread = Thread.currentThread(); LockSupport.unpark(thread);//释放许可 LockSupport.park();// 获取许可 System.out.println("b"); }
LockSupport는 재진입이 불가능합니다. 스레드가 LockSupport .park()를 연속으로 두 번 호출하면 스레드는 확실히 차단된 상태로 유지됩니다.
public static void main(String[] args) throws Exception { Thread thread = Thread.currentThread(); LockSupport.unpark(thread); System.out.println("a"); LockSupport.park(); System.out.println("b"); LockSupport.park(); System.out.println("c"); }
이 코드는 a와 b를 인쇄하지만 c는 인쇄하지 않습니다. 왜냐하면 park가 두 번째 호출될 때 스레드는 권한을 얻을 수 없고 잠금.
인터럽트에 대한 LockSupport의 응답성을 살펴보겠습니다.
public static void t2() throws Exception { Thread t = new Thread(new Runnable() { private int count = 0; @Override public void run() { long start = System.currentTimeMillis(); long end = 0; while ((end - start) <= 1000) { count++; end = System.currentTimeMillis(); } System.out.println("after 1 second.count=" + count); //等待或许许可 LockSupport.park(); System.out.println("thread over." + Thread.currentThread().isInterrupted()); } }); t.start(); Thread.sleep(2000); // 中断线程 t.interrupt(); System.out.println("main over"); }
마지막 스레드는 thread over.true를 인쇄합니다. 이는 스레드가 park를 호출하여 차단된 경우 인터럽트 요청에 응답할 수 있지만(인터럽트 상태는 true로 설정됨) InterruptedException을 발생시키지 않음을 보여줍니다.
LockSupport 기능 목록
// 返回提供给最近一次尚未解除阻塞的 park 方法调用的 blocker 对象,如果该调用不受阻塞,则返回 null。 static Object getBlocker(Thread t) // 为了线程调度,禁用当前线程,除非许可可用。 static void park() // 为了线程调度,在许可可用之前禁用当前线程。 static void park(Object blocker) // 为了线程调度禁用当前线程,最多等待指定的等待时间,除非许可可用。 static void parkNanos(long nanos) // 为了线程调度,在许可可用前禁用当前线程,并最多等待指定的等待时间。 static void parkNanos(Object blocker, long nanos) // 为了线程调度,在指定的时限前禁用当前线程,除非许可可用。 static void parkUntil(long deadline) // 为了线程调度,在指定的时限前禁用当前线程,除非许可可用。 static void parkUntil(Object blocker, long deadline) // 如果给定线程的许可尚不可用,则使其可用。 static void unpark(Thread thread)
LockSupport 예시
아래 "예제 1"과 "예제 2"를 비교하세요 LockSupport의 사용법을 더 명확하게 이해할 수 있습니다.
예시 1
public class WaitTest1 { public static void main(String[] args) { ThreadA ta = new ThreadA("ta"); synchronized(ta) { // 通过synchronized(ta)获取“对象ta的同步锁” try { System.out.println(Thread.currentThread().getName()+" start ta"); ta.start(); System.out.println(Thread.currentThread().getName()+" block"); // 主线程等待 ta.wait(); System.out.println(Thread.currentThread().getName()+" continue"); } catch (InterruptedException e) { e.printStackTrace(); } } } static class ThreadA extends Thread{ public ThreadA(String name) { super(name); } public void run() { synchronized (this) { // 通过synchronized(this)获取“当前对象的同步锁” System.out.println(Thread.currentThread().getName()+" wakup others"); notify(); // 唤醒“当前对象上的等待线程” } } } }
예시 2
import java.util.concurrent.locks.LockSupport; public class LockSupportTest1 { private static Thread mainThread; public static void main(String[] args) { ThreadA ta = new ThreadA("ta"); // 获取主线程 mainThread = Thread.currentThread(); System.out.println(Thread.currentThread().getName()+" start ta"); ta.start(); System.out.println(Thread.currentThread().getName()+" block"); // 主线程阻塞 LockSupport.park(mainThread); System.out.println(Thread.currentThread().getName()+" continue"); } static class ThreadA extends Thread{ public ThreadA(String name) { super(name); } public void run() { System.out.println(Thread.currentThread().getName()+" wakup others"); // 唤醒“主线程” LockSupport.unpark(mainThread); } } }
실행 결과:
main start ta main block ta wakup others main continue
설명: 공원과 대기의 차이. wait가 스레드를 차단하기 전에 동기화를 통해 동기화 잠금을 획득해야 합니다.
Java 멀티 스레드 프로그래밍에서 LockSupport 클래스의 스레드 차단 사용법에 대한 자세한 설명은 PHP 중국어 웹사이트를 참고하세요!

핫 AI 도구

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

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

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











일부 애플리케이션이 제대로 작동하지 않는 회사의 보안 소프트웨어에 대한 문제 해결 및 솔루션. 많은 회사들이 내부 네트워크 보안을 보장하기 위해 보안 소프트웨어를 배포 할 것입니다. ...

데이터베이스 작업에 MyBatis-Plus 또는 기타 ORM 프레임 워크를 사용하는 경우 엔티티 클래스의 속성 이름을 기반으로 쿼리 조건을 구성해야합니다. 매번 수동으로 ...

시스템 도킹의 필드 매핑 처리 시스템 도킹을 수행 할 때 어려운 문제가 발생합니다. 시스템의 인터페이스 필드를 효과적으로 매핑하는 방법 ...

IntellijideAultimate 버전을 사용하여 봄을 시작하십시오 ...

많은 응용 프로그램 시나리오에서 정렬을 구현하기 위해 이름으로 이름을 변환하는 솔루션, 사용자는 그룹으로, 특히 하나로 분류해야 할 수도 있습니다.

Java 객체 및 배열의 변환 : 캐스트 유형 변환의 위험과 올바른 방법에 대한 심층적 인 논의 많은 Java 초보자가 객체를 배열로 변환 할 것입니다 ...

데이터베이스 쿼리에 tkmyBatis를 사용하는 경우 쿼리 조건을 구축하기 위해 엔티티 클래스 변수 이름을 우아하게 가져 오는 방법이 일반적인 문제입니다. 이 기사는 고정 될 것입니다 ...

전자 상거래 플랫폼에서 SKU 및 SPU 테이블의 디자인에 대한 자세한 설명이 기사는 전자 상거래 플랫폼에서 SKU 및 SPU의 데이터베이스 설계 문제, 특히 사용자 정의 판매를 처리하는 방법에 대해 논의 할 것입니다 ...
