【小家java】Java中Future模式衍生出来的高级应用---自己手写一个数据库连接池

简介: 【小家java】Java中Future模式衍生出来的高级应用---自己手写一个数据库连接池

说在前面


阅读本文之前,请确保你已经了解了Java中Future模式。若没有,ring出门左转,参考博文:【小家java】一个例子让就能你彻底理解Java的Future模式,Future类的设计思想


手动写一个数据库连接池


要求:该连接池能够复用数据库连接,并且能在高并发情况下正常工作


package test;
import java.util.concurrent.ConcurrentHashMap;
public class ConnectionPool {
    private ConcurrentHashMap<String, Connection> pool = new ConcurrentHashMap<String, Connection>();
    public Connection getConnection(String key) {
        Connection conn = null;
        if (pool.containsKey(key)) {
            conn = pool.get(key);
        } else {
            conn = createConnection();
            pool.putIfAbsent(key, conn);
        }
        return conn;
    }
    public Connection createConnection() {
        return new Connection();
    }
    class Connection {}
}


我们用了ConcurrentHashMap,这样就不必把getConnection方法置为synchronized,当多个线程同时调用getConnection方法时,性能大幅提升。


难道就这么简单?那你就too young too simple了。


比如我们现在有如下一个场景:


   某一时刻,同时有3个线程进入getConnection方法,调用pool.containsKey(key)都返回false,然后3个线程各自都创建了连接。虽然ConcurrentHashMap的put方法只会加入其中一个,但还是生成了2个多余的连接。如果是真正的数据库连接,那会造成极大的资源浪费。


所以我们的问题来了,为了减少资源的浪费,需要解决如何在多线程访问getConnection方法时,只执行一次createConnection。


结合之前Future模式的实现分析:当3个线程都要创建连接的时候,如果只有一个线程执行createConnection方法创建一个连接,其它2个线程只需要用这个连接就行了。再延伸,把createConnection方法放到一个Callable的call方法里面,然后生成FutureTask。我们只需要让一个线程执行FutureTask的run方法,其它的线程只执行get方法就好了。

啥都不说了,上代码:


package test;
import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
public class ConnectionPool {
    private ConcurrentHashMap<String, FutureTask<Connection>> pool = new ConcurrentHashMap<String, FutureTask<Connection>>();
    public Connection getConnection(String key) throws InterruptedException, ExecutionException {
        FutureTask<Connection> connectionTask = pool.get(key);
        if (connectionTask != null) {
            return connectionTask.get();
        } else {
            //有可能多个线程同时进入从而创建多个callable,但木有关系
            Callable<Connection> callable = new Callable<Connection>() {
                @Override
                public Connection call() throws Exception {
                    return createConnection();
                }
            };
            FutureTask<Connection> newTask = new FutureTask<Connection>(callable);
      //因为使用的ConcurrentHashMap,所以这里最终只会put成功有一个线程的数据。当put成功之后会返回v。如果返回的v是null,
            connectionTask = pool.putIfAbsent(key, newTask);
            if (connectionTask == null) {
                connectionTask = newTask;
                connectionTask.run();
            }
            return connectionTask.get();
        }
    }
    public Connection createConnection() {
        return new Connection();
    }
    class Connection {
    }
}


有必要解释下putIfAbsent():如果指定的键未与某个值关联(或映射到null),则将其与给定值关联并返回null,否则返回当前值。 简单的若,如果key不存在就把值放进去并且返回null,如果key存在,就不放进去并且返回


 public static void main(String[] args) {
        Map<String,String> map = new HashMap<>();
        System.out.println(map.put("a","a")); // null
        System.out.println(map.putIfAbsent("b","b")); //null
        System.out.println(map.putIfAbsent("b","c")); //b 注意此处返回值为b而不是c
        //put的用法  和putIfAbsent有些许区别
        System.out.println(map.put("c","c")); //null
        System.out.println(map.put("d","d")); //null
        System.out.println(map.put("c","d")); //c 注意此处返回值为c而不是d
    }


由此课件,map的put/putIfAbsent成功后的返回值,返回的是oldValue,而不是新的值。这点各位自己看看源码就一目了然了,有时候是需要注意返回值的


咱们模拟这上面的情况,推演一遍:


     当3个线程同时进入else语句块时,各自都创建了一个FutureTask,但是ConcurrentHashMap只会加入其中一个。第一个线程执行pool.putIfAbsent方法后返回null,然后connectionTask被赋值,接着就执行run方法去创建连接,最后get。后面的线程执行pool.putIfAbsent方法不会返回null,就只会执行get方法。


在并发的环境下,通过FutureTask作为中间转换,成功实现了让某个方法只被一个线程执行。

相关文章
|
5天前
|
XML 存储 Java
11:Servlet中初始化参数的获取与应用-Java Web
11:Servlet中初始化参数的获取与应用-Java Web
18 3
|
3天前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能对比
【5月更文挑战第4天】在移动开发的世界中,性能一直是衡量应用质量的重要指标。随着Kotlin的兴起,许多Android开发者开始考虑是否应该从传统的Java迁移到Kotlin。本文通过深入分析两者在Android平台上的性能差异,帮助开发者理解Kotlin在实际项目中的表现,并提供选择编程语言时的参考依据。
17 5
|
5天前
|
安全 IDE Java
Java串口通信技术探究2:RXTX库单例测试及应用
Java串口通信技术探究2:RXTX库单例测试及应用
23 4
|
5天前
|
设计模式 前端开发 Java
19:Web开发模式与MVC设计模式-Java Web
19:Web开发模式与MVC设计模式-Java Web
15 4
|
5天前
|
设计模式 存储 前端开发
18:JavaBean简介及其在表单处理与DAO设计模式中的应用-Java Web
18:JavaBean简介及其在表单处理与DAO设计模式中的应用-Java Web
22 4
|
5天前
|
SQL Java 数据库连接
17:数据库连接池与Servlet整合-Java Web
17:数据库连接池与Servlet整合-Java Web
17 3
|
5天前
|
存储 前端开发 安全
13:会话跟踪技术Session的深度应用与实践-Java Web
13:会话跟踪技术Session的深度应用与实践-Java Web
20 3
|
5天前
|
存储 前端开发 搜索推荐
12:会话跟踪技术Cookie的深度应用与实践-Java Web
12:会话跟踪技术Cookie的深度应用与实践-Java Web
19 4
|
6天前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【5月更文挑战第1天】 在移动开发的世界中,性能优化始终是开发者关注的焦点。随着Kotlin的兴起,许多团队和开发者面临着一个选择:是坚持传统的Java语言,还是转向现代化、更加简洁的Kotlin?本文通过深入分析和对比Kotlin与Java在Android应用开发中的性能表现,揭示两者在编译效率、运行速度和内存消耗等方面的差异。我们将探讨如何根据项目需求和团队熟悉度,选择最适合的语言,以确保应用的高性能和流畅体验。
|
6天前
|
安全 Java 程序员
Java并发编程:理解并应用ReentrantLock
【4月更文挑战第30天】 在多线程的世界中,高效且安全地管理共享资源是至关重要的。本文深入探讨了Java中的一种强大同步工具——ReentrantLock。我们将从其设计原理出发,通过实例演示其在解决并发问题中的实际应用,以及如何比传统的synchronized关键字提供更灵活的锁定机制。文章还将讨论在使用ReentrantLock时可能遇到的一些挑战和最佳实践,帮助开发者避免常见陷阱,提高程序性能和稳定性。