设计模式课程 设计模式精讲 8-11 单例模式源码解析(jdk+spring+mybaties)

Posted 1446358788-qq

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了设计模式课程 设计模式精讲 8-11 单例模式源码解析(jdk+spring+mybaties)相关的知识,希望对你有一定的参考价值。

1    源码解析

1.1  单例解析1

1.2  单例解析2(容器单例)

1.3  单例解析3

1.4  单例解析4

 

 

1    源码解析
1.1  单例解析1

java.lang.Runtime

/**
 *    饿汉式加载,初始化的时候,就已经new出了对象
 */ 
private static Runtime currentRuntime = new Runtime();

    /**
     * Returns the runtime object associated with the current Java application.
     * Most of the methods of class <code>Runtime</code> are instance
     * methods and must be invoked with respect to the current runtime object.
     *
     * @return  the <code>Runtime</code> object associated with the current
     *          Java application.
     */
    public static Runtime getRuntime() 
        return currentRuntime;
    

 

1.2  单例解析2(容器单例)

java.awt.Desktop(cs)

 /**
     * Returns the <code>Desktop</code> instance of the current
     * browser context.  On some platforms the Desktop API may not be
     * supported; use the @link #isDesktopSupported method to
     * determine if the current desktop is supported.
     * @return the Desktop instance of the current browser context
     * @throws HeadlessException if @link
     * GraphicsEnvironment#isHeadless() returns @code true
     * @throws UnsupportedOperationException if this class is not
     * supported on the current platform
     * @see #isDesktopSupported()
     * @see java.awt.GraphicsEnvironment#isHeadless
     */

    /*
     *    同步锁,context取对象,如果该对象为为null,new出新的对象,然后放入context
     */
    public static synchronized Desktop getDesktop()
        if (GraphicsEnvironment.isHeadless()) throw new HeadlessException();
        if (!Desktop.isDesktopSupported()) 
            throw new UnsupportedOperationException("Desktop API is not " +
                                                    "supported on the current platform");
        

        sun.awt.AppContext context = sun.awt.AppContext.getAppContext();
        Desktop desktop = (Desktop)context.get(Desktop.class);

        if (desktop == null) 
            desktop = new Desktop();
            context.put(Desktop.class, desktop);
        


    /**
     *  context put的时候加上同步锁,可以避免多线程put异常
     */
    public Object put(Object var1, Object var2) 
        HashMap var3 = this.table;
        synchronized(this.table) 
            MostRecentKeyValue var4 = this.mostRecentKeyValue;
            if (var4 != null && var4.key == var1) 
                var4.value = var2;
            

            return this.table.put(var1, var2);
        
    

 

1.3  单例解析3(Spring框架获取单例对象)

spring中的单例是bean作用域中的一个,作用域在每个应用程序的上下文中只创建一个我们设置属性的实例,

和我们的单例的区别是:spring将实例的数量限制的作用域在整个应用程序的上下文,而java应用程序中,是将类加载器的数量限制在给定的类加载器的整个空间里。

所以说,在spring中启动多个容器的时候,每个容器即使是单例的,都可以拿到这个对象。

  public final T getObject() throws Exception 
        if (this.isSingleton()) 
            return this.initialized ? this.singletonInstance : this.getEarlySingletonInstance();
         else 
            return this.createInstance();
        
    

/*
 *    如果被初始化,获取早期的单例对象
 * 
 */
//通过代理去拿新对象
    private T getEarlySingletonInstance() throws Exception 
        Class<?>[] ifcs = this.getEarlySingletonInterfaces();
        if (ifcs == null) 
            throw new FactoryBeanNotInitializedException(this.getClass().getName() + " does not support circular references");
         else 
            if (this.earlySingletonInstance == null) 
                this.earlySingletonInstance = Proxy.newProxyInstance(this.beanClassLoader, ifcs, new AbstractFactoryBean.EarlySingletonInvocationHandler());
            

            return this.earlySingletonInstance;
        
    

 

1.4  单例解析4(基于threadLocal的线程案例)(mybaties获取单例对象)

mybaties上下文保证了每个线程各自的数据,每个线程自己的上下文,自己保存好

  private static final ThreadLocal<ErrorContext> LOCAL = new ThreadLocal<ErrorContext>();
  
 private ErrorContext() 
  

  public static ErrorContext instance() 
    ErrorContext context = LOCAL.get();
if (context == null) 
      context = new ErrorContext();
      LOCAL.set(context);
    
    return context;
  

 

以上是关于设计模式课程 设计模式精讲 8-11 单例模式源码解析(jdk+spring+mybaties)的主要内容,如果未能解决你的问题,请参考以下文章

设计模式课程 设计模式精讲 8-6 单例设计模式-序列化破坏单例模式原理解析及解决方案

设计模式课程 设计模式精讲 8-5 单例设计模式-饿汉式

设计模式课程 设计模式精讲 8-2 单例设计模式-懒汉式及多线程Debug实战

设计模式课程 设计模式精讲 10-2 外观模式源码解析

设计模式课程 设计模式精讲 6-3 抽象工厂源码解析

设计模式课程 设计模式精讲 16-5 代理模式源码解析