多線程安全的單例模式

單例模式被認爲是最簡單的設計模式,屬於建立型(設計模式又被分爲:建立型、結構型和行爲型),常常被用到,下面以我在實際項目中用到的一個單例模式爲例,看下如何利用經典的兩次判空方法令其高效、安全得工做在多線程環境(見代碼中註釋)。java

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;
    }
}
相關文章
相關標籤/搜索