热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

java与abap对比,Java和ABAP中的几种引用类型的分析和比较

Java编程语言中几种不同的引用类型是面试时经常容易被问到的问题:强引用,软引用,弱引用,虚引用。其实除了Java之外&#x

Java编程语言中几种不同的引用类型是面试时经常容易被问到的问题:强引用,软引用,弱引用,虚引用。

其实除了Java之外,某些 其他编程语言也有类似概念,比如ABAP。今天我们就来比较一下。

根据ABAP帮助文档,我们可以把某个对象的引用包在一个Weak Reference的实例里。ABAP的Weak Reference实例通过类CL_ABAP_WEAK_REFERENCE实现。

看下面的例子:首先我在堆上创建了一个新的LCL_PERSON实例,然后包到一个ABAP weak reference里。

lo_person = NEW lcl_person( 'Jerry' ).

lo_weak = NEW cl_abap_weak_reference( lo_person ).

b5417c420e7b75131284376fff369277.png

稍后,我们想拿到被包裹的lo_person引用时,使用weak reference提供的get方法。见下图示例:

lo_person = CAST lcl_person( lo_weak->get( ) ).

引用lo_person什么时候会变成initial呢?如果当ABAP垃圾回收器(Garbage Collector)开始工作时,已经没有任何引用再指向lo_person, 则lo_person会变成initial。

看下面这个例子加深理解。REPORT ztest.

PARAMETERS: clear TYPE char1 as CHECKBOX DEFAULT abap_true,

gc TYPE char1 as CHECKBOX DEFAULT abap_true.

CLASS lcl_person DEFINITION.

PUBLIC SECTION.

DATA: mv_name TYPE string.

METHODS: constructor IMPORTING !iv_name TYPE string.

ENDCLASS.

CLASS lcl_person IMPLEMENTATION.

METHOD: constructor.

me->mv_name = iv_name.

ENDMETHOD.

ENDCLASS.

START-OF-SELECTION.

DATA: lo_person TYPE REF TO lcl_person,

lo_weak TYPE REF TO cl_abap_weak_reference.

lo_person = NEW lcl_person( 'Jerry' ).

lo_weak = NEW cl_abap_weak_reference( lo_person ).

IF clear = abap_true.

CLEAR: lo_person.

ENDIF.

IF gc = abap_true.

cl_abap_memory_utilities=>do_garbage_collection( ).

ENDIF.

lo_person = CAST lcl_person( lo_weak->get( ) ).

IF lo_person IS INITIAL.

WRITE: / 'reference not available'.

ELSE.

WRITE: / 'reference still available'.

ENDIF.

这个report有两个开关,如下图。第一个开关控制lo_person这个引用是否被关键字CLEAR显式地置为INITIAL, 第二个开关决定是否在代码中显式地调用ABAP垃圾回收器。

52ed5c049fed27c29b49d829ab246c0d.png

这两个开关的打开和关闭状态,一共有4种组合。

0b3d6c3555f1bb1e5f66f495636e519e.png

在第一种情况下,通过关键字CLEAR清除了lo_person的引用,从ABAP的内存检查器(事务码s_memory_inspector)能发现,lo_person现在已经不指向任何内存中的对象了。

d7d95844eee49b71c42e2fdb7f4b0d3a.png

对于其他三种情况,LCL_PERSON的实例都不会被ABAP垃圾回收器清除:

098ae49ffae9a08a4a8fa58537d04996.png

Java

Java中的weak reference表现行为和ABAP一致。

我把上面的ABAP测试代码用Java程序重新写一遍:import java.lang.ref.WeakReference;class Person{ private String mName; public Person(String name){ this.mName = name;

} public String getName(){ return this.mName;

}

}public class WeakReferenceTest{ public static void check(Person person){ if (person == null) {

System.out.println("Reference invalid");

} else {

System.out.println("Reference still available");

}

} public static void main(String[] args){

Person jerry = null;

WeakReference person = new WeakReference(new Person( "Jerry"));

jerry = new Person("Ben"); // if you comment out this line, Reference will be availableSystem.gc();

Person restore = person.get();

check(restore);

}

}

