Prepared Statements

A prepared statement templatizes multiple SQL statements in which only parameters are different. It separates the SQL statements from the parameters. You can use it to improve the following aspects of SQL statements:

  • Security: Because parameters and statements are separated, the risk of SQL injection attacks is avoided.
  • Performance: Because the statement is parsed in advance on the TiDB server, only parameters are passed for subsequent executions, saving the cost of parsing the entire SQL statements, splicing SQL statement strings, and network transmission.

In most applications, SQL statements can be enumerated. You can use a limited number of SQL statements to complete data queries for the entire application. So using a prepared statement is a best practice.

SQL syntax

This section describes the SQL syntax for creating, running and deleting a prepared statement.

Create a prepared statement

  1. PREPARE {prepared_statement_name} FROM '{prepared_statement_sql}';
Parameter NameDescription
{prepared_statement_name}name of the prepared statement
{prepared_statement_sql}the prepared statement SQL with a question mark as a placeholder

See PREPARE statement for more information.

Use the prepared statement

A prepared statement can only use user variables as parameters, so use the SET statement to set the variables before the EXECUTE statement can call the prepared statement.

  1. SET @{parameter_name} = {parameter_value};
  2. EXECUTE {prepared_statement_name} USING @{parameter_name};
Parameter NameDescription
{parameter_name}user variable name
{parameter_value}user variable value
{prepared_statement_name}The name of the preprocessing statement, which must be the same as the name defined in the Create a prepared statement

See the EXECUTE statement for more information.

Delete the prepared statement

  1. DEALLOCATE PREPARE {prepared_statement_name};
Parameter NameDescription
{prepared_statement_name}The name of the preprocessing statement, which must be the same as the name defined in the Create a prepared statement

See the DEALLOCATE statement for more information.

Examples

This section describes two examples of prepared statements: SELECT data and INSERT data.

SELECT example

For example, you need to query a book with id = 1 in the bookshop application.

  • SQL
  • Java
  1. PREPARE `books_query` FROM 'SELECT * FROM `books` WHERE `id` = ?';

Running result:

  1. Query OK, 0 rows affected (0.01 sec)
  1. SET @id = 1;

Running result:

  1. Query OK, 0 rows affected (0.04 sec)
  1. EXECUTE `books_query` USING @id;

Running result:

  1. +---------+---------------------------------+--------+---------------------+-------+--------+
  2. | id | title | type | published_at | stock | price |
  3. +---------+---------------------------------+--------+---------------------+-------+--------+
  4. | 1 | The Adventures of Pierce Wehner | Comics | 1904-06-06 20:46:25 | 586 | 411.66 |
  5. +---------+---------------------------------+--------+---------------------+-------+--------+
  6. 1 row in set (0.05 sec)
  1. // ds is an entity of com.mysql.cj.jdbc.MysqlDataSource
  2. try (Connection connection = ds.getConnection()) {
  3. PreparedStatement preparedStatement = connection.prepareStatement("SELECT * FROM `books` WHERE `id` = ?");
  4. preparedStatement.setLong(1, 1);
  5. ResultSet res = preparedStatement.executeQuery();
  6. if(!res.next()) {
  7. System.out.println("No books in the table with id 1");
  8. } else {
  9. // got book's info, which id is 1
  10. System.out.println(res.getLong("id"));
  11. System.out.println(res.getString("title"));
  12. System.out.println(res.getString("type"));
  13. }
  14. } catch (SQLException e) {
  15. e.printStackTrace();
  16. }

INSERT example

Using the books table as an example, you need to insert a book with title = TiDB Developer Guide, type = Science & Technology, stock = 100, price = 0.0, and published_at = NOW() (current time of insertion). Note that you don’t need to specify the AUTO_RANDOM attribute in the primary key of the books table. For more information about inserting data, see Insert Data.

  • SQL
  • Java
  1. PREPARE `books_insert` FROM 'INSERT INTO `books` (`title`, `type`, `stock`, `price`, `published_at`) VALUES (?, ?, ?, ?, ?);';

Running result:

  1. Query OK, 0 rows affected (0.03 sec)
  1. SET @title = 'TiDB Developer Guide';
  2. SET @type = 'Science & Technology';
  3. SET @stock = 100;
  4. SET @price = 0.0;
  5. SET @published_at = NOW();

Running result:

  1. Query OK, 0 rows affected (0.04 sec)
  1. EXECUTE `books_insert` USING @title, @type, @stock, @price, @published_at;

Running result:

  1. Query OK, 1 row affected (0.03 sec)
  1. try (Connection connection = ds.getConnection()) {
  2. String sql = "INSERT INTO `books` (`title`, `type`, `stock`, `price`, `published_at`) VALUES (?, ?, ?, ?, ?);";
  3. PreparedStatement preparedStatement = connection.prepareStatement(sql);
  4. preparedStatement.setString(1, "TiDB Developer Guide");
  5. preparedStatement.setString(2, "Science & Technology");
  6. preparedStatement.setInt(3, 100);
  7. preparedStatement.setBigDecimal(4, new BigDecimal("0.0"));
  8. preparedStatement.setTimestamp(5, new Timestamp(Calendar.getInstance().getTimeInMillis()));
  9. preparedStatement.executeUpdate();
  10. } catch (SQLException e) {
  11. e.printStackTrace();
  12. }

As you can see, JDBC helps you control the life cycle of prepared statements and you don’t need to manually create, use, or delete prepared statements in your application. However, note that because TiDB is compatible with MySQL, the default configuration for using MySQL JDBC Driver on the client-side is not to enable the server-side prepared statement option, but to use the client-side prepared statement.

The following configurations help you use the TiDB server-side prepared statements under JDBC:

ParameterMeansRecommended ScenarioRecommended Configuration
useServerPrepStmtsWhether to use the server side to enable prepared statementsWhen you need to use a prepared statement more than oncetrue
cachePrepStmtsWhether the client caches prepared statementsuseServerPrepStmts=truetrue
prepStmtCacheSqlLimitMaximum size of a prepared statement (256 characters by default)When the prepared statement is greater than 256 charactersConfigured according to the actual size of the prepared statement
prepStmtCacheSizeMaximum number of prepared statements (25 by default)When the number of prepared statements is greater than 25Configured according to the actual number of prepared statements

The following is a typical scenario of JDBC connection string configurations. Host: 127.0.0.1, Port: 4000, User name: root, Password: null, Default database: test:

  1. jdbc:mysql://127.0.0.1:4000/test?user=root&useConfigs=maxPerformance&useServerPrepStmts=true&prepStmtCacheSqlLimit=2048&prepStmtCacheSize=256&rewriteBatchedStatements=true&allowMultiQueries=true

You can also see the insert rows chapter if you need to change other JDBC parameters when inserting data.

For a complete example in Java, see: