File indexing completed on 2024-09-08 10:13:02
0001 /* This file is part of the KDE project 0002 Copyright (C) 2005-2016 Jarosław Staniek <staniek@kde.org> 0003 0004 This library is free software; you can redistribute it and/or 0005 modify it under the terms of the GNU Library General Public 0006 License as published by the Free Software Foundation; either 0007 version 2 of the License, or (at your option) any later version. 0008 0009 This library is distributed in the hope that it will be useful, 0010 but WITHOUT ANY WARRANTY; without even the implied warranty of 0011 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 0012 Library General Public License for more details. 0013 0014 You should have received a copy of the GNU Library General Public License 0015 along with this library; see the file COPYING.LIB. If not, write to 0016 the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 0017 * Boston, MA 02110-1301, USA. 0018 */ 0019 0020 #ifndef KDB_PREPAREDSTATEMENT_H 0021 #define KDB_PREPAREDSTATEMENT_H 0022 0023 #include <QVariant> 0024 #include <QStringList> 0025 #include <QSharedData> 0026 0027 #include "KDbField.h" 0028 #include "KDbResult.h" 0029 0030 class KDbFieldList; 0031 class KDbPreparedStatementInterface; 0032 0033 //! Prepared statement paraneters used in KDbPreparedStatement::execute() 0034 typedef QList<QVariant> KDbPreparedStatementParameters; 0035 0036 /*! @short Prepared database command for optimizing sequences of multiple database actions 0037 0038 Currently INSERT and SELECT statements are supported. 0039 For example when using KDbPreparedStatement for INSERTs, 0040 you can gain about 30% speedup compared to using multiple 0041 connection.insertRecord(*tabelSchema, dbRecordBuffer). 0042 0043 To use KDbPreparedStatement, create is using KDbConnection:prepareStatement(), 0044 providing table schema; set up parameters using operator << ( const QVariant& value ); 0045 and call execute() when ready. KDbPreparedStatement objects are accessed 0046 using KDE shared pointers, i.e KDbPreparedStatement, so you do not need 0047 to remember about destroying them. However, when underlying KDbConnection object 0048 is destroyed, KDbPreparedStatement should not be used. 0049 0050 Let's assume tableSchema contains two columns NUMBER integer and TEXT text. 0051 Following code inserts 10000 records with random numbers and text strings 0052 obtained elsewhere using getText(i). 0053 @code 0054 bool insertMultiple(KDbConnection* conn, KDbTableSchema* tableSchema) 0055 { 0056 KDbPreparedStatement statement = conn->prepareStatement( 0057 KDbPreparedStatement::Insert, tableSchema); 0058 for (i=0; i<10000; i++) { 0059 KDbPreparedStatementParameters parameters; 0060 parameters << qrand() << getText(i); 0061 if (!statement.execute(parameters)) 0062 return false; 0063 } 0064 return true; 0065 } 0066 @endcode 0067 0068 If you do not call clearParameters() after every insert, you can insert 0069 the same value multiple times using execute() what increases efficiency even more. 0070 0071 Another use case is inserting large objects (BLOBs or CLOBs). 0072 Depending on database backend, you can avoid escaping BLOBs. 0073 See KexiFormView::storeData() for example use. 0074 */ 0075 class KDB_EXPORT KDbPreparedStatement : public KDbResultable 0076 { 0077 public: 0078 0079 //! Defines type of the prepared statement. 0080 enum Type { 0081 InvalidStatement, //!< Used only in invalid statements 0082 SelectStatement, //!< SELECT statement will be prepared end executed 0083 InsertStatement //!< INSERT statement will be prepared end executed 0084 }; 0085 0086 //! @internal 0087 class KDB_EXPORT Data : public QSharedData { 0088 public: 0089 Data(); 0090 Data(Type _type, KDbPreparedStatementInterface* _iface, KDbFieldList* _fields, 0091 const QStringList& _whereFieldNames); 0092 ~Data(); 0093 Type type; 0094 KDbFieldList *fields; 0095 QStringList whereFieldNames; 0096 const KDbField::List* fieldsForParameters; //!< fields where we'll put the inserted parameters 0097 KDbField::List* whereFields; //!< temporary, used for select statements, based on whereFieldNames 0098 bool dirty; //!< true if the statement has to be internally 0099 //!< prepared (possible again) before calling executeInternal() 0100 KDbPreparedStatementInterface *iface; 0101 quint64 lastInsertRecordId; 0102 }; 0103 0104 //! Creates an invalid prepared statement. 0105 KDbPreparedStatement(); 0106 0107 ~KDbPreparedStatement() override; 0108 0109 bool isValid() const; 0110 0111 KDbPreparedStatement::Type type() const; 0112 0113 void setType(KDbPreparedStatement::Type type); 0114 0115 const KDbFieldList* fields() const; 0116 0117 //! Sets fields for the statement. Does nothing if @a fields is @c nullptr. 0118 void setFields(KDbFieldList* fields); 0119 0120 QStringList whereFieldNames() const; 0121 0122 void setWhereFieldNames(const QStringList& whereFieldNames); 0123 0124 /*! Executes the prepared statement using @a parameters parameters. 0125 A number parameters set up for the statement must be the same as a number of fields 0126 defined in the underlying database table. 0127 @return false on failure. Detailed error status can be obtained 0128 from KDbConnection object that was used to create this statement object. */ 0129 bool execute(const KDbPreparedStatementParameters& parameters); 0130 0131 /*! @return unique identifier of the most recently inserted record. 0132 Typically this is just primary key value. This identifier could be reused when we want 0133 to reference just inserted record. If there was no insertion recently performed, 0134 std::numeric_limits<quint64>::max() is returned. */ 0135 quint64 lastInsertRecordId() const; 0136 0137 protected: 0138 //! Creates a new prepared statement. In your code use 0139 //! Users call KDbConnection:prepareStatement() instead. 0140 KDbPreparedStatement(KDbPreparedStatementInterface* iface, Type type, 0141 KDbFieldList* fields, 0142 const QStringList& whereFieldNames = QStringList()); 0143 0144 friend class KDbConnection; 0145 0146 private: 0147 //! @todo is this portable across backends? 0148 bool generateStatementString(KDbEscapedString* s); 0149 bool generateSelectStatementString(KDbEscapedString * s); 0150 bool generateInsertStatementString(KDbEscapedString * s); 0151 0152 QSharedDataPointer<Data> d; 0153 }; 0154 0155 #endif