> Java > java지도 시간 > 본문

Java의 참조 사용법 소개

黄舟
풀어 주다: 2017-03-16 10:19:07
원래의
2137명이 탐색했습니다.

이 글은 주로 Java의 Reference 사용법에 대한 자세한 설명을 소개하고 있습니다. 필요한 친구는 jdk의

Java Reference 상세 설명

을 참고하세요. 1.2 이후에는 강한 참조 , 소프트 참조, 약한 참조, 가상 참조의 네 가지 개념이 도입되었습니다. 이 네 가지 개념에 대한 설명은 인터넷에 많이 있지만 대부분은 개념적이고 일반적인 내용이므로 오늘은 코드를 기준으로 분석해 보겠습니다. 먼저 정의와 일반적인 설명을 살펴보겠습니다.(참조 유형은 Java 패키지에 있습니다.lang.ref).

 1. 강한 참조(StrongReference)

  강한 참조는 GC에 의해 재활용되지 않으며, java.lang.ref에는 실제 대응되는 타입이 없습니다. 예:


 Object obj = new Object();
로그인 후 복사

여기의 obj 참조는 강력한 참조이며 GC에서 재활용되지 않습니다.

2. 소프트 참조

소프트 참조는 JVM이 메모리 부족을 보고할 때만 GC에 의해 재활용되고 그렇지 않으면 재활용되지 않습니다. 기능 소프트 참조는 캐싱 및 풀링에 널리 사용됩니다. 소프트 참조 사용법:


Object obj = new Object();
SoftReference<Object> softRef = new SoftReference(obj);
// 使用 softRef.get() 获取软引用所引用的对象
Object objg = softRef.get();
로그인 후 복사

3. 약한 참조(WeakReference)

GC가 약한 참조 개체를 발견하면 해당 개체는 WeakReference가 참조하는 내용이 해제됩니다. 약한 참조의 사용법은 소프트 참조와 유사하지만 재활용 전략이 다릅니다.

4. Phantom Reference

GC가 가상 참조 개체를 발견하면 이때 PhantomReference 개체를 ReferenceQueue에 삽입합니다. , PhantomReference가 가리키는 객체는 GC에 의해 재활용되지 않았지만 ReferenceQueue가 실제로 처리될 때까지 재활용되지 않습니다. 가상 참조 사용법:


Object obj = new Object();
ReferenceQueue<Object> refQueue = new ReferenceQueue<Object>();
PhantomReference<Object> phanRef = new PhantomReference<Object>(obj, refQueue);
// 调用phanRef.get()不管在什么情况下会一直返回null
Object objg = phanRef.get();
// 如果obj被置为null,当GC发现了虚引用,GC会将phanRef插入进我们之前创建时传入的refQueue队列
// 注意,此时phanRef所引用的obj对象,并没有被GC回收,在我们显式地调用refQueue.poll返回phanRef之后
// 当GC第二次发现虚引用,而此时JVM将phanRef插入到refQueue会插入失败,此时GC才会对obj进行回收
Reference<? extends Object> phanRefP = refQueue.poll();
로그인 후 복사

간단한 정의를 읽은 후 코드로 테스트해 보았습니다. 말할 필요도 없이 소프트 참조 설명에 강한 참조가 아주 명확하게 나와 있습니다. . , 핵심은 "약한 참조"와 "가상 참조"입니다.

약한 참조:


public static void main(String[] args) throws InterruptedException {
  Object obj = new Object();
  ReferenceQueue<Object> refQueue = new ReferenceQueue<Object>();
  WeakReference<Object> weakRef = new WeakReference<Object>(obj, refQueue);
  System.out.println(weakRef.get());
  System.out.println(refQueue.poll());
  obj = null;
  System.gc();
  System.out.println(weakRef.get());
  System.out.println(refQueue.poll());
}
로그인 후 복사

System.gc()는 지금이 GC를 수행하기에 좋은 시기라고 JVM에 알리고 있기 때문입니다. , 그러나 구체적인 실행 여부는 JVM에 의해 결정되므로 JVM이 GC를 실행하기로 결정하면 결과는 다음과 같습니다(사실 이 코드는 일반적으로 GC를 실행합니다).


  java.lang.Object@de6ced
  null
  null
  java.lang.ref.WeakReference@1fb8ee3
로그인 후 복사

