Open Source Repository

Home /spring/spring-jdbc-3.0.5 | Repository Home



org/springframework/jdbc/support/lob/LobHandler.java
/*
 * Copyright 2002-2009 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.jdbc.support.lob;

import java.io.InputStream;
import java.io.Reader;
import java.sql.ResultSet;
import java.sql.SQLException;

/**
 * Abstraction for handling large binary fields and large text fields in
 * specific databases, no matter if represented as simple types or Large OBjects.
 * Its main purpose is to isolate Oracle's peculiar handling of LOBs in
 {@link OracleLobHandler}; most other databases should be able to work
 * with the provided {@link DefaultLobHandler}.
 *
 <p>Provides accessor methods for BLOBs and CLOBs, and acts as factory for
 * LobCreator instances, to be used as sessions for creating BLOBs or CLOBs.
 * LobCreators are typically instantiated for each statement execution or for
 * each transaction; they are not thread-safe because they might track
 * allocated database resources in order to free them after execution.
 *
 <p>Most databases/drivers should be able to work with {@link DefaultLobHandler},
 * which by default delegates to JDBC's direct accessor methods, avoiding the
 <code>java.sql.Blob</code> and <code>java.sql.Clob</code> API completely.
 {@link DefaultLobHandler} can also be configured to access LOBs using
 <code>PreparedStatement.setBlob/setClob</code> (e.g. for PostgreSQL), through
 * setting the {@link DefaultLobHandler#setWrapAsLob "wrapAsLob"} property.
 *
 <p>Unfortunately, Oracle 9i just accepts Blob/Clob instances created via its own
 * proprietary BLOB/CLOB API, and additionally doesn't accept large streams for
 * PreparedStatement's corresponding setter methods. Therefore, you need to use
 {@link OracleLobHandler} there, which uses Oracle's BLOB/CLOB API for both types
 * of access. The Oracle 10g JDBC driver should basically work with
 {@link DefaultLobHandler} as well, with some limitations in terms of LOB sizes.
 *
 <p>Of course, you need to declare different field types for each database.
 * In Oracle, any binary content needs to go into a BLOB, and all character content
 * beyond 4000 bytes needs to go into a CLOB. In MySQL, there is no notion of a
 * CLOB type but rather a LONGTEXT type that behaves like a VARCHAR. For complete
 * portability, use a LobHandler for fields that might typically require LOBs on
 * some database because of the field size (take Oracle's numbers as a guideline).
 *
 <p><b>Summarizing the recommended options (for actual LOB fields):</b>
 <ul>
 <li><b>JDBC 4.0 driver:</b> {@link DefaultLobHandler} with <code>streamAsLob=true</code>.
 <li><b>PostgreSQL:</b> {@link DefaultLobHandler} with <code>wrapAsLob=true</code>.
 <li><b>Oracle 9i/10g:</b> {@link OracleLobHandler} with a connection-pool-specific
 {@link OracleLobHandler#setNativeJdbcExtractor NativeJdbcExtractor}.
 <li>For all other database drivers (and for non-LOB fields that might potentially
 * turn into LOBs on some databases): a plain {@link DefaultLobHandler}.
 </ul>
 *
 @author Juergen Hoeller
 @since 23.12.2003
 @see DefaultLobHandler
 @see OracleLobHandler
 @see java.sql.ResultSet#getBlob
 @see java.sql.ResultSet#getClob
 @see java.sql.ResultSet#getBytes
 @see java.sql.ResultSet#getBinaryStream
 @see java.sql.ResultSet#getString
 @see java.sql.ResultSet#getAsciiStream
 @see java.sql.ResultSet#getCharacterStream
 */
public interface LobHandler {

