打开APP
userphoto
未登录

开通VIP,畅享免费电子书等14项超值服

开通VIP
ABAP和Java单例模式的攻防

ABAP

CLASS zcl_jerry_singleton DEFINITIONPUBLICFINALCREATE PRIVATE .PUBLIC SECTION.INTERFACES if_serializable_object .CLASS-METHODS class_constructor .CLASS-METHODS get_instanceRETURNINGVALUE(ro_instance) TYPE REF TO zcl_jerry_singleton .PROTECTED SECTION.PRIVATE SECTION.CLASS-DATA so_instance TYPE REF TO zcl_jerry_singleton .DATA mv_name TYPE string .DATA mv_initialized TYPE abap_bool .METHODS constructor .ENDCLASS.CLASS ZCL_JERRY_SINGLETON IMPLEMENTATION.* <SIGNATURE>---------------------------------------------------------------------------------------+* | Static Public Method ZCL_JERRY_SINGLETON=>CLASS_CONSTRUCTOR* +-------------------------------------------------------------------------------------------------+* +--------------------------------------------------------------------------------------</SIGNATURE>METHOD class_constructor.so_instance = NEW zcl_jerry_singleton( ).ENDMETHOD.* <SIGNATURE>---------------------------------------------------------------------------------------+* | Instance Public Method ZCL_JERRY_SINGLETON->CONSTRUCTOR* +-------------------------------------------------------------------------------------------------+* +--------------------------------------------------------------------------------------</SIGNATURE>METHOD constructor.mv_name = 'Jerry'.IF mv_initialized = abap_false.mv_initialized = abap_true.ELSE.MESSAGE 'you are in trouble!' TYPE 'E' DISPLAY LIKE 'I'.ENDIF.ENDMETHOD.* <SIGNATURE>---------------------------------------------------------------------------------------+* | Static Public Method ZCL_JERRY_SINGLETON=>GET_INSTANCE* +-------------------------------------------------------------------------------------------------+* | [<-()] RO_INSTANCE TYPE REF TO ZCL_JERRY_SINGLETON* +--------------------------------------------------------------------------------------</SIGNATURE>METHOD get_instance.ro_instance = so_instance.ENDMETHOD.ENDCLASS.

通过序列化/反序列化攻击单例模式:

DATA(lo_instance) = zcl_jerry_singleton=>get_instance( ).DATA: s TYPE string.CALL TRANSFORMATION id SOURCE model = lo_instance RESULT XML s.DATA: lo_instance2 TYPE REF TO zcl_jerry_singleton.CALL TRANSFORMATION id SOURCE XML s RESULT model = lo_instance2.

绕过了单例的限制,构造了第二个实例。

Java

除了用序列化/反序列化攻击外,还可以用反射攻击。

然而我只需要将这个单例类JerrySingleton的构造函数通过反射设置成可以访问Accessible,然后就能通过反射调用该构造函数,进而生成新的对象实例。这样就破坏了单例模式。

第6行代码会打印false。

针对这种攻击,一种可行的防御措施是在单例类的构造函数内定义一个布尔变量,初始化为false。当构造函数执行后,该变量被置为true。如果接下来构造函数再次被执行,则人为抛出异常,避免构造函数重复执行。

这种防御措施无法从根本上杜绝Singleton被攻击,因为攻击者仍旧可以通过反射来修改布尔变量flag的值,从而绕过这个检查。

最理想的不会受到攻击的单例模式实现是借助Java里枚举类Enumeration的特性:

这种实现类型的单例模式的消费代码:

System.out.println("Name:" + JerrySingletonAnotherApproach.INSTANCE.getName());

如果攻击者通过前面介绍的反射代码对这种实现方式的单例进行攻击,JDK会抛出NoSuchMethodException异常:

究其原因,是因为现在我们是通过Java枚举方式实现的单例,枚举类没有传统意义上的构造函数,因此对这种反射攻击免疫。

要获取更多Jerry的原创文章,请关注公众号"汪子熙"

本站仅提供存储服务,所有内容均由用户发布,如发现有害或侵权内容,请点击举报
打开APP,阅读全文并永久保存 查看更多类似文章
猜你喜欢
类似文章
【热】打开小程序,算一算2024你的财运
C#设计模式
创建型-单例模式(Singleton Pattern)
初级必备:单例模式的7个问题
ABAP面向对象之单例模式(Singleton Pattern)
Java设计模式-之Singleton单例模式 - 设计模式
5.3.6 利用缓存来实现单例模式
更多类似文章 >>
生活服务
热点新闻
分享 收藏 导长图 关注 下载文章
绑定账号成功
后续可登录账号畅享VIP特权!
如果VIP功能使用有故障,
可点击这里联系客服!

联系客服