ABAP Soft reference - ABAP软应用

在我目前使用的ABAP Netweaver 750 SP4系统中,ABAP软应用尚未实现,

a20d7ca0b0b0f701369d802115826595.png

Java和ABAP中的几种引用类型的分析和比较

那么我们就来试试Java的软应用 Soft Reference:package reference;import java.lang.ref.SoftReference;import java.util.ArrayList;class Person2{ private String mName; public Person2(String name){ this.mName = name;

} public String getName(){ return this.mName;

} public void finalize(){

System.out.println("finalize called: " + this.mName);

} public String toString(){ return "Hello, I am " + this.mName;

}

}public class SoftReferenceTest{ public static void main(String[] args){

SoftReference person = new SoftReference(new Person2( "Jerry"));

System.out.println(person.get());

ArrayList big &#61; new ArrayList(); for (int i &#61; 0; i <10000; i&#43;&#43;) {

big.add(new Person2(String.valueOf(i)));

}

System.gc();

System.out.println("End: " &#43; person.get());

}

}

控制台打印出的输出&#xff1a;

Hello, I am Jerry

End: Hello, I am Jerry

即便我创建了1万个Person对象的实例&#xff0c;确实消耗了一些内存&#xff0c;然后内存消耗还远远没有大到会导致包含在软应用中的Person2类的引用被JDK删除掉的程度。因此我在代码中调用Java的垃圾回收器System.gc()之后&#xff0c;该引用仍然存在。

在Java中&#xff0c;软应用通常被用来实现在内存资源很有限的环境下的缓存机制&#xff0c;比如Android手机开发中。

Java 虚引用 PhantomReference

使用下面的代码测试虚引用&#xff1a;package aop;import java.lang.ref.PhantomReference;import java.lang.ref.ReferenceQueue;public class PhantomReferenceTest{ public static void main(String[] args){

Object phantomObj;

PhantomReference phantomRef, phantomRef2;

ReferenceQueue phantomQueue;

phantomObj &#61; new String("Phantom Reference");

phantomQueue &#61; new ReferenceQueue();

phantomRef &#61; new PhantomReference(phantomObj, phantomQueue);

