001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.dbcp2; 018 019import java.sql.PreparedStatement; 020import java.sql.SQLException; 021 022import org.apache.commons.pool2.KeyedObjectPool; 023 024/** 025 * A {@link DelegatingPreparedStatement} that cooperates with {@link PoolingConnection} to implement a pool of 026 * {@link PreparedStatement}s. 027 * <p> 028 * My {@link #close} method returns me to my containing pool. (See {@link PoolingConnection}.) 029 * </p> 030 * 031 * @param <K> 032 * the key type 033 * 034 * @see PoolingConnection 035 * @since 2.0 036 */ 037public class PoolablePreparedStatement<K> extends DelegatingPreparedStatement { 038 039 /** 040 * The {@link KeyedObjectPool} from which I was obtained. 041 */ 042 private final KeyedObjectPool<K, PoolablePreparedStatement<K>> pool; 043 044 /** 045 * My "key" as used by {@link KeyedObjectPool}. 046 */ 047 private final K key; 048 049 private volatile boolean batchAdded; 050 051 /** 052 * Constructs a new instance. 053 * 054 * @param stmt 055 * my underlying {@link PreparedStatement} 056 * @param key 057 * my key as used by {@link KeyedObjectPool} 058 * @param pool 059 * the {@link KeyedObjectPool} from which I was obtained. 060 * @param conn 061 * the {@link java.sql.Connection Connection} from which I was created 062 */ 063 public PoolablePreparedStatement(final PreparedStatement stmt, final K key, 064 final KeyedObjectPool<K, PoolablePreparedStatement<K>> pool, final DelegatingConnection<?> conn) { 065 super(conn, stmt); 066 this.pool = pool; 067 this.key = key; 068 069 // Remove from trace now because this statement will be 070 // added by the activate method. 071 removeThisTrace(conn); 072 } 073 074 @Override 075 public void activate() throws SQLException { 076 setClosedInternal(false); 077 add(getConnectionInternal(), this); 078 super.activate(); 079 } 080 081 /** 082 * Add batch. 083 */ 084 @Override 085 public void addBatch() throws SQLException { 086 super.addBatch(); 087 batchAdded = true; 088 } 089 090 /** 091 * Clear Batch. 092 */ 093 @Override 094 public void clearBatch() throws SQLException { 095 batchAdded = false; 096 super.clearBatch(); 097 } 098 099 /** 100 * Return me to my pool. 101 */ 102 @Override 103 public void close() throws SQLException { 104 // calling close twice should have no effect 105 if (!isClosed()) { 106 try { 107 pool.returnObject(key, this); 108 } catch (final SQLException | RuntimeException e) { 109 throw e; 110 } catch (final Exception e) { 111 throw new SQLException("Cannot close preparedstatement (return to pool failed)", e); 112 } 113 } 114 } 115 116 /** 117 * Package-protected for tests. 118 * 119 * @return The key. 120 */ 121 K getKey() { 122 return key; 123 } 124 125 @Override 126 public void passivate() throws SQLException { 127 // DBCP-372. clearBatch with throw an exception if called when the 128 // connection is marked as closed. 129 if (batchAdded) { 130 clearBatch(); 131 } 132 prepareToReturn(); 133 } 134}