单例模式被认为是最简单的设计模式,属于建立型(设计模式又被分为:建立型、结构型和行为型),常常被用到,下面以我在实际项目中用到的一个单例模式为例,看下如何利用经典的两次判空方法令其高效、安全得工做在多线程环境(见代码中注释)。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; } }