System.out.println("1 Phantom Reference:" &#43; phantomRef.get());

System.out.println("2 Phantom Queued: " &#43; phantomRef.isEnqueued());

phantomObj &#61; null;

System.gc();

System.out.println("3 Anything in Queue? : " &#43; phantomQueue.poll()); if (!phantomRef.isEnqueued()) {

System.out.println("4 Requestion finalization.");

System.runFinalization();

}

System.out.println("5 Anything in Queue?: " &#43; phantomRef.isEnqueued());

phantomRef2 &#61; (PhantomReference) phantomQueue.poll();

System.out.println("6 Original PhantomReference: " &#43; phantomRef);

System.out.println("7 PhantomReference from Queue: " &#43; phantomRef2);

}

}

测试输出&#xff1a;

1. Phantom Reference: null

2. Phantom Queued: false

3. Anything in Queue? : null

5. Anything in Queue?: true

6. Original PhantomReference: java.lang.ref.PhantomReference&#64;2a139a55

7. PhantomReference from Queue: java.lang.ref.PhantomReference&#64;2a139a55

和之前介绍的弱引用(WeakReference)和软引用(SoftReference)不同&#xff0c;包裹在虚引用(PhantomReference)中的对象实例无法通过需引用的get方法返回&#xff0c;因此在第一行输出我们会看到&#xff1a; “1. Phantom Reference: null”.

在上面示例代码中虚引用PhantomReference的构造函数里, 我传入了一个队列作为输入参数。当包裹在虚引用实例中的对象引用被Java垃圾回收器删除时&#xff0c;虚引用实例本身会自动被JVM插入我之前指定到虚引用构造函数输入参数的那个队列中去。

在System.runFinalization()执行之前&#xff0c;phantomRef.isEnqueued()返回false&#xff0c;phantomQueue.poll()返回空。

当phantomObj实例被JVM删除后, 虚引用PhantomReference本身被加入到队列中&#xff0c;并且能够通过队列提供的API所访问&#xff1a;phantomQueue.poll(). 打印输出的第6行和第7行也说明了这一点。

要获取更多Jerry的原创技术文章&#xff0c;请关注公众号"汪子熙"



推荐阅读
  • ListView简单使用
    先上效果:主要实现了Listview的绑定和点击事件。项目资源结构如下:先创建一个动物类,用来装载数据:Animal类如下:packagecom.example.simplelis ... [详细]
  • springMVC JRS303验证 ... [详细]
  • 2017-2018年度《网络编程与安全》第五次实验报告
    本报告详细记录了2017-2018学年《网络编程与安全》课程第五次实验的具体内容、实验过程、遇到的问题及解决方案。 ... [详细]
  • Logback使用小结
    1一定要使用slf4j的jar包,不要使用apachecommons的jar。否则滚动生成文件不生效,不滚动的时候却生效~~importorg.slf ... [详细]
  • 本文介绍了如何使用JFreeChart库创建一个美观且功能丰富的环形图。通过设置主题、字体和颜色等属性,可以生成符合特定需求的图表。 ... [详细]
  • 本文详细介绍了Java中的注解功能,包括如何定义注解类型、设置注解的应用范围及生命周期,并通过具体示例展示了如何利用反射机制访问注解信息。 ... [详细]
  • 深入解析Java多线程与并发库的应用:空中网实习生面试题详解
    本文详细探讨了Java多线程与并发库的高级应用,结合空中网在挑选实习生时的面试题目,深入分析了相关技术要点和实现细节。文章通过具体的代码示例展示了如何使用Semaphore和SynchronousQueue来管理线程同步和任务调度。 ... [详细]
  • 本文将详细探讨 Java 中提供的不可变集合(如 `Collections.unmodifiableXXX`)和同步集合(如 `Collections.synchronizedXXX`)的实现原理及使用方法,帮助开发者更好地理解和应用这些工具。 ... [详细]
  • 本文探讨了如何通过一系列技术手段提升Spring Boot项目的并发处理能力,解决生产环境中因慢请求导致的系统性能下降问题。 ... [详细]
  • 本文详细介绍了Java编程中的基本运算符,包括算术、赋值、关系和逻辑运算符,并深入探讨了三元运算符的使用。此外,还讲解了如何使用Scanner类进行用户输入处理以及if和switch语句等流程控制结构。 ... [详细]
  • 烤鸭|本文_Spring之Bean的生命周期详解
    烤鸭|本文_Spring之Bean的生命周期详解 ... [详细]
  • 深入浅出TensorFlow数据读写机制
    本文详细介绍TensorFlow中的数据读写操作,包括TFRecord文件的创建与读取,以及数据集(dataset)的相关概念和使用方法。 ... [详细]
  • 本文深入探讨了UNIX/Linux系统中的进程间通信(IPC)机制,包括消息传递、同步和共享内存等。详细介绍了管道(Pipe)、有名管道(FIFO)、Posix和System V消息队列、互斥锁与条件变量、读写锁、信号量以及共享内存的使用方法和应用场景。 ... [详细]
  • KMP算法是处理字符串匹配的一种高效算法它首先用O(m)的时间对模板进行预处理,然后用O(n)的时间完成匹配。从渐进的意义上说,这样时间复 ... [详细]
  • 本文介绍了如何使用Java代码在Android设备上检测特定应用程序是否已安装。通过创建一个Intent并利用PackageManager查询该Intent的可用性来实现这一功能。 ... [详细]
author-avatar
心有阳光2502937567_240
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有