Додавання, видалення та оновлення даних у SQLite
Продовжимо роботу з проєктом із минулої теми, де ми отримуємо дані. Тепер додамо в нього стандартну CRUD-логіку (створення, оновлення, видалення).
Щоб не нагромаджувати форму з головною activity, всі інші дії по роботі з даними відбуватимуться на іншому екрані. Додамо в проєкт новий клас activity, який назвемо UserActivity:

У файлі activity_user.xml визначимо універсальну форму для додавання/оновлення/видалення даних:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<EditText
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:hint="Введіть ім'я"
app:layout_constraintBottom_toTopOf="@+id/year"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent" />
<EditText
android:id="@+id/year"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:hint="Введіть рік народження"
app:layout_constraintTop_toBottomOf="@+id/name"
app:layout_constraintBottom_toTopOf="@+id/saveButton"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent" />
<Button
android:id="@+id/saveButton"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:text="Зберегти"
android:onClick="save"
app:layout_constraintHorizontal_weight="1"
app:layout_constraintTop_toBottomOf="@+id/year"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toLeftOf="@+id/deleteButton"
/>
<Button
android:id="@+id/deleteButton"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:text="Видалити"
android:onClick="delete"
app:layout_constraintHorizontal_weight="1"
app:layout_constraintTop_toBottomOf="@+id/year"
app:layout_constraintLeft_toRightOf="@+id/saveButton"
app:layout_constraintRight_toRightOf="parent"
/>
</androidx.constraintlayout.widget.ConstraintLayout>
І також змінимо код UserActivity:
package com.example.sqliteapp;
import androidx.appcompat.app.AppCompatActivity;
import android.content.ContentValues;
import android.content.Intent;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
public class UserActivity extends AppCompatActivity {
EditText nameBox;
EditText yearBox;
Button delButton;
Button saveButton;
DatabaseHelper sqlHelper;
SQLiteDatabase db;
Cursor userCursor;
long userId = 0;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_user);
nameBox = findViewById(R.id.name);
yearBox = findViewById(R.id.year);
delButton = findViewById(R.id.deleteButton);
saveButton = findViewById(R.id.saveButton);
sqlHelper = new DatabaseHelper(this);
db = sqlHelper.getWritableDatabase();
Bundle extras = getIntent().getExtras();
if (extras != null) {
userId = extras.getLong("id");
}
// якщо 0, то додавання
if (userId > 0) {
// отримуємо елемент за id із бд
userCursor = db.rawQuery("select * from " + DatabaseHelper.TABLE + " where " +
DatabaseHelper.COLUMN_ID + "=?", new String[]{String.valueOf(userId)});
userCursor.moveToFirst();
nameBox.setText(userCursor.getString(1));
yearBox.setText(String.valueOf(userCursor.getInt(2)));
userCursor.close();
} else {
// приховуємо кнопку видалення
delButton.setVisibility(View.GONE);
}
}
public void save(View view) {
ContentValues cv = new ContentValues();
cv.put(DatabaseHelper.COLUMN_NAME, nameBox.getText().toString());
cv.put(DatabaseHelper.COLUMN_YEAR, Integer.parseInt(yearBox.getText().toString()));
if (userId > 0) {
db.update(DatabaseHelper.TABLE, cv, DatabaseHelper.COLUMN_ID + "=" + userId, null);
} else {
db.insert(DatabaseHelper.TABLE, null, cv);
}
goHome();
}
public void delete(View view) {
db.delete(DatabaseHelper.TABLE, "_id = ?", new String[]{String.valueOf(userId)});
goHome();
}
private void goHome() {
// закрываем подключение
db.close();
// переход к главной activity
Intent intent = new Intent(this, MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
startActivity(intent);
}
}
Під час оновлення або видалення об'єкта зі списку з головної activity в UserActivity передаватиметься id об'єкта:
long userId=0;
//.................
Bundle extras = getIntent().getExtras();
if (extras != null) {
userId = extras.getLong("id");
}
Якщо з MainActivity не було передано id, то встановлюємо його значення 0, отже, у нас буде додавання, а не редагування/видалення
Якщо id визначено, то отримуємо за ним із бази даних об'єкт для редагування/видалення:
if (id < 0) {
userCursor = db.rawQuery("select * from " + DatabaseHelper.TABLE + " where " +
DatabaseHelper.COLUMN_ID + "=?", new String[]{String.valueOf(id)});
userCursor.moveToFirst();
nameBox.setText(userCursor.getString(1));
yearBox.setText(String.valueOf(userCursor.getInt(2)));
userCursor.close();
}
Інакше просто приховуємо кнопку видалення.
Для виконання операцій по вставці, оновленню та видаленню даних SQLiteDatabase має методи insert(), update() і delete(). Ці методи викликаються в обробниках кнопок:
db.delete(DatabaseHelper.TABLE, "_id = ?", new String[]{String.valueOf(id)});
У метод delete() передається назва таблиці, а також стовпець, за яким відбувається видалення, і його значення. Як критерій можна вибрати кілька стовпців, тому третім параметром йде масив. Знак питання ? позначає параметр, замість якого підставляється значення з третього параметра.
ContentValues
Для додавання або оновлення нам треба створити об'єкт ContentValues. Цей об'єкт представляє собою словник, який містить набір пар "ключ-значення". Для додавання в цей словник нового об'єкта застосовується метод put. Перший параметр методу — це ключ, а другий — значення, наприклад:
ContentValues cv = new ContentValues();
cv.put("NAME", "Tom");
cv.put("YEAR", 30);
Як значення в метод put можна передавати рядки, цілі числа, числа з плаваючою точкою.
У цьому ж випадку додаються введені в текстові поля значення:
ContentValues cv = new ContentValues();
cv.put(DatabaseHelper.COLUMN_NAME, nameBox.getText().toString());
cv.put(DatabaseHelper.COLUMN_YEAR, Integer.parseInt(yearBox.getText().toString()));
При оновленні в метод update() передається назва таблиці, об'єкт ContentValues та критерій, за яким відбувається оновлення (в даному випадку стовпець id):
db.update(DatabaseHelper.TABLE, cv, DatabaseHelper.COLUMN_ID + "=" + userId, null);
Метод insert() приймає назву таблиці, об'єкт ContentValues з додаваними значеннями. Другий параметр є необов'язковим: він передає стовпець, у який треба додати значення NULL:
db.insert(DatabaseHelper.TABLE, null, cv);
Замість цих методів, як у попередній темі, можна використовувати метод execSQL() з точним вказуванням виконуваного SQL-виразу. Водночас методи delete/insert/update мають перевагу — вони повертають id зміненої записи, за яким ми можемо дізнатися про успішність операції, або -1 у разі невдалої операції:
long result = db.insert(DatabaseHelper.TABLE, null, cv);
if(result > 0){
// дії
}
Після кожної операції виконується метод goHome(), який повертає на головну activity.
Після цього нам треба виправити код MainActivity, щоб вона ініціювала виконання коду в UserActivity. Для цього змінимо код activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<Button
android:id="@+id/addButton"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:textSize="18sp"
android:text="Додати"
android:onClick="add"
app:layout_constraintBottom_toTopOf="@+id/list"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
/>
<ListView
android:id="@+id/list"
android:layout_width="0dp"
android:layout_height="0dp"
app:layout_constraintTop_toBottomOf="@+id/addButton"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"/>
</androidx.constraintlayout.widget.ConstraintLayout>
У цьому випадку було додано кнопку для виклику UserActivity.
І також змінимо код класу MainActivity:
package com.example.sqliteapp;
import androidx.appcompat.app.AppCompatActivity;
import android.content.Intent;
import android.view.View;
import android.widget.AdapterView;
import android.widget.SimpleCursorAdapter;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.os.Bundle;
import android.widget.ListView;
public class MainActivity extends AppCompatActivity {
ListView userList;
DatabaseHelper databaseHelper;
SQLiteDatabase db;
Cursor userCursor;
SimpleCursorAdapter userAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
userList = findViewById(R.id.list);
userList.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Intent intent = new Intent(getApplicationContext(), UserActivity.class);
intent.putExtra("id", id);
startActivity(intent);
}
});
databaseHelper = new DatabaseHelper(getApplicationContext());
}
@Override
public void onResume() {
super.onResume();
// відкриваємо підключення
db = databaseHelper.getReadableDatabase();
// отримуємо дані з бд у вигляді курсора
userCursor = db.rawQuery("select * from " + DatabaseHelper.TABLE, null);
// визначаємо, які стовпці з курсору будуть виводитися в ListView
String[] headers = new String[]{DatabaseHelper.COLUMN_NAME, DatabaseHelper.COLUMN_YEAR};
// створюємо адаптер, передаємо в нього курсор
userAdapter = new SimpleCursorAdapter(this, android.R.layout.two_line_list_item,
userCursor, headers, new int[]{android.R.id.text1, android.R.id.text2}, 0);
userList.setAdapter(userAdapter);
}
// після натискання на кнопку запускаємо UserActivity для додавання даних
public void add(View view) {
Intent intent = new Intent(this, UserActivity.class);
startActivity(intent);
}
@Override
public void onDestroy() {
super.onDestroy();
// Закриваємо підключення і курсор
db.close();
userCursor.close();
}
}
При натисканні на кнопку запускається UserActivity, при цьому не передається жодного id, тобто в UserActivity id буде рівним нулю, що означає, що буде виконуватися додавання даних:
public void add(View view){
Intent intent = new Intent(this, UserActivity.class);
startActivity(intent);
}
Інша ситуація виникає при обробнику натискання на елемент списку — при натисканні також запускається UserActivity, але тепер передається id вибраної запису:
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Intent intent = new Intent(getApplicationContext(), UserActivity.class);
intent.putExtra("id", id);
startActivity(intent);
}
Запустимо додаток і натиснемо на кнопку, яка повинна перенаправляти на UserActivity:

Під час натискання в MainActivity на елемент списку цей елемент потрапить на UserActivity, де його можна буде видалити або підредагувати:
