zoukankan      html  css  js  c++  java
  • 连接池

    什么是连接池呢?

      我们登陆上某个网站,去修改个人信息、点击下单、修改购买商品的数量......当我们做这些操作时候,

    我们已经连接上数据库并修改、增加、删除数据库中的数据,完成操作后,关闭连接(避免占用资源)。

    如果很多人访问这个网站并进行操作,那要创建很多个连接,而且只操作一次就关闭,这会显著的影响系统的

    性能。连接池  就为解决这个难题应运而生。

    连接池: 

      在系统初始化的时候,将数据库连接作为对象存放在内存中。当用户要访问数据库时,取出一个连接,

    完成操作后归还到池中,等待下个用户使用。连接的创建、断开都由池来管理。我们可以设置连接池的参

    数控制初始连接数、连接的上下限数、最大空闲时间......这项技术明显的提高对数据库操作的性能。

    线程池:

    package org3.pool;
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    import java.sql.Connection;
    import java.util.concurrent.LinkedBlockingQueue;
    
    /**
     * Created by Administrator on 2016/11/25.
     * 编写一个线程池
     */
    public class Pool {
    
    
        private int minSize = 1;    //定义最小连接数,默认为2
        private int maxSize = 2;    //定义最大连接数,默认为4
        private int fetchCount = minSize;   //记录从数据库抓取的连接数
        private LinkedBlockingQueue<Connection> pool;   //定义一个阻塞队列,充当连接池
    
        public void setMinSize(int minSize) {
            this.minSize = minSize;
            this.fetchCount = minSize;
        }
    
        public void setMaxSize(int maxSize) {
            this.maxSize = maxSize;
        }
    
        //初始化线程池方法
        public void initPool() {
            //初始化连接池的最大值
            pool = new LinkedBlockingQueue<>( maxSize );
            //预先存放minSize个连接
            for ( int i = 0 ; i < minSize ; i++ ) {
                pool.add( DBUtil.getConnection() );
            }
        }
    
        //从数据库抓取连接方法
        public synchronized Connection getConnection() {
            //检查连接池状况,如果检查未通过将直接引发异常,程序终止
            checkFetch();
            Connection conn = null;
            try {
                conn = getProxyConnection();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return conn;
        }
    
        //检查连接池的状况
        private void checkFetch() {
            //当连接池大小为0 且 连接记录数还没达到最大值时
            if ( pool.isEmpty() && fetchCount < maxSize ) {
                pool.add( DBUtil.getConnection() );
            }
        }
    
        //从连接池获取连接,并将获取的连接创建一个代理对象
        private Connection getProxyConnection() throws InterruptedException {
            //先从池中获取未代理的连接对象
            Connection conn = pool.take();
            //将获取的链接创建代理对象
            return createProxy( conn );
        }
    
        //使用内部类来实现InvocationHandler
        private Connection createProxy( Connection conn ) {
            return ( Connection )Proxy.newProxyInstance( Connection.class.getClass().getClassLoader(), new Class[] { Connection.class }, new InvocationHandler() {
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    //如果客户端调用的是close方法,就将连接放回连接池
                    if( "close".equals(method.getName() ) ) {
                        pool.put( conn );
                        fetchCount--;
                    }
                    //如果是Connection其他的方法,将正常回调,也就是放行
                    return method.invoke(conn,args);
                }
            });
        }
    
        //查看连接池当前连接数
        public int size() {
            return pool.size();
        }
    }
    

    DBUtil:

    package org3.pool;
    
    import java.sql.*;
    
    /**
     * Created by Administrator on 2016/11/25.
     */
    public class DBUtil {
    
        private static String driver = "oracle.jdbc.driver.OracleDriver";   //数据库驱动
        private static String url = "jdbc:oracle:thin:@localhost:1521:orcl";    //数据库的资源地址
        private static String userName = "mark";    //用户名
        private static String password = "mark";    //密码
    
        static {
            try {
                Class.forName( driver );
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
            }
        }
    
        public static Connection getConnection() {
            Connection conn = null;
            try {
                conn = DriverManager.getConnection( url, userName, password );
            } catch (SQLException e) {
                e.printStackTrace();
            }
            return conn;
        }
    
    
        //main方法(用于测试)
        public static void main(String[] args) {
            System.out.println( getConnection() );
        }
    }
    

    编写一个线程:

    package org3.pool;
    
    import java.sql.Connection;
    import java.sql.SQLException;
    
    /**
     * Created by Administrator on 2016/11/26.
     * 编写线程1
     */
    public class User1 extends Thread {
    
        //定义连接池对象
        private Pool pool;
    
        public User1( Pool pool ) {
            this.pool = pool;
        }
    
        //重写Thread的run方法
        @Override
        public void run() {
            Connection conn = pool.getConnection();
            System.out.println(Thread.currentThread().getName() + "获取了一个连接:" + conn);
            try {
                Thread.sleep( 500 );
                conn.close();
                System.out.println(Thread.currentThread().getName() + "归还了一个连接: " + conn);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
    

    main方法:

    package org3.pool;
    
    import java.sql.SQLException;
    
    /**
     * Created by Administrator on 2016/11/25.
     * main方法
     */
    public class Main {
        public static void main(String[] args) throws SQLException {
            //实例化连接池对象
            Pool pool = new Pool();
            //给连接池设置属性
            pool.setMinSize( 2 );
            pool.setMaxSize( 5 );
            //初始化连接池
            pool.initPool();
    
            for( int i = 0; i < 10; i++ ){
                User1 u1 = new User1( pool );
                u1.start();
            }
    
            try {
                Thread.sleep( 10000 );
                System.out.println( "线程大小= " + pool.size() );
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
    
        }
    }
    

    运行结果:

    -------------------------------------------------------------------------

  • 相关阅读:
    [Unity3D]蓝港面试题
    BZOJ 2186 SDOI2008 沙拉公主的困惑 数论
    JSONObject与JSONArray的使用
    一个int类型究竟占多少个字节
    软件开发的金字塔
    poj 1064 Cable master ,二分 精度!!!
    php实现工厂模式
    数据库索引的作用和长处缺点
    C++中使用class和structkeyword的不同
    提交时提示错误This Bundle is invalid.New apps and app updates submitted to the App Store must be built wit
  • 原文地址:https://www.cnblogs.com/hmake/p/6107280.html
Copyright © 2011-2022 走看看