This post explores some of the concepts of query optimization & transaction management for Android apps. Most of the applications won't be using so much of the SQLite features but it's always good to know about possible optimizations when you deal with databases. I referenced the book Pro Android Apps Performance Optimization by published Apress.
1. Basic SQLite Statements
The basic SQLite statements are simple strings like-
- CREATE TABLE phone(devicename TEXT, company TEXT)
- INSERT INTO phone VALUES('M8','HTC')
The above statements are just simple strings & they have to be interpreted,or compiled before they can be executed. The compilation of these statements is down internally when we call the .execSQL().
The following code demonstrates how these statements are executed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public void basic_queries(){ | |
SQLiteDatabase db = SQLiteDatabase.create(null); | |
db.execSQL("CREATE TABLE phone (devicename TEXT, company TEXT)"); | |
db.execSQL("INSERT INTO phone VALUES ('M8','HTC')"); | |
db.close(); // remember to close database when you’re done with it | |
} |
The problems with the above implementation is that SQLite statements takes some quite some time to execute. In addition to the compilation, the statements themselves may need to be created. String is also immutable(String is immutable means that you cannot change the object itself, but you can change the reference ) ,which poses performance issues here.
2. The INSERT Statement String
INSERT is the most used statement in any SQLite database as you are bound to insert several entries into the database.A basic implementation of inserting several values into the database is shown below.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private static final String[] phoneNames={"M8","X1","Grand"}; | |
private static final String[] compNames={"HTC","MOTO","SAMSUNG"}; | |
private final SQLiteDatabase db; | |
public void populateWithData () { | |
db = SQLiteDatabase.create(null); // memory-backed database | |
db.execSQL("CREATE TABLE phone(devicename TEXT, company TEXT)"); | |
int i = 0; | |
for (String name : phoneNames) { | |
String company = compNames[i++]; | |
String sql = "INSERT INTO phone VALUES(\"" + name + "\",\"" + company + | |
"\")"; | |
db.execSQL(sql); | |
} | |
} |
The problem with the above implementation is that for every insert statement, a String would be created and execSQL would be called.This prcedure takes a lot of time and memory.
One solution for this problem is to avoid using the + operator for String concatenation & instead use a StringBuilder object or calling String.format.The two new methods are shown below
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private static final String[] phoneNames={"M8","X1","Grand"}; | |
private static final String[] compNames={"HTC","MOTO","SAMSUNG"}; | |
private final SQLiteDatabase db; | |
public void init () { | |
db = SQLiteDatabase.create(null); // memory-backed database | |
db.execSQL("CREATE TABLE phone(devicename TEXT, company TEXT)"); | |
} | |
public void UsingStingFormat(){ | |
int i = 0; | |
for (String name : phoneNames) { | |
String company = compNames[i++]; | |
String sql = String.format("INSERT INTO phone VALUES(\"%s\",\"%s\")",name,company); | |
db.execSQL(sql); | |
} | |
} | |
public void UsingStringBuilder(){ | |
StringBuilder builder = new StringBuilder(); | |
builder.append("INSERT INTO phone VALUES(\""); | |
int resetLength = builder.length(); | |
int i = 0; | |
for (String name : phoneNames) { | |
String company = compNames[i++]; | |
builder.setLength(resetLength); // reset position | |
builder.append(name).append("\",\"").append(company).append("\")"); // chain calls | |
db.execSQL(builder.toString()); | |
} | |
} |
3. The Compile Statement
Now since we have addressed the issue of using & interpreting the sting statements, the next issue is of execSQL method.The execSQL does the actual compilation(parsing) of the statement.In the above examples the execSQL statement is called for every insertion to the database. All this can be avoided by only compiling the statement once & binding later.This is depicted below.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public void UseCompileStatement () { | |
SQLiteStatement stmt = db.compileStatement("INSERT INTO phone VALUES(?,?)"); | |
int i = 0; | |
for (String name : phoneNames) { | |
String company = compNames[i++]; | |
stmt.clearBindings(); | |
stmt.bindString(1, name); // replace first question mark with name | |
stmt.bindString(2, company); // replace second question mark with company | |
stmt.executeInsert(); | |
} | |
} |
4. Using ContentValues
As we avoided the use of execSQL staement above, the same can be achieved using a ContentValues object. Its not better than the Compile statement method. but either way its an elegant method of getting things done.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public void UseContentValues () { | |
ContentValues values = new ContentValues(); | |
int i = 0; | |
for (String name : phoneNames) { | |
String comp= compNames[i++]; | |
values.clear(); | |
values.put("devicename", name); | |
values.put("company", comp); | |
db.insert("phone", null, values); | |
} | |
} |
5. Transactions
Transactions are automatically created when we perform any read/write from the the database. But creating explicit transaction can improve the performance tremendously.The implementation of explicit transaction on compile statement is depicted below.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public void UseCompileStatementOneTransaction () { | |
try { | |
db.beginTransaction();//strat a transaction | |
SQLiteStatement stmt = db.compileStatement("INSERT INTO phone | |
VALUES(?,?)"); | |
int i = 0; | |
for (String name : phoneNames) { | |
String comp = compNames[i++]; | |
stmt.clearBindings(); | |
stmt.bindString(1, name); | |
stmt.bindString(2, comp); | |
stmt.executeInsert(); | |
} | |
db.setTransactionSuccessful(); // remove that call and none of the changes will be committed! | |
} catch (Exception e) { | |
// handle exception here | |
} finally { | |
db.endTransaction(); // this must be in the finally block | |
} | |
} |
0 comments:
Post a Comment