AspectJ实现设计模式(六)—单例模式

原创 2003年07月22日 08:00:00

本文介绍使用AspectJ实现设计模式之单例模式,文章会实现一个AspectJ版本的登记式单例类。

 <?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

示例说明

单例模式有以下一些基本特点

·单例类只能有一个实例

·单例类必须自己创建自己的唯一实例

·单例类必须给所有其他对象提供这一实例

 

Java中我们实现单例类一般需要使用私有构造子和一个静态实例变量,还要提供一个方法如getInstance()来构造并返回这个实例变量。而使用AspectJ实现单例模式,可以完全将这些细节交由一个具体方面处理,客户端仍可以使用new操作符创建对象的实例,而且该方面将利用错误声明(declare error)机制联合pointcut捕捉单例类子类的构造行为并将其禁止,以保证单例类的实例不会被其子类构造。下面就让我们来通过使用AspectJ实现一个登记式的单例类,来看看到底具体方面变了怎样的魔术。

<?xml:namespace prefix = v ns = "urn:schemas-microsoft-com:vml" />

              1AspectJ实现登记式单例类的UML

 

如图所示,抽象方面RegSingletonProtocol使用inter-type声明定义了一个Singleton标识接口,具体单例类需要实现此接口。私有变量singletons用来登记各种单例类的实例变量。抽象pointcut由子方面实现,而Advice around则实现具体的单例类构造和登记入表的逻辑。

 

具体方面RegSingletonImpl则继承了抽象方面,它定义了具体的subClassConstructor的行为并声明了一个RegSingleton类,声明它实现Singleton接口,它就是具体的单例类。还声明了一个编译时错误,它当客户代码调用RegSingleton子类的构造子时发生。

 

图左侧的类就是用来测试单例类的功能以及单例类的子类能否构造单例类,所以逻辑很简单只包含构造子。

 

      源代码

抽象方面RegSingletonProtocol.java

import java.util.*;

public abstract aspect RegSingletonProtocol {

  private Hashtable singletons=new Hashtable();

  public interface Singleton{}

  abstract pointcut subClassConstructor();

 

  Singleton around() : call((Singleton+).new(..)){//捕捉Singleton子类的构造子调用

       //获取类变量

    Class singleton=thisJoinPoint.getSignature().getDeclaringType();

    if(singletons.get(singleton)==null){

      singletons.put(singleton,proceed());//若没有此类变量,则将其实例登记入表

    }

    return (Singleton)singletons.get(singleton);//返回指定类别的实例

  }

}

值得一提的是around,它首先利用thisJoinPoint实例的方法获得类变量,然后搜索表,若该类型不存在则利用该类型本身的构造子创建实例(使用proceed()方法执行捕捉到的方法调用的原有逻辑,这里表示Singleton子类的构造函数)并将其存入表中利用,然后返回该类型的实例。

 

具体方面RegSingletonImpl..java

public aspect RegSingletonImpl extends RegSingletonProtocol{

  declare parents : RegSingleton implements Singleton;

  pointcut subClassConstructor() : call((RegSingletonChild+).new(..));

  declare error : subClassConstructor() : "Sub class can't access the singleton instance";

}

具体方面声明具体的单例类,让它实现Singleton接口subClassConstructor捕捉所有RegSingletonChild及其子类的构造子调用(”+”表示当前类和它的子类)declare error当捕捉到此调用时产生编译错误。

 

单例类RegSingleton..java

public class RegSingleton {

  private static int count=0;

  public RegSingleton() {

    count++;

    System.out.println("Class No. "+count);

  }

}

 

单例类子类RegSingletonChild.java

public class RegSingletonChild extends RegSingleton{

  public RegSingletonChild() {

    super();

  }

}

 

客户端演示Demo.java

public class Demo {

  public static void main(String [] args){

    System.out.println(new RegSingleton());

//    System.out.println(new RegSingletonChild());

    System.out.println(new RegSingleton());

  }

}

 

结果分析

Demo输出的结果如下

Class No. 1

aopsingleton.RegSingleton@6e1408

aopsingleton.RegSingleton@6e1408

表明类RegSingleton只创建了一次,两次构造子调用返回同样的实例,这符合单例类的特点。注意如果将Demo中被注释掉的RegSingletonChild创建打开并编译就会得到如下错误

