먼저 코드를 살펴보겠습니다.
class MyThread implements Runnable{ private int count = 10; @Override public void run() { while(count > 0) { System.out.println(Thread.currentThread().getName() + "开始执行"); System.out.println("count还剩" + --count); System.out.println(Thread.currentThread().getName() + "执行结束"); System.out.println(); } } } public class Test{ public static void main(String[] args) { MyThread myThread = new MyThread(); new Thread(myThread).start(); new Thread(myThread).start(); new Thread(myThread).start(); } }
실행 결과는 다음과 같습니다.
Thread-0开始执行 Thread-2开始执行 Thread-1开始执行 count还剩8 count还剩9 Thread-2执行结束 count还剩7 Thread-1执行结束 Thread-1开始执行 Thread-0执行结束 Thread-0开始执行 count还剩5 Thread-0执行结束 Thread-2开始执行 count还剩4 Thread-2执行结束 Thread-2开始执行 count还剩3 count还剩6 Thread-2执行结束 Thread-2开始执行 Thread-0开始执行 count还剩2 Thread-2执行结束 Thread-2开始执行 count还剩0 Thread-2执行结束 Thread-1执行结束 count还剩1 Thread-0执行结束
분명히 이것은 우리가 원하는 결과는 아닙니다. 자체 run()을 실행하는 run() 메서드입니다. 그러나 위의 결과는 순서가 잘못되었습니다. 이전 스레드가 작업을 완료하기 전에 두 번째 스레드가 들어옵니다. 그렇다면 이 문제를 해결하는 방법은 무엇입니까?
먼저 3개의 스레드가 우리가 생성한 myThread 객체를 동시에 가져온 다음 이 3개의 스레드가 run() 메서드에 동시에 들어가 실행된다는 것을 분석해 보겠습니다. 각 스레드가 진입 후 해당 작업을 완료하도록 하려면 myThread 개체를 잠그고 다른 스레드가 해당 작업을 수행하지 못하도록 해야 합니다. 예를 들어, 방이 있는데, 그 방에 들어갔다가 다른 사람이 다시 들어오지 못하게 하려면 문을 잠가야 합니다.
Java는 "동기화"라는 잠금 기능을 제공합니다. 그렇다면 어떻게 사용해야 할까요?
"동기화"의 두 가지 모드: 동기화 방법과 동기화 코드 블록.
1. 객체 잠금:
1. 동기화 코드 블록: 코드 블록 앞에 동기화(잠금할 객체)를 추가합니다.
class MyThread implements Runnable{ private int count = 10; @Override public void run() { synchronized (this) { while(count > 0) { System.out.println(Thread.currentThread().getName() + "开始执行"); System.out.println("count还剩" + --count); System.out.println(Thread.currentThread().getName() + "执行结束"); System.out.println(); } } } } public class Test{ public static void main(String[] args) { MyThread myThread = new MyThread(); new Thread(myThread).start(); new Thread(myThread).start(); new Thread(myThread).start(); } }
2. 동기화된 메서드: 메서드의 반환 값 앞에 동기화된 키워드를 추가합니다.
class MyThread implements Runnable{ private int count = 10; @Override public void run() { while(count > 0) { count--; fun(); } } public synchronized void fun() { System.out.println(Thread.currentThread().getName() + "开始执行"); System.out.println("count还剩" + count); System.out.println(Thread.currentThread().getName() + "执行结束"); System.out.println(); } } public class Test{ public static void main(String[] args) { MyThread myThread = new MyThread(); new Thread(myThread).start(); new Thread(myThread).start(); new Thread(myThread).start(); } }
2. 전역 잠금:
위에서는 모든 myThread 개체를 잠급니다. 여러 개체에 대해 동일한 방법을 잠그려면 어떻게 해야 할까요?
객체 잠금에는 동기화 방법과 동기화 코드 블록이라는 두 가지 방법도 있습니다.
1. 동기화 코드 블록 앞의 괄호 안에는 잠그고 싶은 메소드의 클래스를 적어주세요.
class MyThread implements Runnable{ private static int count = 10; @Override public void run() { synchronized (MyThread.class) { while(count > 0) { count--; fun(); } } } public void fun() { System.out.println(Thread.currentThread().getName() + "开始执行"); System.out.println("count还剩" + count); System.out.println(Thread.currentThread().getName() + "执行结束"); System.out.println(); } } public class Test{ public static void main(String[] args) { MyThread myThread = new MyThread(); new Thread(myThread).start(); new Thread(myThread).start(); new Thread(myThread).start(); } }
관련 글:
Java 동기화 잠금(동기화) 샘플 코드 공유Java의 동기화 키워드 교착 상태 및 메모리 사용 문제에 대한 자세한 설명 Java 주석 종합 분석위 내용은 Java: 객체 잠금 '동기화' 및 전역 잠금에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!