> Java > java지도 시간 > 변수에 1을 추가하는 두 개의 Java 스레드 분석 예

변수에 1을 추가하는 두 개의 Java 스레드 분석 예

WBOY
풀어 주다: 2023-04-28 21:52:19
앞으로
1173명이 탐색했습니다.

1--잘못된 관례적 글쓰기

<code>public static int i=0;</code><code>public static void add(){</code><code>    i=i+1;</code><code>    action();</code><code>}</code><code>public static void action(){</code><code>    System.out.println("==>"+Thread.currentThread().getName()+":"+i);</code><code>}</code><code>public static void main(String[] args) throws InterruptedException {</code><code>    Thread t1 = new Thread(SysUserServiceImpl::add,"t1");</code><code>    Thread t2= new Thread(SysUserServiceImpl::add,"t2");</code><code>    t1.start();</code><code>    t2.start();</code><code>}</code><code>运行结果==></code><code>==>t1:1</code><code>==>t2:2</code><code><br></code><code>==>t1:2</code><code>==>t2:1</code><code><br></code><code>==>t1:2</code><code>==>t2:2</code>
로그인 후 복사

결과가 매번 일치하지 않습니다. 멀티 스레드 환경에서 t1은 공유 메모리의 i에 대해 +1 작업을 수행하지만 이때 주 메모리에 값을 새로 고치지 않습니다. i가 0인지 아닌지에 대한 +1 연산이므로 최종 결과 i는 모두 1이 됩니다. 마찬가지로 처리 후 t1은 1이고 처리 후 t2는 2입니다. 여러 번 실행한 결과가 일관되지 않습니다.

개선방법 1 - 동기화 잠금

<code>public class ThreadException {</code><code>    public static volatile int i=0;</code><code>    public static void add(){</code><code>        synchronized (ThreadException.class){</code><code>            i=i+1;</code><code>            action();</code><code>        }</code><code>    }</code><code>    public static void action(){</code><code>        System.out.println("==>"+Thread.currentThread().getName()+":"+i);</code><code>    }</code><code>    public static void main(String[] args) throws InterruptedException {</code><code>        Thread t1 = new Thread(ThreadException::add,"t1");</code><code>        Thread t2= new Thread(ThreadException::add,"t2");</code><code>        t1.start();</code><code>        t2.start();</code><code><br></code><code>    }</code><code>}</code>
로그인 후 복사

장점: 간단한 구현

단점: 큰 잠금 세분성, 낮은 성능, 분산 환경, 다중 JVM 조건, 동기화 실패, 동기화는 로컬 잠금일 뿐이며 현재 jvm 아래의 개체만 잠깁니다. 분산 시나리오에서는 분산 잠금 사용

개선된 방법 2 AtomicInteger

public class ThreadException {    private static AtomicInteger num = new AtomicInteger(0);    public static void add(){        int i = num.getAndIncrement();        action(i);    }    public static void action(int i){        System.out.println("由"+i+"==>"+Thread.currentThread().getName()+":"+num);    }    public static void main(String[] args) throws InterruptedException {        Thread t1 = new Thread(ThreadException::add,"t1");        Thread t2= new Thread(ThreadException::add,"t2");        t1.start();        t2.start();    }}
로그인 후 복사

개선방법 3 잠금

<code>public class ThreadException {</code><code>    public static volatile int i=0;</code><code>    public static void action(){</code><code>        System.out.println("==>"+Thread.currentThread().getName()+":"+i);</code><code>    }</code><code><br></code><code>    static Lock lock=new ReentrantLock();</code><code>    public static void inc() {</code><code>        lock.lock();</code><code>        try {</code><code>            Thread.sleep(1);</code><code>            i=i+1;</code><code>            action();</code><code>        } catch (InterruptedException e) {</code><code>            e.printStackTrace();</code><code>        } finally {</code><code>            lock.unlock();</code><code>        }</code><code>    }</code><code>    public static void main(String[] args) throws InterruptedException {</code><code>        Thread t1 = new Thread(ThreadException::inc,"t1");</code><code>        Thread t2= new Thread(ThreadException::inc,"t2");</code><code>        t1.start();</code><code>        t2.start();</code><code>    }</code><code>}</code><code><br></code>
로그인 후 복사

분산 잠금: 여러 노드의 동시 실행 보장
구현 계획: 1. 데이터베이스 기반, 2. Redis Cache 기반, 3. Zookeeper 기반

위 내용은 변수에 1을 추가하는 두 개의 Java 스레드 분석 예의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:yisu.com
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