乐趣区

多线程安全的单例模式

单例模式被认为是最简单的设计模式,也经常被用到,下面以我在实际项目中用到的一个单例模式为例,看下如何利用经典的两次判空方法令其高效、安全得工作在多线程环境(见代码中注释)。

package core;

import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.io.Reader;
import java.util.Properties;

public class SqlSessionFactorySingleton {private static Logger logger = LoggerFactory.getLogger("SqlSessionFactorySingleton");
    private static final String MYBATIS_CONFIG_FILE = "mybatis-config.xml";
    // 使用 volatile 关键字令 A 线程的修改对 B 线程立即可见
    private static volatile SqlSessionFactory factory = null;
    
    // 屏蔽默认的公共构造函数
    private SqlSessionFactorySingleton() {}

    public static SqlSessionFactory getInstance() {if (factory == null) { // 第一次判空
            // 只有创建 SqlSessionFactory 实例时才需要同步,不直接在方法上加 synchronized 关键字可以避免在每次判断实例是否创建时加锁,极大得提高并发效率
            synchronized (SqlSessionFactorySingleton.class) {
                // 如果 A、B 两个线程同时通过第一次判空,A 获得锁,B 等待,等 A 创建完 SqlSessionFactory 实例释放锁,B 获得锁,此时 B 需要再次判断实例是否已创建来避免重复创建
                if (factory == null) { // 第二次判空
                    String configFile = "config.properties";
                    try (Reader configReader = Resources.getResourceAsReader(configFile); Reader mybatisReader = Resources.getResourceAsReader(MYBATIS_CONFIG_FILE)) {Properties properties = new Properties();
                        properties.load(configReader);
                        SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
                        factory = builder.build(mybatisReader, properties);
                    } catch (IOException e) {logger.error("Exception when reading {} and {}:", configFile, MYBATIS_CONFIG_FILE, e);
                    }
                }
            }
        }
        return factory;
    }
}
退出移动版