单例模式,它用以确保一个特定的类只有一个对象被实例化。它包含两种类型,有些书上叫singleton模式和Double—Checked Locking模式。
单例模式注意问题:
1、拥有一个特定的方法,这个方法被用于实例化需要的对象。当该方法被调用的时候,它检查这个对象是否被实例化。如果已实例化,这个方法仅仅返回这个对象的一个引用。如果对象未被实例化,这个方法将对象实例化并返回这个新的实例的引用。
2、类的构造函数定义为protected或private。
单例模式的格式
(1)懒汉式
- public class Singleton{
-
- private static Singleton instance = null;
-
-
- public static Singleton getInstance(){
- if(instance==null){
- instance = new Singleton();
- return instance;
- }
- }
- }
(2)饿汉式
饿汉式是线程安全的,在类创建的同时就已经创建好一个静态的对象供系统使用,以后不再改变 。
懒汉式优点是延时加载、 是在需要的时候才创建对象。缺点是应该用同步。如果在创建实例对象时不加上synchronized则会导致对对象的访问不是线程安全的。
- public class Singleton{
- private static Singleton singleton = new Singleton();
- private Singleton (){ }
- public Singleton getInstance(){
- return singletion;
- }
- }
典型应用:
数据库连接池管理类的应用
1、数据库连接池管理类
- package com.cvicse.util;
-
- import java.io.FileInputStream;
- import java.io.FileNotFoundException;
- import java.io.FileWriter;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.PrintWriter;
- import java.sql.Connection;
- import java.sql.Driver;
- import java.sql.DriverManager;
- import java.sql.SQLException;
- import java.util.Date;
- import java.util.Enumeration;
- import java.util.Hashtable;
- import java.util.Properties;
- import java.util.StringTokenizer;
- import java.util.Vector;
-
-
-
-
-
-
-
-
-
-
-
- public class DBConnectionManager {
-
- private static DBConnectionManager instance;
- private static int clients;
- private Vector drivers = new Vector();
- private Hashtable pools = new Hashtable();
- private Properties dbProps;
- private PrintWriter log;
-
-
-
-
- private DBConnectionManager() {
- this.init();
- }
-
-
-
-
-
-
- public static synchronized DBConnectionManager getInstance() {
- if (instance == null) {
- instance = new DBConnectionManager();
- }
- clients++;
- return instance;
- }
-
-
-
-
-
-
-
-
- public Connection getConnection(String name) {
- DBConnectionPool dbPool = (DBConnectionPool) pools.get(name);
- if (dbPool != null) {
- return dbPool.getConnection();
- }
- return null;
- }
-
-
-
-
-
-
-
-
-
-
- public Connection getConnection(String name, long time) {
- DBConnectionPool dbPool = (DBConnectionPool) pools.get(name);
- if (dbPool != null) {
- return dbPool.getConnection(time);
- }
- return null;
- }
-
-
-
-
-
-
-
-
-
- public void freeConnection(String name, Connection con) {
- DBConnectionPool dbPool = (DBConnectionPool) pools.get(name);
- if (dbPool != null) {
- dbPool.freeConnection(con);
- }
- }
-
-
-
-
- public synchronized void release() {
-
- if (--clients != 0) {
- return;
- }
- Enumeration allPools = pools.elements();
- while (allPools.hasMoreElements()) {
- DBConnectionPool pool = (DBConnectionPool) allPools.nextElement();
- pool.release();
- }
- Enumeration allDrivers = drivers.elements();
- while (allDrivers.hasMoreElements()) {
- Driver driver = (Driver) allDrivers.nextElement();
- try {
- DriverManager.deregisterDriver(driver);
- log("撤销JDBC驱动程序 " + driver.getClass().getName() + "的注册");
- } catch (SQLException e) {
- log(e, "无法撤销下列JDBC驱动程序的注册: " + driver.getClass().getName());
- }
- }
- }
-
-
-
-
- private void init() {
-
-
- InputStream fileinputstream = null;
- try {
- fileinputstream = new FileInputStream("./src/db.properties");
- } catch (FileNotFoundException e) {
- e.printStackTrace();
- }
- try {
- dbProps = new Properties();
- dbProps.load(fileinputstream);
- } catch (Exception e) {
- e.printStackTrace();
- System.err.println("不能读取属性文件. "
- + "请确保db.properties在CLASSPATH指定的路径中");
- return;
- }
-
- String logFile = dbProps.getProperty("logfile",
- "DBConnectionManager.log");
- try {
- log = new PrintWriter(new FileWriter(logFile, true), true);
- } catch (IOException e) {
- System.err.println("无法打开日志文件: " + logFile);
- log = new PrintWriter(System.err);
- }
-
- loadDrivers(dbProps);
-
- createPools(dbProps);
- }
-
-
-
-
-
-
-
- private void loadDrivers(Properties props) {
- String driverClasses = props.getProperty("drivers");
- StringTokenizer st = new StringTokenizer(driverClasses);
- while (st.hasMoreElements()) {
- String driverClassName = st.nextToken().trim();
- try {
- Driver driver = (Driver) Class.forName(driverClassName)
- .newInstance();
- DriverManager.registerDriver(driver);
- drivers.addElement(driver);
- log("成功注册JDBC驱动程序" + driverClassName);
- } catch (Exception e) {
- log("无法注册JDBC驱动程序: " + driverClassName + ", 错误: " + e);
- }
- }
- }
-
-
-
-
-
-
-
- private void createPools(Properties props) {
- Enumeration propNames = props.propertyNames();
- while (propNames.hasMoreElements()) {
- String name = (String) propNames.nextElement();
- if (name.endsWith(".url")) {
- String poolName = name.substring(0, name.lastIndexOf("."));
- System.out.println(" poolName ||" + poolName + "|");
- String url = props.getProperty(poolName + ".url");
- if (url == null) {
- log("没有为连接池" + poolName + "指定URL");
- continue;
- }
- String user = props.getProperty(poolName + ".user");
- String password = props.getProperty(poolName + ".password");
- String maxconn = props.getProperty(poolName + ".maxconn", "0");
- int max;
- try {
- max = Integer.valueOf(maxconn).intValue();
- } catch (NumberFormatException e) {
- log("错误的最大连接数限制: " + maxconn + " .连接池: " + poolName);
- max = 0;
- }
- DBConnectionPool pool = new DBConnectionPool(poolName, url,
- user, password, max);
- pools.put(poolName, pool);
- log("成功创建连接池" + poolName);
- }
- }
- }
-
-
-
-
- private void log(String msg) {
- log.println(new Date() + ": " + msg);
- }
-
-
-
-
- private void log(Throwable e, String msg) {
- log.println(new Date() + ": " + msg);
- e.printStackTrace(log);
- }
-
-
-
-
-
-
-
-
-
-
-
-
-
- class DBConnectionPool {
-
- private String poolName;
- private String dbConnUrl;
- private String dbUserName;
- private String dbPassWord;
- private int maxConn;
- private int checkedOut;
- private Vector<Connection> freeConnections;
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- public DBConnectionPool(String poolName, String dbConnUrl,
- String dbUserName, String dbPassWord, int maxConn) {
- this.poolName = poolName;
- this.dbConnUrl = dbConnUrl;
- this.dbUserName = dbUserName;
- this.dbPassWord = dbPassWord;
- this.maxConn = maxConn;
- this.freeConnections = new Vector<Connection>();
- }
-
-
-
-
-
- public synchronized Connection getConnection() {
- Connection conn = null;
- if (freeConnections != null && freeConnections.size() > 0) {
-
- conn = (Connection) freeConnections.firstElement();
- freeConnections.removeElementAt(0);
- try {
- if (conn.isClosed()) {
- log("从连接池" + poolName + "删除一个无效连接");
-
- conn = getConnection();
- }
- } catch (SQLException e) {
- log("从连接池" + poolName + "删除一个无效连接");
-
- conn = getConnection();
- }
- } else if (maxConn == 0 || checkedOut < maxConn) {
- conn = newConnection();
- }
- if (conn != null) {
- checkedOut++;
- }
- return conn;
- }
-
-
-
-
-
-
-
- public synchronized Connection getConnection(long timeout) {
- long startTime = System.currentTimeMillis();
- Connection conn = null;
- while ((conn = getConnection()) == null) {
- try {
- wait(timeout);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- if ((System.currentTimeMillis() - startTime) >= timeout) {
-
- return null;
- }
- }
- return conn;
- }
-
-
-
-
-
-
- private Connection newConnection() {
- Connection conn = null;
- try {
- if (dbUserName == null) {
- conn = DriverManager.getConnection(dbConnUrl);
- } else {
- conn = DriverManager.getConnection(dbConnUrl, dbUserName,
- dbPassWord);
- }
- log("连接池" + poolName + "创建一个新的连接");
- } catch (SQLException e) {
- log(e, "无法创建下列URL的连接: " + dbConnUrl);
- return null;
- }
- return conn;
- }
-
-
-
-
-
-
-
- public synchronized void freeConnection(Connection conn) {
-
- freeConnections.addElement(conn);
- checkedOut--;
- notifyAll();
- }
-
-
-
-
- public synchronized void release() {
- Enumeration<Connection> allConnections = freeConnections.elements();
- while (allConnections.hasMoreElements()) {
- Connection con = (Connection) allConnections.nextElement();
- try {
- con.close();
- log("关闭连接池" + poolName + "中的一个连接");
- } catch (SQLException e) {
- log(e, "无法关闭连接池" + poolName + "中的连接");
- }
- }
- freeConnections.removeAllElements();
- }
- }
- }
2、测试类
-
-
-
-
-
-
-
-
-
-
- package com.cvicse.util;
-
- import java.sql.Connection;
-
-
-
-
-
-
-
-
-
- public class Test {
-
-
-
-
- public static void main(String[] args) {
-
- DBConnectionManager connectionManager = DBConnectionManager
- .getInstance();
- try {
- Thread.sleep(10);
- } catch (InterruptedException e) {
-
- e.printStackTrace();
- }
- Connection conn1 = connectionManager.getConnection("mysql");
- Connection conn2 = connectionManager.getConnection("mysql");
- Connection conn3 = connectionManager.getConnection("mysql");
- Connection conn4 = connectionManager.getConnection("mysql");
- Connection conn5 = connectionManager.getConnection("mysql");
- System.out.println(" conn1 == " + conn1);
- System.out.println(" conn2 == " + conn2);
- System.out.println(" conn3 == " + conn3);
- System.out.println(" conn4 == " + conn4);
- System.out.println(" conn5 == " + conn5);
-
- connectionManager.freeConnection("mysql", conn1);
- connectionManager.freeConnection("mysql", conn2);
- connectionManager.freeConnection("mysql", conn3);
- connectionManager.freeConnection("mysql", conn4);
- connectionManager.freeConnection("mysql", conn5);
- try {
- Thread.sleep(10);
- } catch (InterruptedException e) {
-
- e.printStackTrace();
- }
- Connection conn6 = connectionManager.getConnection("mysql");
- Connection conn7 = connectionManager.getConnection("mysql");
- Connection conn8 = connectionManager.getConnection("mysql");
- Connection conn9 = connectionManager.getConnection("mysql");
- Connection conn10 = connectionManager.getConnection("mysql");
- System.out.println(" conn6 == " + conn6);
- System.out.println(" conn7 == " + conn7);
- System.out.println(" conn8 == " + conn8);
- System.out.println(" conn9 == " + conn9);
- System.out.println(" conn10 == " + conn10);
- }
- }
3、测试结果
conn1 == com.mysql.jdbc.Connection@1f5d386
conn2 == com.mysql.jdbc.Connection@121f1d
conn3 == com.mysql.jdbc.Connection@1b8e059
conn4 == com.mysql.jdbc.Connection@910040
conn5 == com.mysql.jdbc.Connection@1a786c3
conn6 == com.mysql.jdbc.Connection@1f5d386
conn7 == com.mysql.jdbc.Connection@121f1d
conn8 == com.mysql.jdbc.Connection@1b8e059
conn9 == com.mysql.jdbc.Connection@910040
conn10 == com.mysql.jdbc.Connection@1a786c3
启动五个线程,释放后再启动仍然是原先的五个线程。
4、实际应用中的配置
上面所实现的连接池在程序开发时如何应用到系统中呢?下面以Servlet为例说明连接池的使用。
Servlet的生命周期是:在开始建立servlet时,调用其初始化(init)方法。之后每个用户请求都导致一个调用前面建立的实例的service方法的线程。最后,当服务器决定卸载一个servlet时,它首先调用该servlet的 destroy方法。
根据servlet的特点,我们可以在初始化函数中生成连接池管理类的唯一实例(其中包括创建一个或多个连接池)。如:
public void init() throws ServletException { connMgr = DBConnectionManager.getInstance(); } |
然后就可以在service方法中通过连接池名称使用连接池,执行数据库操作。最后在destroy方法中释放占用的系统资源,如:
public void destroy() { connMgr.release(); super.destroy(); } |
分享到:
相关推荐
单例模式是一种设计模式,旨在确保一个类只有一个实例,并提供全局访问点。在单例模式中,类的构造函数是私有的,防止外部直接创建对象,而是通过静态方法获取该类的唯一实例。单例模式的唯一性通常是在进程范围内,...
单例模式是软件设计模式中的一种,用于控制类的实例化过程,确保一个类在整个程序运行期间只有一个实例存在。在Android开发中,单例模式应用广泛,尤其在管理全局资源、提供公共服务或优化性能时非常实用。下面我们...
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。在Java或类似的面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。在这个...
Java单例模式简单示例 Java单例模式是一种非常常用的设计模式,它可以确保某个类只有一个实例,并提供一个全局访问点。下面我们将通过一个简单的示例来介绍Java单例模式的定义和使用技巧。 单例模式的定义 单例...
单例模式是软件设计模式中的一种经典模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下非常有用,比如控制共享资源、管理配置对象等。下面将详细介绍七种常见的单例模式实现...
Python中的单例模式是一种设计模式,它确保一个类只有一个实例,并提供一个全局访问点来获取这个实例。这种模式常用于管理共享资源,如数据库连接、配置对象等,以避免资源的重复创建和销毁,提高效率。 单例模式的...
### 使用Java单例模式实现一个简单的日志记录器 #### 一、单例模式简介 单例模式是一种常用的软件设计模式,在该模式中,一个类只能创建一个实例,并且提供了一个全局访问点来访问该实例。单例模式的主要优点包括...
c# 单例模式的实现方法 单例模式是所有设计模式中最简单的一种,主要用于确保整个应用程序中只有一个实例存在。c# 中实现单例模式有多种方法,本文将详细介绍饿汉式和懒汉式两种实现方法。 饿汉式 饿汉式是最简单...
单例模式是设计模式的一种,它用于确保某一个类只有一个实例存在,而且自行实例化并向系统提供这个实例。在PHP中,单例模式的实现通常涉及以下几个关键点: 首先,需要有一个用于保存类的唯一实例的静态成员变量,...
"浅谈Spring单例Bean与单例模式的区别" 本文主要介绍了Spring单例Bean与单例模式的区别,通过对比两者的定义、实现机制和应用场景,帮助读者更好地理解这两种概念的异同。 一、单例模式的定义和实现 单例模式是一...
Java单例模式是一种常见的设计模式,它在软件开发中用于控制类的实例化过程,确保在任何情况下,对于特定类,系统中只有一个实例存在。这种模式对于那些需要频繁创建和销毁的对象,或者需要全局访问的对象,如配置...
C#单例模式的应用,这里给的是C#里面一个单例模式的应用案例,面向对象过程中不可避免的需要用到单例模式,我在博客园中有详细说明单例模式怎么应用的,链接地址:...
单例模式是一种常见的设计模式,用于确保一个类只有一个实例,同时向整个系统提供这个唯一的实例。这种模式在许多场景中都十分有用,例如,在一个系统中打印机假脱机程序(PrinterSpooler)应该是唯一的,以避免打印...
单例模式常用于控制实例的创建,以确保一个类只有一个实例,并且提供一个全局的访问点。在PHP开发中,单例模式非常有用,尤其是在需要限制对象实例数量时,比如数据库连接池的创建。 单例模式的核心思想在于它封装...
C++ 单例模式的详解及实例 1.什么叫单例模式? 单例模式也称为单件模式、单子模式,可能是使用最广泛的设计模式。其意图是保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。有很...
饿汉式 线程安全,调用效率高,但是不能延时加载。 public class Singleton { private static Singleton instance = new ...如果方法没有 synchronized,单例没有 volatile , 则线程不安全。 public class Singleton {
Java 使用静态关键字实现单例模式 在软件开发中,单例模式是一种常用的设计模式,它可以确保某个类只有一个实例,并提供了全局访问点。Java 语言提供了多种方式来实现单例模式,其中使用静态关键字是其中的一种。...
### JS单例模式的两种实现方案 #### 一、引言 在JavaScript中,单例模式是一种常用的软件设计模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式广泛应用于需要频繁创建的对象,例如日志...
单例模式分析代码优化方法 单例模式是23种设计模式之一,是比较简单的一种设计模式,它的目的是无论调用多少次,都返回同一个对象,它的特点是构造器私有化。单例模式分为两种结构,一种是懒汉式的,一种是饿汉式的...
Java单例模式及实现 Java单例模式是一种常见的设计模式,确保某一个类只有一个实例,而且向这个系统提供这个实例。单例模式可以分为三种:懒汉式单例、饿汉式单例、登记式单例。 单例模式的要点 1. 某个类只能有...