독자들도 인터넷에서 ThreadLocal에 대한 많은 정보를 읽었을 것이라고 생각합니다. 많은 블로그에서는 다음과 같이 말합니다. ThreadLocal은 다중 스레드 프로그램의 동시성 문제를 해결하기 위한 새로운 아이디어를 제공합니다. -스레드 액세스 리소스 공유 문제. 그렇게 생각하신다면 ThreadLocal에 대한 이전의 오해를 바로잡을 수 있도록 10초의 시간을 주십시오!
JDK의 소스 코드가 어떻게 작성되었는지 살펴보세요:
This class provides thread-local variables. These variables differ from their normal counterparts in that each thread that accesses one (via its {@code get} or {@code set} method) has its own, independently initialized copy of the variable. {@code ThreadLocal} instances are typically private static fields in classes that wish to associate state with a thread (e.g., a user ID or Transaction ID).
번역은 아마도 다음과 같습니다(영어가 좋지 않습니다. 더 나은 번역이 있으면 메시지를 남겨주세요):
ThreadLocal 클래스는 다음과 같이 사용됩니다. 스레드 내부에 지역 변수를 제공하십시오. 이런 종류의 변수가 다중 스레드 환경(get 또는 set 메소드를 통해 액세스됨)에서 액세스되면 각 스레드의 변수가 다른 스레드의 변수와 상대적으로 독립적이라는 것을 확인할 수 있습니다. ThreadLocal 인스턴스는 일반적으로 전용 정적 유형이며 스레드와 스레드 컨텍스트를 연결하는 데 사용됩니다.
한 문장으로 요약할 수 있습니다. ThreadLocal의 기능은 스레드 내에서 로컬 변수를 제공하는 것입니다. 이러한 변수는 스레드의 수명 주기 내에서 작동하여 동일한 여러 함수 또는 구성 요소 간에 일부 공용 변수를 전달하는 복잡성을 줄입니다. 스레드.
예를 들어 외출할 때 버스를 먼저 타고 지하철을 타야 하는데 여기서 지하철을 타는 것은 같은 스레드의 두 가지 기능과 같습니다. 동일한 기능이 필요합니다. 버스 카드(베이징 버스와 지하철은 버스 카드를 사용함), 버스 카드 변수를 두 기능 모두에 전달하지 않기 위해(항상 버스 카드를 도로에 가지고 다니지 않는 것과 동일) 다음과 같이 할 수 있습니다. : 버스카드 변경 대행업체에 미리 건네주고, 카드 긁어야 할 땐 해당 대행업체에 버스카드 달라고 하지요(물론 매번 같은 버스카드를 받습니다). 이런 식으로 나(동일 스레드)가 버스 카드가 필요한 한 언제 어디서나 이 기관에 요청할 수 있습니다.
누군가는 버스카드를 전역변수로 설정하면 언제 어디서나 버스카드를 얻을 수 있다고 하더군요. 하지만 개인(스레드)이 많으면 어떻게 될까요? 모두가 같은 버스카드를 사용할 수는 없습니다(버스카드는 실명으로 인증된다고 가정합니다). 그렇지 않으면 혼란스러울 것입니다. 이제 이해가 되셨나요? 이것이 ThreadLocal 설계의 원래 의도입니다. 스레드 내에서 로컬 변수를 제공하는 것입니다. 이 변수는 다른 스레드를 격리하면서 이 스레드 내에서 언제 어디서나 액세스할 수 있습니다.
(1) ThreadContext8742468051c85b06f0a0af9e3e506b5c는 키/값 쌍을 기반으로 현재 스레드에 대한 개체를 바인딩 및 바인딩 해제하는 방법을 제공합니다.
이 클래스는 스레드 지역 변수를 제공합니다. 이러한 변수는 get 또는 set 메서드를 통해 스레드에 액세스하는 각 스레드가 독립적으로 초기화된 변수 복사본을 가지고 있다는 점에서 일반 변수와 다릅니다.
ThreadLocal 인스턴스는 일반적으로 상태를 스레드와 연결하려는 클래스의 비공개 정적 필드입니다(예: 사용자 ID 또는 트랜잭션 ID). 각 스레드에는 스레드 로컬 변수의 복사본인
에 대한 암시적 참조가 있습니다. 스레드가 살아있는 한 ThreadLocal 인스턴스는 스레드 하나가 종료된 후에 액세스할 수 있습니다. 로컬 인스턴스는 가비지 수집됩니다(다른 참조가 존재하지 않는 한).
8742468051c85b06f0a0af9e3e506b5c는 스레드에 저장된 개체입니다. 즉, 클래스 T는 스레드의 클래스 속성입니다.
일반적으로 사용되는 방법은 다음과 같습니다.
1 public class ThreadLocal8742468051c85b06f0a0af9e3e506b5c { 2 3 //设置属性 4 5 public void set(T value) { 6 Thread t = Thread.currentThread(); 7 ThreadLocalMap map = getMap(t); 8 if (map != null) 9 map.set(this, value);10 else11 createMap(t, value);12 }13 14 //获取属性15 16 public T get() {17 Thread t = Thread.currentThread();18 ThreadLocalMap map = getMap(t);19 if (map != null) {20 ThreadLocalMap.Entry e = map.getEntry(this);21 if (e != null)22 return (T)e.value;23 }24 return setInitialValue();25 }26 27 //获取线程的 ThreadLocal.ThreadLocalMap28 29 ThreadLocalMap getMap(Thread t) {30 return t.threadLocals;31 }32 33 }34 35 //新建一个线程本地的localMap36 37 void createMap(Thread t, T firstValue) {38 t.threadLocals = new ThreadLocalMap(this, firstValue);39 }
(2) 사용 예: 연결 및 세션은 다음과 같습니다.
1 private static ThreadLocal2918aec6680a0636f336c3e736b1ff5a connectionHolder 2 = new ThreadLocal2918aec6680a0636f336c3e736b1ff5a() { 3 public Connection initialValue() { 4 return DriverManager.getConnection(DB_URL); 5 } 6 }; 7 8 public static Connection getConnection() { 9 return connectionHolder.get();10 }
1 private static final ThreadLocal threadSession = new ThreadLocal(); 2 3 public static Session getSession() throws InfrastructureException { 4 Session s = (Session) threadSession.get(); 5 try { 6 if (s == null) { 7 s = getSessionFactory().openSession(); 8 threadSession.set(s); 9 }10 } catch (HibernateException ex) {11 throw new InfrastructureException(ex);12 }13 return s;14 }
위 내용은 ThreadLocal은 다중 스레드 프로그램의 예를 해결합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!