1.概述
DBCP的poolPreparedStatements是在每个connection中缓存一定数量的preparestatement,用于减少driver的statement对象创建,和数据库的网络交互以及数据库的cursor创建。
根据我们的测试结果,基本是节省1ms左右。
但在使用过程中遇到两个诡异问题,均为开源软件的bug,这里记录下排查过程。
2.问题1
1. 并发情况下的PreparedStatement is closed.(spring的bug)
2.1 现象
在一个高并发的性能测试过程中,发现如下问题:
1. Cause: java.sql.SQLException: org.apache.commons.dbcp.PoolablePreparedStatement is closed.; nested exception is
2. n.jdbc.exception.NestedSQLException:
3. --- The error occurred while applying a parameter map.
4. --- Check the CALL_INFO.SELECT_CALLINFO_BY_LOGINID-InlineParameterMap.
5. --- Check the parameter mapping for the 'loginIds[0]' property.
6. --- Cause: java.sql.SQLException: org.apache.commons.dbcp.PoolablePreparedStatement is closed.
配置如下:
1. <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
2. <property name="maxActive" value="20"/>
3. <property name="poolPreparedStatements" value="true"></property>
4. <property name="maxOpenPreparedStatements" value="50"></property>
5. </bean>
使用30个线程并发压测时发现。
2.2 原因
在spring的SqlMapClientTemplate的execute方法的finally中,先执行springCon.close()然后执行session.close().
资源这样释放是有问题的,原因如下:
connection关闭过程如下:
1. springCon.close()->DataSourceUtils.doReleaseConnection->PoolGuardConnectionWrapper.close()
2. ->PoolableConnection.close()->_pool.returnObject(this)->_factory.passivateObject(obj)->DelegatingConnection.passivate()
statment关闭过程如下:
1. DelegatingStatement.close()->PoolablePreparedStatement.close()->statPool.returnObj()->
2. factory.passivateObject(key,obj)-> ((DelegatingPreparedStatement)obj).passivate()->closed=true
在线程A的connection的关闭过程中,会触发对statment的关闭 (DelegatingConnection.passivate()中),这样我们的connection和statment都被归还到连接池了,然后 由于高并发(而且连接池默认是FILO),这个连接很快又被借出去使用(线程B),使用到一半,线程A的session.close()触发了,把 statment的状态又置为了close,这样就出问题了。简单的说,是因为一个归还的连接池中的statment,还有一个外部引用持有并发生关闭操 作导致的。
DelegatingPreparedStatement代码:
修复后的版本是先执行session.close()然后执行springCon.close()就可以了。
2.3 排查过程
这个问题的排查过程相对简单:
1.和性能测试同学发现问题后,先定位是并发略大于池大小的时候出现,由此基本断定是并发导致的问题。
2.google搜索相关报错,发现https://jira.springsource.org/browse/SPR-7760描述基本符合我们遇到的问题
3.在开发环境重现该问题(尽量用最小依赖的代码,单元测试或者main函数来重现)
4.分析原因,这是这次排查中最复杂的一步,因为原因还是比较隐蔽。基本上分析的思路如下:先看报错的触发条件,在看该触发条件的改变入口点,然后再考虑程序执行过程中调用该入口点的几个地方,结合上面多线程的初步判断来分析。
2.问题2
1. 偶发性报“关闭的语句"问题,commons-pool 1.3 的bug
2.1 现象
线上环境偶尔出现,测试环境不能重现
1. Caused by: java.sql.SQLException: 关闭的语句
2. at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:112)
3. at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:146)
2.2 原因
commons-pool的GenericKeyedObjectPool结构大概是 HashMap<Object,LinkedList<ObjectTimestampPair>>, 也就是说,每个key可以有多个元素,用list连起来(maxactive代表list大小控制,maxtotal代表所有的原始,比如3个key,一 个key有2个元素,就是6) 在returnObject的时候,会将元素封装为ObjectTimestampPair放入:
1. LinkedList pool = (LinkedList) (_poolMap.get(key));
2. ...
3. pool.addLast(new
而ObjectTimestampPair结构如下:
1. Object value;
2. long
3.
4. ObjectTimestampPair(Object val) {
5. this(val, System.currentTimeMillis());
6. }
其目的是保存每个元素的最后归还时间,这样在需要的时候可以逐出最久没有使用的(LRU),事实上这个过程发生在borrowObject的时候:
1. if ((_maxTotal > 0) && (_totalActive + _totalIdle >= _maxTotal)) {
2. clearOldest();
3. }
然后就是最重要的clearOldest方法:
1. public synchronized void
2. // build sorted map of idle objects
3. new
4. for
5. Object key = keyiter.next();
6. LinkedList list = (LinkedList) _poolMap.get(key);
7. for
8. // each item into the map uses the objectimestamppair object
9. // as the key. It then gets sorted based on the timstamp field
10. // each value in the map is the parent list it belongs in.
11. ObjectTimestampPair pair = (ObjectTimestampPair) it.next();
12. map.put(pair, key);
13. }
14. }
15.
16. // Now iterate created map and kill the first 15% plus one to account for zero
17. Set setPairKeys = map.entrySet();
18. int itemsToRemove = ((int) (map.size() * 0.15)) + 1;
19.
20. Iterator iter = setPairKeys.iterator();
21. while (iter.hasNext() && itemsToRemove > 0) {
22. Map.Entry entry = (Map.Entry) iter.next();
23. // kind of backwards on naming. In the map, each key is the objecttimestamppair
24. // because it has the ordering with the timestamp value. Each value that the
25. // key references is the key of the list it belongs to.
26. Object key = entry.getValue();
27. ObjectTimestampPair pairTimeStamp = (ObjectTimestampPair) entry.getKey();
28. LinkedList list = (LinkedList) _poolMap.get(key);
29. list.remove(pairTimeStamp);
30.
31. try
32. _factory.destroyObject(key, pairTimeStamp.value);
33. catch
34. // ignore error, keep destroying the rest
35. }
36. // if that was the last object for that key, drop that pool
37. if
38. _poolMap.remove(key);
39. }
40. _totalIdle--;
41. itemsToRemove--;
42. }
43. notifyAll();
44. }
可以看到,这个方法主要过程是将所有元素取出来,放到一个treemap中(按归还时间排序),然后根据逐出的比例,取出部分的entry,并取得 在池中的key(sql生成的那个key),然后根据key取出对应的list,将该ObjectTimestampPair从list中移除,同时调用 _factory.destroyObject关闭statement。
问题就是关闭statement的操作成功了,但是从pool中移除的动作失败了,为什么呢,是因为在map.put(pair, key)的过程中出错了。注意这里的map是个treeMap,它的key比较是使用元素的compareTo的(参考Treemap),而如果我们有同 一个connection的两个statment是在同一ms内放回来的,那么他们的tstamp就会相同,而且ObjectTimestampPair 的comparTo方法如下:
1. public int
2. return (int) (this.tstamp - other.tstamp);
3. }
因此,在map.put(pair, key)动作的时候,过程是这样的,
1. put(pair1,key1)
2. put(pair2,key2)
因为Treemap的put是:
1. int
2. if (diff == 0) return
而正好我们的pair1和pair2的tstamp又相等,当pair2放进去的时候,就只更新了value,没有更新key。因此map中就是
(pair1,key2),这样在后续的过程中,取出key2的list,然后执行remove(pair1)的操作,肯定是不会移除的 (pair1不在key2的list中),但是_factory.destroyObject(key, pairTimeStamp.value);方法确是实实在在的执行了:
1. public void destroyObject(Object key, Object obj) throws
2. if(obj instanceof
3. ((DelegatingPreparedStatement)obj).getInnermostDelegate().close();
4. else
5. ((PreparedStatement)obj).close();
6. }
7. }
因此,池中就多了一个关闭后的连接,再次使用时报错。
2.3 排查过程
1.这个排查过程相当纠结,花了很久的时间。
2.首先是问题重现,普通压测不能重现,初步定位可能是环境不复杂或者环境配置不同造成的,环境唯一可能不同的就是数据库端配置,但通过看了下 jdbc的driver的源码,抛错的地方是一个状态位,而改变这个状态位的入口肯定是要app端去执行一下的,一个statment和 connection放在池中,没有线程处理的时候,是没有办法改变这个状态位的,而如果是使用过程中出问题,应该在不缓存的时候也会出错,所以基本排除 数据库配置的猜想。
3.那还有一个可能就是出错环境的访问和sql执行比较复杂,这个复杂性最主要的是影响statement的逐出(超过50个就出触发 clearoldest),那么我们如果要想模拟这样的情况,就应该减小pool的大小,于是减小到1,用两个不同sql来交替执行(并启多个线程),没 有重现。
4.上面这条路不通,换了一个思路,用btrace在出错环境监控jdbc的statement的关闭,观察发现所有的关闭都是在 clearOldest中触发的,由于有问题1的经验,又怀疑是多线程或者资源关闭顺序造成,于是用btrace监控连接的brrow和return,没 有发现问题。
5.和老大讨论,结合前面的分析,初步还是觉得在逐出的地方(clearOldest)出了问题,再考虑到pool大小为1可能比较极端,将pool大小设置为2,sql换为3个,重现了问题!
6.重现后再次重btrace观察池的借出,归还和逐出,发现所有操作都是正常的,而且由于抛异常后不会从池中清除(这里是有点问题的,应该清除掉),所以每次重现刚刚出问题的瞬间都要重启,浪费了较多时间。
7.既然初步定位了问题,考虑用java代码来重现,而不是起服务器,结果比较顺利,也成功重现了。然后开始排除无关因素,发现不需要多线程,单线程也可以重现,于是确定和并发无关。
8.在3个sql之间插入一些sleep,发现在sql1和2之间插入sleep后问题消失,而sql1,2执行时pool还没有满,不会触发clearoldest,又纠结了。
9.尝试debug下能否重现,开始以debug就重现不了了,结合上面分析,应该和时间有关,于是为了避免在sql1和2间停顿,只在clearoldest打了断点,然后先只是在断点停一下,看能否重现。果然重现了,于是开始看debug的内容,发现了具体原因。
commons-pool 在1.5.4修复了该问题。
3.其他一些相关点
- 一般来说:driver.createConnection()花费几百到1000ms左右;一个connection第一次ps.exeute()的时候,数据库有一些初始化工作,需要100ms左右后续的使用就是正常的数据库查询时间了。
- oracle的sql解析和preparestatement基本无关的,只要sql(或者statement中的str)相同就可以,是一个全局缓存,LRU逐出。
- oracle的max-open-cursor是instance级别的,多个schema共用,要考虑缓存后cursor太多的情况,20个conn,一个缓存50个statement,就有1000个了。
- 第一次执行一个sql是硬解析,然后再缓存逐出前是软解析,一个session中(对应app的connection)是软软解析,缓存了就是open cursor,层层变快。查询结果不会缓存。但从磁盘角度会有块的缓存,如果近期又查询的话。
- driver和db间一次网络交互的大小有fetchsize决定,oracle driver默认是10. 那么preparestatement的缓存buf大小就是一行大小*fetchsize个byte。太大会占用较多内存。
4.参考文献
1.DBCP浅析: http://zldeng1984.iteye.com/admin/blogs/1368384
2.http://zldeng1984.iteye.com/blog/1500140