1  error

"D:/JavaDev/jbuilder/AOPSingleton/src/Demo.java":

Sub class can't access the singleton instance at line 15, column 0

这样就克服了在Java实现登记式单例类子类创建其他实例的缺点。

 

读者可以很容易的将这个例子改为非登记的单例类。只要在RegSingletonProtocol中加入如下语句即可

  private Singleton instance=null;

然后修改around中的逻辑如下

  Singleton around() : call((Singleton+).new(..)){

    if(instance==null){

      try{

      instance=(Singleton)proceed();

      }catch(Exception e){

        throw new RuntimeException("Construction failed");

      }

    }

return instance;

  }

 

声明

本文由starchu1981保留版权,如果需要转贴请写明作者和出处。

跟我学aspectj之六 ----- 插曲

其实,写这样的文章不太好写, 因为都是牵一发而动全身。介绍pointcut,的时候必要涉及到语法,也会涉及advice,以及一些其他的语法。    所以,如果大家在看博客的过程中,如果有疑问...
  • zl3450341
  • zl3450341
  • 2012年06月18日 17:10
  • 6727

设计模式——单例模式的几种实现方式

1.饿汉式:静态常量 这种方法非常的简单,因为单例的实例被声明成static和final变量了,在第一次加载类到内存中时就会初始化,所以会创建实例本身是线程安全的。public class Singl...
  • bingogirl
  • bingogirl
  • 2016年09月01日 00:03
  • 1267

C++设计模式5--单例模式Singleton--当前对象只有一个实例

很多情况下,我们在开发项目的过程中,都希望自己运行的某个部件只有一个实例, 比如我们天天用QT开发界面,QTCreate里帮助菜单下的关于Qt Create菜单,弹出来的关于对话框,在QTCreate...
  • gatieme
  • gatieme
  • 2014年01月08日 13:25
  • 21467

设计模式系列(一)单例模式

一、简单介绍 单例模式是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中一个类只有一个实例。 《设计模式》书中对单例模式动机的介绍 对于系统中的某些类...
  • robertcpp
  • robertcpp
  • 2016年06月05日 11:00
  • 2226

单例模式的6种实现方式

为什么使用单例模式需要确保某个类只要一个对象,或创建一个类需要消耗的资源过多,如访问IO和数据库操作等,这时就需要考虑使用单例模式了。使用单例模式需要注意的关键点 将构造函数访问修饰符设置为priva...
  • ts1900
  • ts1900
  • 2016年05月13日 16:03
  • 1647

Python如何实现单例模式?其他23种设计模式python如何实现?

不会。。。
  • u011860731
  • u011860731
  • 2015年07月19日 17:32
  • 522

C#设计模式(一)单例模式

一、引言 最近在设计模式的一些内容,主要的参考书籍是《Head First 设计模式》,同时在学习过程中也查看了很多博客园中关于设计模式的一些文章的,在这里记录下我的一些学习笔记,一是为了帮助我...
  • kone0611
  • kone0611
  • 2017年07月20日 11:49
  • 165

跟我学AspectJ(一)

跟我学AspectJ(一) 编者的话       关于AspectJ的开发资料好象目前还只有英文版的,而且还不是很多,这对于有兴趣学习AOP而英语不是很好的开发人员来是一件很苦闷的事情,所以我决定总结...
  • starchu1981
  • starchu1981
  • 2003年07月22日 07:58
  • 17266

Picasso源码分析(一):单例模式、建造者模式、面向接口编程

Picasso简单介绍Picasso使用建造者设计模式,使异步网络加载图片到控件这一复杂流程可以用一条方法链搞定。典型的Picasso加载图片的使用方式如下:Picasso.with(context)...
  • shihui512
  • shihui512
  • 2016年06月12日 17:06
  • 1155

设计模式1:单例模式(考虑多线程的情况)

单例模式是设计模式中最简单的形式之一。这一模式的目的是使得类的一个对象成为系统中的唯一实例。要实现这一点,可以从客户端对其进行实例化开始。因此需要用一种只允许生成对象类的唯一实例的机制,“阻止”所有想...
  • mrfly
  • mrfly
  • 2013年10月28日 22:50
  • 6776
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:AspectJ实现设计模式(六)—单例模式
举报原因:
原因补充:

(最多只允许输入30个字)