Tutoriel Android : Apprendre à utiliser la base de donnée SQLite

Chapitre 4.3
Image non disponible
Android2ee

L'auteur

Profil ProSite personnel

Liens sociaux

Viadeo Twitter Facebook Share on Google+   

I. Base de données locale

La base de données installée sur Android est SQLite, par contre, elle ne se pilote pas avec le JDBC usuel, mais avec un SGBDR (Système de Gestion de Base de Données Relationnelle) spécifique. Il faut modifier nos vieilles habitudes Java d'utilisateurs d'Hibernate, d'Ibatis ou de notre bon vieux JDBC. Une grosse différence concerne le typage des données de vos colonnes qui ne sera pas utilisé par SQLite (il l'est de manière indicative uniquement) : SQLite permet de stocker une valeur de n'importe quel type dans n'importe quelle colonne, quel que soit le type déclaré de cette colonne. D'autres différences : SQLite ne reconnait pas les contraintes de clefs étrangères, les transactions imbriquées, les jointures externes, certaines variantes d'ALTER TABLE…

Cf http://www.sqlite.org/docs.html et surtout http://www.sqlite.org/lang.html pour connaître la syntaxe à utiliser.

Comment cela marche-t-il ?

La base de données est liée à votre activité (pour pouvoir partager son accès, il faut mettre en place un ContentProvider, cela sera explicité dans un autre chapitre), c'est elle qui la crée, la met à jour, la remplit,  l'interroge et la clôt. Ainsi dans le onCreate de votre activité vous appelez votre base. Si celle-ci n'existe pas, le code que vous avez implémenté pour la créer est appelé. Si sa structure a évolué, le code que vous avez écrit pour la mettre à jour est appelé. Quand vous récupérez votre base, vous la manipulez au travers d'une classe spécifique qui encapsule les appels que vous avez écrits (votre DAO). Quand vous quittez votre activité, dans la méthode Destroy, vous fermez votre base.

Dans la pratique cela implique au moins deux classes spécifiques :

  • votre classe de création et de mise à jour de la base (de la structure de la base) qui étend SQLiteHelper. N'oubliez pas de mettre votre clef primaire technique unique qui s'auto-incrémente pour chacune de vos tables. Enfin, il est préconisé que tous vos noms de tables, de colonnes de la base soient des variables public final static de votre classe. Cette classe implémente au moins trois méthodes :
    • le constructeur qui appelle son super (et qui prend en paramètre le numéro de version de la structure de la base),
    • onCreate qui crée la base si elle n'existe pas,
    • onUpdate qui met à jour votre structure de table si celle-ci a évolué. Pour savoir si elle a évolué, un numéro de version de la structure de la base est utilisé. En effet elle prend en paramètre un oldVersion et un newVersion ;
  • votre classe DAO (vos classes DAO) qui encapsule vos appels à la base. Deux manières d'obtenir votre base de données sont à votre disposition :
    • getWritableDatabase() pour l'obtenir en mode écriture (mais du coup évidemment c'est un mode lecture/écriture),
    • getReadableDatabase() pour l'obtenir en mode lecture seule.

Enfin, souvenez-vous que vous devez utiliser les méthodes beginTransaction, endTransaction et commitTransaction si vous faites des traitements qui ne sont pas atomiques.

II. Ouverture et fermeture

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
private SQLiteDatabase db;
				
@Override
public void open() throws SQLiteException { 
    try {
        db = dbHelper.getWritableDatabase();
    } catch (SQLiteException ex) {
        db = dbHelper.getReadableDatabase();
    }
} 

public void close() {
    db.close();
}