Execute from 결과적으로 WeakRef.get()을 호출하여 obj 객체를 얻습니다. GC가 실행되지 않으므로 refQueue.poll()은 이때 obj = null; 는 힙의 obj 객체에 대한 참조가 없습니다. 여기서 JVM은 GC를 수행했으며, WeakRef.get()을 통해 null이 반환되었고 refQueue.poll()이 WeakReference 객체를 반환했기 때문에 JVM이 WeakRef를 삽입했습니다. obj를 재활용한 후 refQueue 대기열.

더미 참조:


public static void main(String[] args) throws InterruptedException {
  Object obj = new Object();
  ReferenceQueue<Object> refQueue = new ReferenceQueue<Object>();
  PhantomReference<Object> phanRef = new PhantomReference<Object>(obj, refQueue);
  System.out.println(phanRef.get());
  System.out.println(refQueue.poll());
  obj = null;
  System.gc();
  System.out.println(phanRef.get());
  System.out.println(refQueue.poll());
}
로그인 후 복사

마찬가지로 JVM이 GC를 실행할 때 결과는 다음과 같습니다.


  null
  null
  null
  java.lang.ref.PhantomReference@1fb8ee3
로그인 후 복사

실행 결과를 보면 앞서 말한 내용이 정확하다는 것을 알 수 있습니다. JVM이 GC를 실행하고 가상 참조를 찾으면 phanRef.get()은 null을 반환합니다. JVM Obj는 재활용되지 않지만 PhantomReference 객체는 해당 가상 참조 큐 refQueue에 삽입됩니다. refQueue.poll()을 호출하여 PhantomReference 객체를 반환하면 폴 메소드는 먼저 PhantomReference의 보유 큐(ReferenceQueue)를 큐에 넣습니다. super T>)가 NULL로 설정되고, NULL 객체가 ReferenceQueue에서 을 상속하고, enqueue(Reference paramReference) 메서드가 return false로 덮어쓰여지며, obj가 발견되면 다시 GC에 의해 JVM은 PhantomReference를 NULL 큐에 삽입하고 삽입이 실패하고 false를 반환합니다. 이때 GC는 obj를 재활용합니다. 실제로 이 코드를 통해서는 obj가 재활용되었는지 알 수 없지만 PhantomReference의 javadoc 주석 에 다음과 같은 문장이 있습니다.

가비지 수집기가 결정하면 객체 obj가 팬텀-R가능하고 해당 큐에 대기열에 포함되지만 해당 대상은 지우지되지 않습니다. 즉, 팬텀 참조의 참조 큐입니다. 일부 애플리케이션 코드에 의해 명시적으로 처리되어야 합니다.

번역(이 문장은 매우 간단하므로 많은 사람들이 이해해야 한다고 생각합니다):

GC가 "obj"를 결정하면 가상 연결이 가능하면 해당 항목(PhantomReference 참조)이 해당 대기열에 추가되지만 해당 참조는 지워지지 않습니다. 즉, 가상 참조의 참조 큐는 일부 애플리케이션 코드에 의해 명시적으로 처리되어야 합니다.

약한 참조와 가상 참조의 활용

  软引用很明显可以用来制作caching和pooling,而弱引用与虚引用呢?其实用处也很大,首先我们来看看弱引用,举个例子:


class Registry {
  private Set registeredObjects = new HashSet();
 
  public void register(Object object) {
    registeredObjects.add( object );
  }
}
로그인 후 복사

所有我添加进 registeredObjects 中的object永远不会被GC回收,因为这里有个强引用保存在registeredObjects里,另一方面如果我把代码改为如下:


class Registry {
   private Set registeredObjects = new HashSet();
 
   public void register(Object object) {
     registeredObjects.add( new WeakReference(object) );
   }
 }
로그인 후 복사

  现在如果GC想要回收registeredObjects中的object,便能够实现了,同样在使用HashMap如果想实现如上的效果,一种更好的实现是使用WeakHashMap。

而虚引用呢?我们先来看看javadoc的部分说明:

Phantom references are useful for implementing cleanup operations that are necessary before an object gets garbage-collected. They are sometimes more flexible than the finalize() method.

翻译一下:

虚引用在实现一个对象被回收之前必须做清理操作是很有用的。有时候,他们比finalize()方法更灵活。

很明显的,虚引用可以用来做对象被回收之前的清理工作。

위 내용은 Java의 참조 사용법 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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