"+Thread.currentThread().getName( )+":"+i);}publicstaticvoidmain(String[]args)throwsInterruptedException{Threadt1"/> "+Thread.currentThread().getName( )+":"+i);}publicstaticvoidmain(String[]args)throwsInterruptedException{Threadt1">

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

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

WBOY
WBOY앞으로
2023-04-28 21:52:191183검색

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으로 문의하시기 바랍니다. 삭제