Java에서 동기화 사용 방법 요약
1. 동기화를 함수 수정자로 사용하는 경우 샘플 코드는 다음과 같습니다.
Public synchronized void method(){ //…. }
이것이 동기화 방법입니다. 그러면 이때 동기화가 잠겨 있는 개체는 무엇입니까? 그가 잠그는 것은 이 동기화된 메소드 객체를 호출하는 것입니다. 즉, 객체 P1이 서로 다른 스레드에서 이 동기화 방법을 실행하면 이들 사이에 상호 배제가 형성되어 동기화 효과를 얻을 수 있습니다. 그러나 이 객체가 속한 Class에 의해 생성된 또 다른 객체 P2는 동기화 키워드를 추가하여 이 메소드를 임의로 호출할 수 있습니다.
위의 샘플 코드는 다음 코드와 동일합니다.
public void method() { synchronized (this) // (1) { //….. } }
(1)의 내용은 무엇을 의미하나요? 그는 P1과 같이 이 메서드를 호출하는 개체를 참조합니다. 동기화 방법의 본질은 개체 참조에 동기화를 적용하는 것임을 알 수 있습니다. ——P1 개체 잠금을 획득한 스레드만 P1의 동기화 방법을 호출할 수 있습니다. P2에 관한 한 P1 잠금은 이와 관련이 없습니다. 프로그램은 동기화 메커니즘의 제어도 제거할 수 있습니다. 이 경우 데이터 혼란이 발생합니다.
2. 동기화 블록, 샘플 코드는 다음과 같습니다.
public void method(SomeObject so) { synchronized(so) { //….. } }
이때 잠금은 so 객체이며 잠금을 얻은 사람은 누구나 자신이 제어하는 코드를 실행할 수 있습니다. 잠금으로 사용할 클리어 객체가 있는 경우에는 다음과 같이 프로그램을 작성할 수 있지만 잠금으로 사용할 클리어 객체가 없고 코드 조각만 동기화하려는 경우에는 특수 인스턴스 변수를 생성할 수 있습니다(반드시
class Foo implements Runnable { private byte[] lock = new byte[0]; // 特别的instance变量 Public void method() { synchronized(lock) { //… } } //….. }
참고: 길이가 0인 바이트 배열 객체는 어떤 객체보다 생성하는 것이 더 경제적입니다. 컴파일된 바이트코드를 살펴보세요. 길이가 0인 byte[] 객체를 생성하려면 3개의 opcode만 필요합니다. , Object lock = new Object ()에는 7줄의 opcode가 필요합니다.
3. 동기화를 정적 함수에 적용합니다. 샘플 코드는 다음과 같습니다.
Class Foo { public synchronized static void method1() // 同步的static 函数 { //…. } public void method2() { synchronized(Foo.class) // class literal(类名称字面常量) } }
코드의 method2() 메서드는 클래스 리터럴을 잠금으로 사용합니다. 이는 동기화된 정적 함수와 동일한 효과를 가지며 획득된 잠금은 다음과 같습니다. 매우 특별합니다. 는 현재 이 메서드를 호출하는 개체가 속한 클래스입니다(이 클래스에 의해 생성된 특정 개체가 아닌 클래스).
"Effective Java"라는 책에서 동기화 잠금을 위해 Foo.class와 P1.getClass()를 사용하는 것은 동일하지 않다는 내용을 읽은 기억이 납니다. 이 클래스를 잠그는 목적을 달성하기 위해 P1.getClass()를 사용할 수 없습니다. P1은 Foo 클래스에 의해 생성된 객체를 참조합니다.
추론할 수 있습니다. 동기화된 정적 함수 A가 클래스에 정의되고 동기화된 인스턴스 함수 B도 정의된 경우 이 클래스의 동일한 개체 Obj가 여러 스레드에서 각각 A 및 B 메서드에 액세스하면 not 잠금이 모두 다르기 때문에 동기화를 구성합니다. 메소드 A의 잠금은 Obj가 속한 클래스이고, 메소드 B의 잠금은 Obj가 속한 객체입니다.
Java에서 동기화를 사용하는 방법을 요약하면 다음과 같습니다.
어떤 객체 동기화 잠금을 이해하면 보다 안전한 멀티 스레드 프로그램을 설계하는 데 도움이 될 수 있습니다.
공유 리소스에 대한 동기식 액세스를 더욱 안전하게 만들 수 있는 몇 가지 기술도 있습니다.
1. 공용/보호된 인스턴스 변수 대신 개인용 인스턴스 변수 + 해당 get 메소드를 정의하십시오. 변수를 public으로 정의하면 동기화 방식의 제어를 우회하여 객체가 외부에서 직접 변수를 얻어서 변경할 수 있습니다. 이는 JavaBean의 표준 구현 방법 중 하나이기도 합니다.
2. 인스턴스 변수가 배열이나 ArrayList와 같은 객체인 경우 위의 메소드는 여전히 안전하지 않습니다. 왜냐하면 외부 객체가 get 메소드를 통해 인스턴스 객체의 참조를 가져와 다른 객체를 가리킬 때 전용 변수가 바뀌면 위험하지 않을까요? 이때 get 메소드에 동기화된 동기화를 추가하고 이 전용 객체의 clone()만 반환해야 합니다. 이런 방식으로 호출자가 얻는 것은 객체 복사본에 대한 참조입니다.
위 내용은 Java에서 동기화 메커니즘을 구현하기 위해 동기화를 사용하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!