Have you ever been tired of parsing SQLite query results using Android cursors? You have to write lots of boilerplate code just to parse query result rows, and enclose it countless try..finally
blocks to properly close all opened resources.
Anko provides lots of extension functions to simplify communication with SQLite database.
- Db package
- Accessing database
- Creating and dropping tables
- Inserting data
- Querying data
- Parsing query result
- Custom row parsers
- Cursor streams
- Updating values
- Transactions
All database-related tools are in the org.jetbrains.anko.db
package. You could probably want to import all children at once:
import org.jetbrains.anko.db.*
If you use SQLiteOpenHelper
, you generally call getReadableDatabase()
or getWritableDatabase()
(result is actually the same in production code), but then you must be sure to call the close()
method on the received SQLiteDatabase
. Also, you have to cache the helper class somewhere, and if you use it from several threads, you must be aware of the concurrent access. All this is pretty though. That is why Android developers not really keen on default SQLite API and prefer to use fairly expensive wrappers such as ORMs instead.
Anko provides a special class ManagedSQLiteOpenHelper
that seamlessly replaces the default one. That's how you can use it:
class MyDatabaseOpenHelper(ctx: Context) : ManagedSQLiteOpenHelper(ctx, "MyDatabase", null, 1) {
class object {
private var instance: MyDatabaseOpenHelper? = null
synchronized fun getInstance(ctx: Context): MyDatabaseOpenHelper {
if (instance == null) {
instance = MyDatabaseOpenHelper(ctx.getApplicationContext())
}
return instance!!
}
}
override fun onCreate(db: SQLiteDatabase) {
// Here you create tables (more info about that is below)
}
override fun onUpgrade(db: SQLiteDatabase, oldVersion: Int, newVersion: Int) {
// Here you can upgrade tables, as usual
}
}
// Access property for Context
val Context.database: MyDatabaseOpenHelper
get() = MyDatabaseOpenHelper.getInstance(getApplicationContext())
So what's the sense? Instead of enclosing your code into try
blocks, now you can just write this:
database.use {
// Now `this` is a SQLiteDatabase instance
}
Database will be closed for sure after executing all code inside {}
.
Asynchronous call example (please read here about async
):
class SomeActivity : Activity() {
private fun loadAsync() {
async {
val result = database.use {
// Now `this` is a SQLiteDatabase instance
}
uiThread { loadComplete(result) }
}
}
}
🐧 |
These and all methods mentioned below may throw SQLiteException . You have to handle it by yourself because it would be unreasonable for Anko to pretend that errors don't occur.
|
With Anko you can easily create new tables and drop existing. The syntax is straightforward.
database.use {
createTable("Customer", ifNotExists = true,
"_id" to INTEGER + PRIMARY_KEY + UNIQUE,
"name" to TEXT,
"photo" to BLOB)
}
In SQLite there are five main types: NULL
, INTEGER
, REAL
, TEXT
and BLOB
. But each column may have some modificators like PRIMARY KEY
or UNIQUE
. You can append such modificators with "adding" them to the primary type name.
To drop table, use the dropTable
function:
dropTable("User", ifNotExists = true)
Usually, you need a ContentValues
instance to insert a row into the table. Here is an example:
val values = ContentValues()
values.put("_id", 5)
values.put("name", "John Smith")
values.put("email", "[email protected]")
db.insert("User", null, values)
Anko lets you to eliminate such ceremonies by passing values directly as arguments for the insert()
function:
db.insert("User",
"_id" to 42,
"name" to "John",
"email" to "[email protected]"
)
Functions insertOrThrow()
, replace()
, replaceOrThrow()
are also exist and have the similar semantics.
Anko provides a convenient query builder. It may be created with
db.select(tableName, vararg columns)
where db
is an instance of SQLiteDatabase
.
Method | Description |
---|---|
column(String) |
Add a column to select query |
distinct(Boolean) |
Distinct query |
where(String) |
Specify raw String where query |
where(String, args) ⭐ |
Specify a where query with arguments |
where?(String, args) |
Specify a where query with ? mark arguments |
orderBy(String, [ASC/DESC]) |
Order by this column |
groupBy(String) |
Group by this column |
limit(count: Int) |
Limit query result row count |
limit(offset: Int, count: Int) |
Limit query result row count with an offset |
having(String) |
Specify raw having expression |
having(String, args) ⭐ |
Specify a having expression with arguments |
Functions marked with ⭐ parse its arguments in a special way. They allow you to provide values in any order and support escaping seamlessly.
db.select("User", "name")
.where("(_id > {userId}) and (name = {userName})",
"userName" to "John",
"userId" to 42)
Here, {userId}
part will be replaced with 42
and {userName}
— with 'John'
. Value will be escaped if its type is not numeric (Int
, Float
etc.) or Boolean
. For any other types, toString()
representation will be used.
where?
function accepts arguments of String
type. It works the same as query()
from SQLiteDatabase
(question marks ?
will be replaced with actual values from arguments).
How can we execute the query? Using exec()
function. It accepts an extension function which type is Cursor.() -> T
. It simply launches the received extension function and then closes Cursor
so you don't need to do it by yourself:
db.select("User", "email").exec {
// Doing some stuff with emails
}
So we have some Cursor
, and how can we parse it into regular classes? Anko provides functions parseSingle
, parseOpt
and parseList
to do it much more easier.
Method | Description |
---|---|
parseSingle(rowParser): T |
Parse exactly one row |
parseOpt(rowParser): T? |
Parse zero or one row |
parseList(rowParser): List<T> |
Parse zero or more rows |
Note that parseSingle()
and parseOpt()
will throw an exception if the received Cursor contains more than one row.
Now the question is: what is rowParser
? Well, each function support two different types of parsers: RowParser
and MapRowParser
:
public trait RowParser<T> {
fun parseRow(columns: Array<Any>): T
}
public trait MapRowParser<T> {
fun parseRow(columns: Map<String, Any>): T
}
If you want to write your query in a very efficient way, use RowParser (but then you must know the index of each column). parseRow
accepts a list of Any
(the type of Any
could practically be nothing but Long
, Double
, String
or ByteArray
). MapRowParser
, on the other hand, lets you to get row values by using column names.
Anko already has parsers for simple single-column rows:
ShortParser
IntParser
LongParser
FloatParser
DoubleParser
StringParser
BlobParser
Also you can create a row parser from class constructor. Assuming you have a class:
class Person(val firstName: String, val lastName: String, val age: Int)
The parser will be as simple as:
val rowParser = classParser<Person>()
For now Anko does not support creating such parsers if the primary constructor has optional parameters. Also note that constructor will be invoked using Java Reflection so writing a custom RowParser
is more reasonable for huge data sets.
If you are using Anko db.select()
builder, you can directly call parseSingle
, parseOpt
or parseList
on it and pass an appropriate parser.
For instance, let's make a new parser for columns (Int, String, String)
. The most naive way to do so is:
public class MyRowParser : RowParser<Triple<Int, String, String>> {
override fun parseRow(columns: Array<Any>): Triple<Int, String, String> {
return Triple(columns[0] as Int, columns[1] as String, columns[2] as String)
}
}
Well, now we have three explicit checkcasts in our code. Let's get rid of them by using the rowParser
function:
val parser = rowParser { (id: Int, name: String, email: String) ->
Triple(id, name, email)
}
And that's it! rowParser
makes all class casts under the hood and you can name the lambda parameters as you want.
Anko provides a way to access SQLite Cursor
in functional way. Just call cursor.asSequence()
or cursor.asMapSequence()
extension functions to get a sequence or rows. Do not forget to close a Cursor
:)
Let's give the a new name to one of our users:
update("User", "name" to "Alice")
.where("_id = {userId}", "userId" to 42)
.exec()
Update also contains where?()
function in case you want to provide query in a traditional way:
update("User", "name" to "Alice")
.`where?`("_id = ?", 42)
.exec()
There is a special function called transaction()
which allows you to enclose several database operations in a single SQLite transaction.
transaction {
// Your transaction code
}
Transaction will be marked marked as successful if no exception was thrown inside the {}
block.
🐧 |
If you want to abort a transaction for some reason, just throw TransactionAbortException . You don't need to handle this exception by yourself in this case.
|