Où la classe DBHelper est votre classe de création de la base qui étend SQLiteHelper. (cf. ci-dessous pour l'explication.)

III. Création et mise à jour de la structure de la base

Commençons par un exemple :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
33.
34.
35.
36.
37.
38.
39.
40.
41.
42.
43.
44.
45.
46.
47.
48.
49.
50.
51.
52.
53.
54.
55.
56.
57.
58.
59.
60.
61.
62.
63.
64.
65.
66.
67.
68.
69.
70.
71.
72.
73.
74.
75.
76.
77.
78.
79.
80.
81.
82.
83.
84.
85.
86.
87.
88.
89.
90.
91.
92.
93.
94.
95.
96.
97.
98.
99.
public class DBOpenHelper extends SQLiteOpenHelper {
    /**
     * @goals This class aims to show the constant to use for the DBOpenHelper
     */
    public static class Constants implements BaseColumns {
	
	// The database name
	public static final String DATABASE_NAME = "myData.db";
	
	// The database version
	public static final int DATABASE_VERSION = 1;
	
	// The table Name
	public static final String MY_TABLE = "Human";
	
	// ## Column name ##
	// My Column ID and the associated explanation for end-users 
	public static final String KEY_COL_ID = "_id";// Mandatory

	// My Column Name and the associated explanation for end-users 
	public static final String KEY_COL_NAME = "name";
	
	// My Column First Name and the associated explanation for end-users 
	public static final String KEY_COL_FIRSTNAME = "firstName";
	
	// My Column Eyes Color and the associated explanation for end-users 
	public static final String KEY_COL_EYES_COLOR = "eyesColor";
	
	// My Column Hair color and the associated explanation for end-users 
	public static final String KEY_COL_HAIR_COLOR = "hairColor";
	
	// My Column age and the associated explanation for end-users 
	public static final String KEY_COL_AGE = "age";

	// Indexes des colonnes
	// The index of the column ID 
	public static final int ID_COLUMN = 1;
	
	// The index of the column NAME 
	public static final int NAME_COLUMN = 2;
	
	// The index of the column FIRST NAME 
	public static final int FIRSTNAME_COLUMN = 3;
	
	// The index of the column EYES COLOR 
	public static final int EYES_COLOR_COLUMN = 4;
	
	// The index of the column HAIR COLOR 
	public static final int HAIR_COLOR_COLUMN = 5;
	
	// The index of the column AGE 
	public static final int AGE_COLUMN = 6;
    }

    /**
     * The static string to create the database.
     */
    private static final String DATABASE_CREATE = "create table "
	    + Constants.MY_TABLE + "(" + Constants.KEY_COL_ID
	    + " integer primary key autoincrement, " + Constants.KEY_COL_AGE
	    + " INTEGER, " + Constants.KEY_COL_NAME + " TEXT, "
	    + Constants.KEY_COL_FIRSTNAME + " TEXT, "
	    + Constants.KEY_COL_EYES_COLOR + " TEXT, "
	    + Constants.KEY_COL_HAIR_COLOR + " TEXT) ";

    /**
     * @param context
     *            = the context of the caller
     * @param name
     *            = Database's name
     * @param factory
     *            = null
     * @param version
     *            = Database's version
     */
    public DBOpenHelper(Context context, String name, CursorFactory factory,
	    int version) {
	super(context, name, factory, version);
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
	// Create the new database using the SQL string Database_create
	db.execSQL(DATABASE_CREATE);

    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
	Log.w("DBOpenHelper", "Mise à jour de la version " + oldVersion
		+ " vers la version " + newVersion
		+ ", les anciennes données seront détruites ");
	// Drop the old database
	db.execSQL("DROP TABLE IF EXISTS " + Constants.MY_TABLE);
	// Create the new one
	onCreate(db);
	// or do a smartest stuff
    }
}

Qui est appelé par votre activité par :

 
Sélectionnez
1.
2.
dbOpenHelper = new DBOpenHelper(this, Constants.DATABASE_NAME, null,
Constants.DATABASE_VERSION);

Où :

·         Context est le contexte de votre activité (this quand vous êtes dans votre activité) ;

·         le nom de votre base de données ;

·         une fabrique de cureurs (le plus souvent null) ;

·         la version de la structure de votre base.

La syntaxe de création est la suivante :

Image non disponible

Ainsi l'exemple précédent crée une table dont le nom est DATABASE_TABLE avec une colonne KEY_ID qui est la clef primaire de type entier qui s'auto-incrémente, une colonne KEY_TASK de type String, non null, et une colonne KEY_CREATION_DATE de type long.

De plus voilà la syntaxe complète de création d'une table µSQLite :

Collumn-def :

Image non disponible

Type-name :

Image non disponible

Column-constraint :

Image non disponible

Table-constraint :

Image non disponible

ForeignKey clauses :

Image non disponible

IV. Gérer l'instanciation et l'ouverture

Pour la création et l'instanciation d'une base de données :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
// The database.
private SQLiteDatabase db;

// The database creator and updater helper.
DBOpenHelper dbOpenHelper;

@Override
public void onCreate(Bundle savedInstanceState) {
	super.onCreate(savedInstanceState);
	setContentView(R.layout.main);
	// Create or retrieve the database
	dbOpenHelper = new DBOpenHelper(this, Constants.DATABASE_NAME, null,
		Constants.DATABASE_VERSION);
	// open the database
	openDB();

	// Insert a new record
	ContentValues contentValues = new ContentValues();
	long rowId = insertRecord(contentValues);

	// Update that line
	rowId = updateRecord(contentValues, rowId);

	// Query that line
	queryTheDatabase();

	// And then delete it:
	deleteRecord(rowId);
}

V. Mise à jour des données de la table : Insert, Update et Delete de la classe SQLiteDatabase

Pour l'insert :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
// Crée une nouvelle ligne à insérer.	
ContentValues newTaskValues = new ContentValues();
	
// Assign the values for each column.
contentValues.put(Constants.KEY_COL_NAME, "name");
contentValues.put(Constants.KEY_COL_FIRSTNAME, "firstName");
contentValues.put(Constants.KEY_COL_EYES_COLOR, "green");
contentValues.put(Constants.KEY_COL_HAIR_COLOR, "blond");
contentValues.put(Constants.KEY_COL_AGE, 6);
	
// Insert the line in the database
long rowId = db.insert(Constants.MY_TABLE, null, contentValues);
	
// test to see if the insertion was ok
if (rowId == -1) {
    Toast.makeText(this, "Error when creating an human",
	    Toast.LENGTH_LONG).show();
} else {
    Toast.makeText(this, "Human created and stored in database", Toast.LENGTH_LONG).show();
}
return rowId;

On définit un objet ContentValues dans lequel on place des paires (NomDeColonne, Valeur) à créer. Puis on utilise notre objet SQLiteDatabase pour faire l'insert en lui passant, le nom de la table, le nom d'une colonne (souvent null) et l'objet contenant la carte des paires (colonne, valeur) à insérer.

Pour l'update :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
contentValues.clear();
contentValues.put(Constants.KEY_COL_NAME, "Georges");
contentValues.put(Constants.KEY_COL_FIRSTNAME, "Walter Bush");

// Update the database
rowId = db.update(Constants.MY_TABLE, contentValues, Constants.KEY_COL_ID + "=" + rowId, null);

// Test to see if the insertion was ok
if (rowId == -1) {
    Toast.makeText(this, "Error when updating an human", Toast.LENGTH_LONG).show();
} else {
    Toast.makeText(this, "Human updated in database", Toast.LENGTH_LONG).show();
}
return rowId;

On définit l'objet ContentValues dans lequel on place des paires (NomDeColonne, Valeur) à mettre à jour. Puis on utilise notre objet SQLiteDatabase pour faire l'update en lui passant le nom de la table, les valeurs à mettre et la clause where (ici la clef primaire de l'objet).

Pour le delete :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
rowId = db.delete(DBOpenHelper.Constants.MY_TABLE,DBOpenHelper.Constants.KEY_COL_ID + "=" + rowId, null);
if (rowId == -1) {
    Toast.makeText(this, "Error when deleting an human", Toast.LENGTH_LONG).show();
} else {
    Toast.makeText(this, "Human deleted in database", Toast.LENGTH_LONG).show();
}

On utilise notre objet SQLiteDatabase pour faire le delete en lui passant le nom de la table et la clause where (ici la clef primaire de l'objet).

VI. Récupération des données de la table Query de la classe SQLiteDatabase

VI-1. Parcourir le résultat des requêtes

Les résultats de requêtes sont renvoyés sous forme d'une liste de pointeurs vers les objets retournés. Cette liste de pointeurs se nomme Cursor. Cette classe possède de nombreuses méthodes :

  • de mouvements (moveToFirst, moveToLast, moveNext, movePrevious, moveToPosition(int index)) ;
  • de récupérations (getDouble, getFloat, getInt, getLong, getShort, getString, getBlob) ;
  • d'informations (getColunmCount, getColumnName, getColumnIndex, getPosition, getCount...) ;
  • d'états (isAfterLast, isBeforeFirst, isClosed, isFirst, isLast, isNull) ;
  • d'autres méthodes de gestion de l'objet Cursor lui-même.

L'amalgame entre la liste de curseurs et la position du curseur est clair. On bouge dans la liste et en même temps quand on pointe sur un élément on l'interroge. Il faut s'y faire :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
Cursor cursor = db.query(Constants.MY_TABLE,projections ,selection, selectionArg, groupBy,having,orderBy, maxResultsListSize);
// Then browse the result:
if (cursor.moveToFirst()){
    // The elements to retrieve
    Integer colId;
    String name;
    String firstname;
    
	// The associated index within the cursor
    int indexId=cursor.getColumnIndex(Constants.KEY_COL_ID);
    int indexName=cursor.getColumnIndex(Constants.KEY_COL_NAME);
    int indexFirstName=cursor.getColumnIndex(Constants.KEY_COL_FIRSTNAME);
    
	// Browse the results list:
    int count=0;
    do {
		colId=cursor.getInt(indexId);
		name=cursor.getString(indexName);
		firstname=cursor.getString(indexFirstName);
		Toast.makeText(this, "Retrieve element :"+name+","+firstname+" ("+colId+")", Toast.LENGTH_LONG).show();
		count++;
	} while(cursor.moveToNext());
	Toast.makeText(this, "The number of elements retrieved is "+count, Toast.LENGTH_LONG).show();
} else {
	Toast.makeText(this, "No element found : ", Toast.LENGTH_LONG).show();
}

VI-2. Effectuer une requête à la main

Il faut utiliser la méthode query de SQLiteDatabase.

Cette méthode prend en paramètres les éléments suivants :

  • un booléen optionnel spécifiant si le résultat se compose d'éléments uniques (Boolean) ;
  • le nom de la table à interroger (String) ;
  • une projection sous forme de tableau de chaines listant les colonnes à inclure dans le résultat (String[]) ;
  • une clause where avec des jokers (?) pour les paramètres si vous le souhaitez (String) ;
  • la liste des paramètres qui remplaceront les jokers (?)(String[]) ;
  • une clause groupBy (String) ;
  • une clause orderBy dont la syntaxe est nomDeColonne + « ASC||Desc »(String) ;
  • une limite maximale optionnelle du nombre d'éléments à récupérer (String).

Un exemple :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
// Using man made query
// The projection define what are the column you want to retrieve
String[] projections=new String[] { Constants.KEY_COL_ID, Constants.KEY_COL_NAME, Constants.KEY_COL_FIRSTNAME };

// And then store the column index answered by the request (we present an other way to retireve data)
final int cursorIdColNumber=0,cursorNameColNumber=1,cursorFirstNameColNumber=2;

// To add a Where clause:
String selection=Constants.KEY_COL_HAIR_COLOR+"=?";
String[] selectionArg=new String[] {"blond"};

// The groupBy clause:
String groupBy=Constants.KEY_COL_EYES_COLOR;

// The having clause
String having=null;

// The order by clause (column name followed by Asc or DESC)
String orderBy = Constants.KEY_COL_AGE+"  ASC";

// Maximal size of the results list
String maxResultsListSize="60";
Cursor cursor = db.query(Constants.MY_TABLE,projections ,selection, selectionArg, groupBy,having,orderBy, maxResultsListSize);
displayResults(cursor);

VI-3. Effectuer une requête avec SQLiteQueryBuilder

Vous pouvez utiliser aussi l'objet SQLiteQueryBuilder qui permet d'effectuer des unions et quelques autres opérations de ce type.

Nous reprenons dans l'exemple uniquement une recherche simple :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
// Using SQLiteQueryBuilder
// The projection define what are the column you want to retrieve
String[] projections=new String[] { Constants.KEY_COL_ID, Constants.KEY_COL_NAME, Constants.KEY_COL_FIRSTNAME };

// And then store the column index answered by the request (we present an other way to retireve data)
final int cursorIdColNumber=0,cursorNameColNumber=1,cursorFirstNameColNumber=2;

// Adding a where clause:
String selection=Constants.KEY_COL_HAIR_COLOR+"=?";
String[] selectionArg=new String[] {"blond"};

// The groupBy clause:
String groupBy=Constants.KEY_COL_EYES_COLOR;
// The having clause
String having=null;

// The order by clause (column name followed by Asc or DESC)
String orderBy = Constants.KEY_COL_AGE+"  ASC";
 
// Using the QueryBuilder
// Create a Query SQLite object
SQLiteQueryBuilder qb = new SQLiteQueryBuilder();
qb.setTables(Constants.MY_TABLE);          
// qb.appendWhere(Constants.KEY_COL_HAIR_COLOR+ "=blond");
qb.setDistinct(true);
cursor=qb.query(db,projections ,selection, selectionArg, groupBy,having,orderBy);
displayResults(cursor);

VII. Gestion des transactions

Les transactions peuvent avoir un listener, ce qui est une bonne pratique, mais vous pouvez vous en passer (ce qui est une mauvaise pratique).

Le pattern standard d'utilisation des transactions est :

soit :

 
Sélectionnez
db.beginTransactionWithListener(listener);

soit :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
db.beginTransaction();
try {
	...
	db.setTransactionSuccessful();
} finally {
	db.endTransaction();
}

Le listener est du type : public void beginTransactionWithListener (SQLiteTransactionListener transactionListener)

VIII. Un exemple complet

Cet exemple montre comment mettre en place une base de données et communiquer avec elle à partir d'une activité. Il y a deux classes, la classe d'instanciation de la base et la classe d'utilisation de cette base.

La classe de création :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
33.
34.
35.
36.
37.
38.
39.
40.
41.
42.
43.
44.
45.
46.
47.
48.
49.
50.
51.
52.
53.
54.
55.
56.
57.
58.
59.
60.
61.
62.
63.
64.
65.
66.
67.
68.
69.
70.
71.
72.
73.
74.
75.
76.
77.
78.
79.
80.
81.
82.
83.
84.
85.
86.
87.
88.
89.
90.
91.
92.
93.
public class DBOpenHelper extends SQLiteOpenHelper {
    // @goals This class aims to show the constant to use for the DBOpenHelper */
    public static class Constants implements BaseColumns {
	// The database name 
	public static final String DATABASE_NAME = "myData.db";
	
	// The database version 
	public static final int DATABASE_VERSION = 1;
	
	// The table Name 
	public static final String MY_TABLE = "Human";
	
	// Noms de colonnes
	// /!\Si vous utilisez une base de données, les noms des colonnes ont
	// les mêmes que ceux de
	// votre base, de même pour les index.
	// My Column ID and the associated explanation for end-users 
	public static final String KEY_COL_ID = "_id";// Mandatory

	// My Column Name and the associated explanation for end-users 
	public static final String KEY_COL_NAME = "name";
	
	// My Column First Name and the associated explanation for end-users 
	public static final String KEY_COL_FIRSTNAME = "firstName";
	
	// My Column Eyes Color and the associated explanation for end-users 
	public static final String KEY_COL_EYES_COLOR = "eyesColor";
	
	// My Column Hair color and the associated explanation for end-users 
	public static final String KEY_COL_HAIR_COLOR = "hairColor";
	
	// My Column age and the associated explanation for end-users 
	public static final String KEY_COL_AGE = "age";

	// Index des colonnes
	// The index of the column ID 
	public static final int ID_COLUMN = 1;
	
	// The index of the column NAME 
	public static final int NAME_COLUMN = 2;
	
	// The index of the column FIRST NAME 
	public static final int FIRSTNAME_COLUMN = 3;
	
	// The index of the column EYES COLOR 
	public static final int EYES_COLOR_COLUMN = 4;
	
	// The index of the column HAIR COLOR 
	public static final int HAIR_COLOR_COLUMN = 5;
	
	// The index of the column AGE 
	public static final int AGE_COLUMN = 6;
    }

    // The static string to create the database.
    private static final String DATABASE_CREATE = "create table "
	    + Constants.MY_TABLE + "(" + Constants.KEY_COL_ID
	    + " integer primary key autoincrement, " + Constants.KEY_COL_AGE
	    + " INTEGER, " + Constants.KEY_COL_NAME + " TEXT, "
	    + Constants.KEY_COL_FIRSTNAME + " TEXT, "
	    + Constants.KEY_COL_EYES_COLOR + " TEXT, "
	    + Constants.KEY_COL_HAIR_COLOR + " TEXT) ";

    /**
     * @param context
     * @param name
     * @param factory
     * @param version
     */
    public DBOpenHelper(Context context, String name, CursorFactory factory,
	    int version) {
	super(context, name, factory, version);
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
	// Create the new database using the SQL string Database_create
	db.execSQL(DATABASE_CREATE);

    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
	Log.w("DBOpenHelper", "Mise à jour de la version " + oldVersion
		+ " vers la version " + newVersion
		+ ", les anciennes données seront détruites ");
	// Drop the old database
	db.execSQL("DROP TABLE IF EXISTS " + Constants.MY_TABLE);
	// Create the new one
	onCreate(db);
	// or do a smartest stuff
    }
}

La classe d'utilisation :

 
Sélectionnez
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
33.
34.
35.
36.
37.
38.
39.
40.
41.
42.
43.
44.
45.
46.
47.
48.
49.
50.
51.
52.
53.
54.
55.
56.
57.
58.
59.
60.
61.
62.
63.
64.
65.
66.
67.
68.
69.
70.
71.
72.
73.
74.
75.
76.
77.
78.
79.
80.
81.
82.
83.
84.
85.
86.
87.
88.
89.
90.
91.
92.
93.
94.
95.
96.
97.
98.
99.
100.
101.
102.
103.
104.
105.
106.
107.
108.
109.
110.
111.
112.
113.
114.
115.
116.
117.
118.
119.
120.
121.
122.
123.
124.
125.
126.
127.
128.
129.
130.
131.
132.
133.
134.
135.
136.
137.
138.
139.
140.
141.
142.
143.
144.
145.
146.
147.
148.
149.
150.
151.
152.
153.
154.
155.
156.
157.
158.
159.
160.
161.
162.
163.
164.
165.
166.
167.
168.
169.
170.
171.
172.
173.
174.
175.
176.
177.
178.
179.
180.
181.
182.
183.
184.
185.
186.
187.
188.
189.
190.
191.
192.
193.
194.
195.
196.
197.
198.
199.
200.
201.
202.
203.
204.
205.
206.
207.
208.
209.
210.
211.
212.
213.
214.
215.
216.
217.
218.
219.
220.
221.
222.
public class DataBaseTuto extends Activity {
    // The database 
    private SQLiteDatabase db;
    // The database creator and updater helper 
    DBOpenHelper dbOpenHelper;

    // Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
	super.onCreate(savedInstanceState);
	setContentView(R.layout.main);
	
	// Create or retrieve the database
	dbOpenHelper = new DBOpenHelper(this, Constants.DATABASE_NAME, null,
		Constants.DATABASE_VERSION);
	
	// open the database
	openDB();
	
	// Insert a new record
	// -------------------
	ContentValues contentValues = new ContentValues();
	long rowId = insertRecord(contentValues);
	
	// update that line
	// ----------------
	rowId = updateRecord(contentValues, rowId);
	
	// Query that line
	// ---------------
	queryTheDatabase();
	
	// And then delete it:
	// -------------------
	deleteRecord(rowId);
    }

    /*********************************************************************************/
    /** Managing LifeCycle and database open/close operations *********************************/
    /*********************************************************************************/
    @Override
    protected void onResume() {
	super.onResume();
	openDB();
    }

    @Override
    protected void onPause() {
	super.onPause();
	closeDB();
    }

    /**
     * * Open the database* *
     * 
     * @throws SQLiteException
     */
    public void openDB() throws SQLiteException {
	try {
	    db = dbOpenHelper.getWritableDatabase();
	} catch (SQLiteException ex) {
	    db = dbOpenHelper.getReadableDatabase();
	}
    }

    /** *Close Database */
    public void closeDB() {
	db.close();
    }

    /****************************************************************************************/
    /** Managing Records **********************************************************************/
    /****************************************************************************************/
    /**
     * Insert a record
     *
     * @param contentValues
     *            (an empty contentValues)
     * @return the inserted row id
     */
    private long insertRecord(ContentValues contentValues) {
	// Assign the values for each column.
	contentValues.put(Constants.KEY_COL_NAME, "name");
	contentValues.put(Constants.KEY_COL_FIRSTNAME, "firstName");
	contentValues.put(Constants.KEY_COL_EYES_COLOR, "green");
	contentValues.put(Constants.KEY_COL_HAIR_COLOR, "blond");
	contentValues.put(Constants.KEY_COL_AGE, 6);

	// Insert the line in the database
	long rowId = db.insert(Constants.MY_TABLE, null, contentValues);
	
	// Test to see if the insertion was ok
	if (rowId == -1) {
	    Toast.makeText(this, "Error when creating an human",
		    Toast.LENGTH_LONG).show();
	} else {
	    Toast.makeText(this, "Human created and stored in database",
		    Toast.LENGTH_LONG).show();
	}
	return rowId;
    }

    /**
     * * Update a record
     *
     * @param contentValues
     * @param rowId
     * @return the updated row id
     */
    private long updateRecord(ContentValues contentValues, long rowId) {
	contentValues.clear();
	contentValues.put(Constants.KEY_COL_NAME, "Georges");
	contentValues.put(Constants.KEY_COL_FIRSTNAME, "Walter Bush");
	// update the database
	rowId = db.update(Constants.MY_TABLE, contentValues,
		Constants.KEY_COL_ID + "=" + rowId, null);
	// test to see if the insertion was ok
	if (rowId == -1) {
	    Toast.makeText(this, "Error when updating an human",
		    Toast.LENGTH_LONG).show();
	} else {
	    Toast.makeText(this, "Human updated in database", Toast.LENGTH_LONG)
		    .show();
	}
	return rowId;
    }

    /**
     * * Delete a Record *
     * 
     * @param rowId
     */
    private void deleteRecord(long rowId) {
	rowId = db.delete(DBOpenHelper.Constants.MY_TABLE,
		DBOpenHelper.Constants.KEY_COL_ID + "=" + rowId, null);
	if (rowId == -1) {
	    Toast.makeText(this, "Error when deleting an human",
		    Toast.LENGTH_LONG).show();
	} else {
	    Toast.makeText(this, "Human deleted in database", Toast.LENGTH_LONG)
		    .show();
	}
    }

    /**
     * Query a the database
     */
    private void queryTheDatabase() {
	// Using man made query
	// The projection define what are the column you want to retrieve
	String[] projections = new String[] { Constants.KEY_COL_ID,
		Constants.KEY_COL_NAME, Constants.KEY_COL_FIRSTNAME };
	// And then store the column index answered by the request (we present
	// an other way to
	// retireve data)
	final int cursorIdColNumber = 0, cursorNameColNumber = 1, cursorFirstNameColNumber = 2;
	// To add a Where clause you can either do that:
	// qb.appendWhere(Constants.KEY_COL_HAIR_COLOR+ "=blond");
	// Or that:
	String selection = Constants.KEY_COL_HAIR_COLOR + "=?";
	String[] selectionArg = new String[] { "blond" };
	// The groupBy clause:
	String groupBy = Constants.KEY_COL_EYES_COLOR;
	// The having clause
	String having = null;
	// The order by clause (column name followed by Asc or DESC)
	String orderBy = Constants.KEY_COL_AGE + "  ASC";
	// Maximal size of the results list
	String maxResultsListSize = "60";
	Cursor cursor = db.query(Constants.MY_TABLE, projections, selection,
		selectionArg, groupBy, having, orderBy, maxResultsListSize);
	displayResults(cursor);

	// Using the QueryBuilder
	// Create a Query SQLite object
	SQLiteQueryBuilder qb = new SQLiteQueryBuilder();
	qb.setTables(Constants.MY_TABLE);
	// qb.appendWhere(Constants.KEY_COL_HAIR_COLOR+ "=blond");
	qb.setDistinct(true);
	cursor = qb.query(db, projections, selection, selectionArg, groupBy,
		having, orderBy);
	displayResults(cursor);
    }

    /**
     * Display the results stored in the cursor
     *
     * @param cursor
     */
    private void displayResults(Cursor cursor) {
	// then browse the result:
	if (cursor.moveToFirst()) {
	    // The elements to retrieve
	    Integer colId;
	    String name;
	    String firstname;
	    // The associated index within the cursor
	    int indexId = cursor.getColumnIndex(Constants.KEY_COL_ID);
	    int indexName = cursor.getColumnIndex(Constants.KEY_COL_NAME);
	    int indexFirstName = cursor
		    .getColumnIndex(Constants.KEY_COL_FIRSTNAME);
	    // Browse the results list:
	    int count = 0;
	    do {
		colId = cursor.getInt(indexId);
		name = cursor.getString(indexName);
		firstname = cursor.getString(indexFirstName);
		Toast.makeText(
			this,
			"Retrieve element :" + name + "," + firstname + " ("
				+ colId + ")", Toast.LENGTH_LONG).show();
		count++;
	    } while (cursor.moveToNext());
	    Toast.makeText(this,
		    "The number of elements retrieved is " + count,
		    Toast.LENGTH_LONG).show();
	} else {
	    Toast.makeText(this, "No element found : ", Toast.LENGTH_LONG)
		    .show();
	}
    }
}

IX. Remerciements

Nous tenons à remercier Claude Leloup pour la relecture orthographique de cet article.

Vous avez aimé ce tutoriel ? Alors partagez-le en cliquant sur les boutons suivants : Viadeo Twitter Facebook Share on Google+   

  

Les sources présentées sur cette page sont libres de droits et vous pouvez les utiliser à votre convenance. Par contre, la page de présentation constitue une œuvre intellectuelle protégée par les droits d'auteur. Copyright © 2016 Mathias Seguy. Aucune reproduction, même partielle, ne peut être faite de ce site et de l'ensemble de son contenu : textes, documents, images, etc. sans l'autorisation expresse de l'auteur. Sinon vous encourez selon la loi jusqu'à trois ans de prison et jusqu'à 300 000 € de dommages et intérêts.