单例模式,它用以确保一个特定的类只有一个对象被实例化。它包含两种类型,有些书上叫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(); } |
分享到:
相关推荐
单例模式是一种设计模式,旨在确保一个类只有一个实例,并提供全局访问点。在单例模式中,类的构造函数是私有的,防止外部直接创建对象,而是通过静态方法获取该类的唯一实例。单例模式的唯一性通常是在进程范围内,...
"设计模式单例模式和工厂模式综合应用"的主题聚焦于两种常用的设计模式:单例模式和工厂模式,并探讨它们如何协同工作来实现高效、灵活的代码结构。这个主题尤其适用于Java编程语言,因为Java的面向对象特性使得设计...
单例模式是软件设计模式中的一种经典模式,它保证了类只有一个实例存在,并提供一个全局访问点。在Java等面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。结合工厂模式,...
单例模式是软件设计模式中的一种经典模式,用于确保一个类只有一个实例,并提供一个全局访问点。在Java中,有多种实现单例模式的方法,每种都有其特点和适用场景。接下来,我们将深入探讨这些实现方式。 首先,我们...
在C++编程中,单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。线程安全的单例模式在多线程环境下尤其重要,因为不正确的实现可能导致多个线程创建多个实例,这违反了单例模式...
**设计模式——单例模式** 在软件工程中,设计模式是一种在特定场景下解决常见问题的标准方案,可以被复用并提升代码质量。单例模式是设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。这种模式...
单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。在C#中,单例模式常用于管理共享资源或控制类的实例化过程,以提高性能、节约系统资源,特别是在整个应用程序生命周期内只需要一...
单例模式是软件设计模式中的一种经典模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下非常有用,比如控制共享资源、管理配置对象等。下面将详细介绍七种常见的单例模式实现...
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。在Java或类似的面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。在这个...
在C++编程中,单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。在这个特定的场景中,我们讨论的是一个实现了单例模式的日志类,该类专为多线程环境设计,具备日志等级控制、...
首先向关注过我这个系列...这立刻让我想到了最常用也是最简单最容易理解的一个设计模式 单例模式 何为 单例模式 ? 故名思议 即 让 类 永远都只能有一个实例。 由于 示例代码 比较简单 我也加了注释,这里就不在赘述
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制资源的唯一性、管理共享配置或者创建昂贵的对象时避免频繁创建销毁。 ...
Java中的单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供全局访问点。在Java编程中,单例模式常用于控制资源的访问,比如数据库连接池、线程池或者日志对象等。本篇文章将深入探讨如何在Java中...
单例模式是一种设计模式,它的主要目标是确保一个类只有一个实例,并提供一个全局访问点。在软件工程中,单例模式常用于控制资源的共享,比如数据库连接池、线程池或者日志系统等,这些资源通常需要全局唯一且高效地...
单例模式是软件设计模式中的一种,它的核心思想是确保一个类在整个系统中只有一个实例,并提供一个全局访问点。在Java或类似编程语言中,单例模式常常被用来管理资源,比如数据库连接、线程池或者配置信息,因为这些...
在Qt的Qml环境中,单例模式是一种设计模式,它允许在整个应用程序中创建一个全局访问点,确保某个类只有一个实例存在。这样的设计模式在需要共享数据或者服务时非常有用,避免了多处创建相同对象导致的数据不一致或...
单例模式是软件设计模式中的经典模式之一,其主要目的是控制类的实例化过程,确保在应用程序的整个生命周期中,某个类只有一个实例存在。这样的设计通常适用于那些需要频繁创建和销毁,但资源消耗较大的对象,如...
单例模式是一种常用的设计模式,它的核心思想是在整个应用程序中,一个类只能有一个实例存在。单例模式常用于控制资源的共享,例如数据库连接池、日志服务等。单例模式有多种实现方式,常见的包括懒汉式、饿汉式以及...
单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。在C++中,实现单例模式有多种方法,我们将会深入探讨这一模式的原理、优缺点以及如何在实际编程中应用。 单例模式的核心在于...