>  기사  >  Java  >  Java에서 객체 이스케이프 문제를 게시하고 방지하는 방법은 무엇입니까?

Java에서 객체 이스케이프 문제를 게시하고 방지하는 방법은 무엇입니까?

WBOY
WBOY앞으로
2023-04-22 14:37:08969검색

객체 게시

쉽게 말하면 범위 밖의 코드에 객체에 대한 참조를 제공하는 것입니다. 예를 들어 개체를 반환하거나 다른 클래스의 메서드에 매개 변수로 전달합니다.

안전하지 않은 게시 개체의 예:

<code>@Slf4j</code><code>@NotThreadSafe</code><code>public class UnsafePublish {</code><code><br></code><code>    private String[] states = {"a", "b", "c"};</code><code><br></code><code>    public String[] getStates() {</code><code>        return states;</code><code>    }</code><code><br></code><code>    public static void main(String[] args) {</code><code>        UnsafePublish unsafePublish = new UnsafePublish();</code><code>        log.info("{}", Arrays.toString(unsafePublish.getStates()));</code><code>        // 发布对象不安全,可被修改</code><code>        unsafePublish.getStates()[0] = "d";</code><code>        log.info("{}", Arrays.toString(unsafePublish.getStates()));</code><code>    }</code><code>}</code>

객체 탈출

클래스가 외부 코드에 객체 참조를 제공하고 생성이 완료되기 전에 객체를 해제한 경우 이를 객체 탈출이라고 하며 스레드의 보안이 파괴됩니다.

<code>public class Escape {</code><code>    private int thisCanBeEscape = 1;</code><code><br></code><code>    public Escape() {</code><code>        new InnerClass();</code><code>        // 还有业务需要执行</code><code>        thisCanBeEscape++;</code><code>    }</code><code><br></code><code>    private class InnerClass {</code><code>        public InnerClass() {</code><code>            log.info("{}", Escape.this.thisCanBeEscape);</code><code>        }</code><code>    }</code><code><br></code><code>    public static void main(String[] args) {</code><code>        new Escape();</code><code>    }</code><code>}</code>
  • 이 내부 클래스의 인스턴스에는 캡슐화된 인스턴스의 개인 도메인 객체에 대한 참조가 포함되어 있습니다. 객체가 올바르게 구성되기 전에 해제될 수 있으며 이로 인해 이 참조가 발생하게 됩니다. 건설 중 오버플로 오류가 발생했습니다.

  • 위 코드는 함수 생성 중에 스레드를 시작합니다. 암시적 시작이든 명시적 시작이든 관계없이 이 참조의 오버플로가 발생합니다. 새 스레드는 생성되기 전에 항상 소유 개체를 확인합니다.

클래스 이름.this에 대한 설명

"class name.this"의 구문을 Java 언어에서는 "qualified this"라고 합니다. 이 구문의 주요 목적은 내부 클래스의 메서드에서 특정 중첩 수준에서 외부 클래스의 "this" 참조를 지정하려는 경우 "외부 클래스 이름.this" 구문을 사용하는 것입니다. 예:

class Foo {  class Bar {    Foo getFoo() {      return Foo.this;    }  }}

Foo.Bar 클래스의 getFoo() 메서드에서 "this"를 직접 쓰면 Foo.Bar 클래스의 인스턴스를 참조합니다. 주변 Foo 클래스의 this 인스턴스를 지정하려면 여기에 Foo.this로 작성해야 합니다. 특히, 위의 예에서 getFoo() 메소드에 Bar.this를 작성하면 현재 Foo.Bar 클래스 인스턴스를 지정하여 this를 직접 작성하는 것과 효과가 동일합니다.

객체의 안전한 릴리스

  1. 정적 초기화 함수에서 객체 참조를 초기화합니다.

  2. 객체의 참조를 휘발성 유형 필드 또는 AtomicReference 객체에 저장합니다.

  3. 객체의 참조를 올바르게 저장합니다. 생성된 객체 최종 유형 필드에서

  4. 객체 참조를 잠금으로 보호되는 필드에 저장합니다.

싱글톤 모드에서 배고픈 모드/게으른 모드를 생각할 수 있습니다.

안전한 공유 객체 정책

  1. 스레드 제한: 스레드 제한 객체는 스레드에만 적용되며 이를 소유한 스레드에 의해서만 수정될 수 있습니다.

  2. 공유 읽기 전용: 공유 읽기 전용 개체는 추가 동기화 없이 여러 스레드에서 동시에 액세스할 수 있지만 어떤 스레드도 이를 수정할 수 없습니다.

  3. 스레드로부터 안전한 객체: 내부적으로 스레드 안전성을 보장하기 위해 동기화 메커니즘을 사용하는 스레드로부터 안전한 객체 또는 컨테이너로, 추가 동기화 없이 다른 스레드가 공개 인터페이스를 통해 마음대로 액세스할 수 있습니다.

  4. 보호된 개체: 보호된 개체는 특정 잠금을 획득해야만 접근할 수 있습니다.

위 내용은 Java에서 객체 이스케이프 문제를 게시하고 방지하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 yisu.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제