  /**
   * Retrieve the given column as bytes from the given ResultSet.
   * Might simply invoke <code>ResultSet.getBytes</code> or work with
   <code>ResultSet.getBlob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnName the column name to use
   @return the content as byte array, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getBytes
   */
  byte[] getBlobAsBytes(ResultSet rs, String columnNamethrows SQLException;

  /**
   * Retrieve the given column as bytes from the given ResultSet.
   * Might simply invoke <code>ResultSet.getBytes</code> or work with
   <code>ResultSet.getBlob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnIndex the column index to use
   @return the content as byte array, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getBytes
   */
  byte[] getBlobAsBytes(ResultSet rs, int columnIndexthrows SQLException;

  /**
   * Retrieve the given column as binary stream from the given ResultSet.
   * Might simply invoke <code>ResultSet.getBinaryStream</code> or work with
   <code>ResultSet.getBlob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnName the column name to use
   @return the content as binary stream, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getBinaryStream
   */
  InputStream getBlobAsBinaryStream(ResultSet rs, String columnNamethrows SQLException;

  /**
   * Retrieve the given column as binary stream from the given ResultSet.
   * Might simply invoke <code>ResultSet.getBinaryStream</code> or work with
   <code>ResultSet.getBlob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnIndex the column index to use
   @return the content as binary stream, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getBinaryStream
   */
  InputStream getBlobAsBinaryStream(ResultSet rs, int columnIndexthrows SQLException;

  /**
   * Retrieve the given column as String from the given ResultSet.
   * Might simply invoke <code>ResultSet.getString</code> or work with
   <code>ResultSet.getClob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnName the column name to use
   @return the content as String, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getString
   */
  String getClobAsString(ResultSet rs, String columnNamethrows SQLException;

  /**
   * Retrieve the given column as String from the given ResultSet.
   * Might simply invoke <code>ResultSet.getString</code> or work with
   <code>ResultSet.getClob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnIndex the column index to use
   @return the content as String, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getString
   */
  String getClobAsString(ResultSet rs, int columnIndexthrows SQLException;

  /**
   * Retrieve the given column as ASCII stream from the given ResultSet.
   * Might simply invoke <code>ResultSet.getAsciiStream</code> or work with
   <code>ResultSet.getClob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnName the column name to use
   @return the content as ASCII stream, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getAsciiStream
   */
  InputStream getClobAsAsciiStream(ResultSet rs, String columnNamethrows SQLException;

  /**
   * Retrieve the given column as ASCII stream from the given ResultSet.
   * Might simply invoke <code>ResultSet.getAsciiStream</code> or work with
   <code>ResultSet.getClob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnIndex the column index to use
   @return the content as ASCII stream, or <code>null</code> in case of SQL NULL
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getAsciiStream
   */
  InputStream getClobAsAsciiStream(ResultSet rs, int columnIndexthrows SQLException;

  /**
   * Retrieve the given column as character stream from the given ResultSet.
   * Might simply invoke <code>ResultSet.getCharacterStream</code> or work with
   <code>ResultSet.getClob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnName the column name to use
   @return the content as character stream
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getCharacterStream
   */
  Reader getClobAsCharacterStream(ResultSet rs, String columnNamethrows SQLException;

  /**
   * Retrieve the given column as character stream from the given ResultSet.
   * Might simply invoke <code>ResultSet.getCharacterStream</code> or work with
   <code>ResultSet.getClob</code>, depending on the database and driver.
   @param rs the ResultSet to retrieve the content from
   @param columnIndex the column index to use
   @return the content as character stream
   @throws SQLException if thrown by JDBC methods
   @see java.sql.ResultSet#getCharacterStream
   */
  Reader getClobAsCharacterStream(ResultSet rs, int columnIndexthrows SQLException;

  /**
   * Create a new {@link LobCreator} instance, i.e. a session for creating BLOBs
   * and CLOBs. Needs to be closed after the created LOBs are not needed anymore -
   * typically after statement execution or transaction completion.
   @return the new LobCreator instance
   @see LobCreator#close()
   */
  LobCreator getLobCreator();

}