Статическое соединение с базой данных SQLite - PullRequest
0 голосов
/ 13 мая 2018

Я хотел бы попросить любого помочь мне изменить код подключения к моей базе данных.Теперь у меня есть часть одного из моих классов методов, и он будет устанавливать соединение каждый раз, когда я его вызываю (что, я знаю, очень неэффективно).Но когда я пытаюсь убрать часть соединения из метода, это нарушает код.Мне просто интересно, смогу ли я где-нибудь создать статический класс, чтобы я мог использовать его и в другом методе.

Большое спасибо за любую помощь заранее, это действительно ценится.

Вот код метода:

    public void getListDetail(){

   //listDetailData.clear();

    ShoppingListDatabase databaseConnection = new ShoppingListDatabase(this);
    final SQLiteDatabase db = databaseConnection.open();
    final ArrayList<ShoppingItem> lists = ShoppingListItemTable.selectAllItems(db, selectedID);
    databaseConnection.close();

    //create a list adapter and set adapter
    ShoppingListItemAdapter adapter = new ShoppingListItemAdapter(this, R.layout.activity_item_detail_list, lists);
    ListView_ListDetail = findViewById(R.id.ListView_ListDetail);
    ListView_ListDetail.setAdapter(adapter);
    adapter.notifyDataSetChanged();
   // ((ArrayAdapter<String>)ListView_ListDetail.getAdapter()).notifyDataSetChanged();
}

Класс базы данных:

package com.puyakul.prin.psychic_shopping;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.provider.ContactsContract;
import android.util.Log;

public class ShoppingListDatabase {

    private static final String TAG = "ShoppingListDatabase";
    private static final String DATABASE_NAME = "ShoppingListDatabase";
    private static final int DATABASE_VERSION = 3;

    private SQLiteDatabase mDb;
    private DatabaseHelper mDbHealper;
    private final Context mCtx;

    public ShoppingListDatabase(Context ctx){
        this.mCtx = ctx;
    }

    /**
     * DatabaseHelper class.
     *
     * Database helper class to manage connections with the database.
     */
    private static class DatabaseHelper extends SQLiteOpenHelper
    {
        DatabaseHelper(Context context){
            super(context, DATABASE_NAME, null, DATABASE_VERSION);
        }

        @Override
        public void onCreate(SQLiteDatabase db) {
            Log.d(TAG, "DatabaseHelper onCreate");
            db.execSQL(ShoppingListItemTable.CREATE_STATEMENT);
            db.execSQL(ShoppingListTable.CREATE_STATEMENT);
        }

        @Override
        public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
            Log.d(TAG, "DatabaseHelper onUpgrade");
            db.execSQL("DROP TABLE IF EXISTS " + ShoppingListItemTable.TABLE_NAME);
            db.execSQL("DROP TABLE IF EXISTS " + ShoppingListTable.TABLE_NAME);
            onCreate(db); //this will recreate the database as if it were new
        }
    }

    public SQLiteDatabase open(){
        mDbHealper = new DatabaseHelper(mCtx);
        mDb = mDbHealper.getReadableDatabase();
        return mDb;
    }

    public void close(){
        mDb = null;
    }

}

Я попытался объявить переменные в onCreate, чтобы я мог использовать их с другими методами вкласс как этот

    //================DATABASE CONNECTION=====================//
    private ShoppingListDatabase databaseConnection;
    private SQLiteDatabase db  = databaseConnection.open();


    private String selectedList;
    private int selectedID;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        this.getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
        setContentView(R.layout.activity_main_lists_detail);

      ShoppingListDatabase databaseConnection = new ShoppingListDatabase(this);
      SQLiteDatabase db = databaseConnection.open();
      lists = ShoppingListItemTable.selectAllItems(db, selectedID); 

и это ошибка

 Process: com.puyakul.prin.psychic_shopping, PID: 5635
    java.lang.RuntimeException: Unable to instantiate activity ComponentInfo{com.puyakul.prin.psychic_shopping/com.puyakul.prin.psychic_shopping.MainListsDetail}: java.lang.NullPointerException: Attempt to invoke virtual method 'android.database.sqlite.SQLiteDatabase com.puyakul.prin.psychic_shopping.ShoppingListDatabase.open()' on a null object reference
        at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2548)
        at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2707)
        at android.app.ActivityThread.-wrap12(ActivityThread.java)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1460)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:154)
        at android.app.ActivityThread.main(ActivityThread.java:6077)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:866)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:756)
     Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'android.database.sqlite.SQLiteDatabase com.puyakul.prin.psychic_shopping.ShoppingListDatabase.open()' on a null object reference
        at com.puyakul.prin.psychic_shopping.MainListsDetail.<init>(MainListsDetail.java:46)
        at java.lang.Class.newInstance(Native Method)
        at android.app.Instrumentation.newActivity(Instrumentation.java:1078)
        at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2538)

1 Ответ

0 голосов
/ 13 мая 2018

Открытие и закрытие соединения может быть неэффективным / дорогостоящим, поэтому, возможно, введите / используйте следующее: -

public SQLiteDatabase open(){
    if (mDb == null) {
        mDbHealper = new DatabaseHelper(mCtx);
        mDb = mDbHealper.getReadableDatabase();
    }
    return mDb;
}

или поочередно: -

public synchronized SQLiteDatabase open(){
    if (mDb == null) {
        mDbHealper = new DatabaseHelper(mCtx);
        mDb = mDbHealper.getReadableDatabase();
    }
    return mDb;
}

и никогда не вызывайте закрытие,кроме случаев, когда основная деятельность разрушается.Затем вы получите одно соединение.


Вот несколько примеров использования, основанных на вашем ShoppingListDatabase классе.

Сначала измененный класс с несколькими дополнительными методами внутри (addShoppingList, getAllShoppingListsAsCursor и logDBTables) и просто очень простая таблица shoppinglist : -

public class ShoppingListDatabase {

    private static final String TAG = "ShoppingListDatabase";
    private static final String DATABASE_NAME = "ShoppingListDatabase";
    private static final int DATABASE_VERSION = 3;

    private static final String TBNAME = "shoppinglist";
    public static final String COL_SHOPPINGLIST_NAME = "shoppinglist_name";

    private SQLiteDatabase mDb;
    private DatabaseHelper mDbHealper;
    private final Context mCtx;

    public ShoppingListDatabase(Context ctx){
        this.mCtx = ctx;
    }

    /**
     * DatabaseHelper class.
     *
     * Database helper class to manage connections with the database.
     */
    private static class DatabaseHelper extends SQLiteOpenHelper
    {
        DatabaseHelper(Context context){
            super(context, DATABASE_NAME, null, DATABASE_VERSION);
        }

        @Override
        public void onCreate(SQLiteDatabase db) {
            Log.d(TAG, "DatabaseHelper onCreate");
            //db.execSQL(ShoppingListItemTable.CREATE_STATEMENT);
            //db.execSQL(ShoppingListTable.CREATE_STATEMENT);
            String crtsql = "CREATE TABLE If NOT EXISTS " + TBNAME + "(" +
                    COL_SHOPPINGLIST_NAME + " TEXT " +
                    ")";
            db.execSQL(crtsql);
        }

        @Override
        public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
            Log.d(TAG, "DatabaseHelper onUpgrade");

            //db.execSQL("DROP TABLE IF EXISTS " + ShoppingListItemTable.TABLE_NAME);
            //db.execSQL("DROP TABLE IF EXISTS " + ShoppingListTable.TABLE_NAME);
            onCreate(db); //this will recreate the database as if it were new
        }
    }

    public synchronized SQLiteDatabase open(){
        if (mDb == null) {
            mDbHealper = new DatabaseHelper(mCtx);
            mDb = mDbHealper.getReadableDatabase();
        }
        return mDb;
    }

    public void close(){
        mDb = null;
    }

    public long addShoppingList(String name) {
        ContentValues cv = new ContentValues();
        cv.put(COL_SHOPPINGLIST_NAME,name);
        return mDb.insert(TBNAME,null,cv);
    }

    public Cursor getAllShoppingListAsCursor() {
        return mDb.query(TBNAME,
                null,
                null,
                null,
                null,null,
                null
        );
    }

    public void logDBTables() {
        Cursor csr = mDb.query("sqlite_master",null,null,null,null,null,null);
        while (csr.moveToNext()) {
            Log.d(TAG,"Item " +
                    csr.getString(csr.getColumnIndex("name")) +
                    " was created using " +
                    csr.getString(csr.getColumnIndex("sql"))
            );
        }
    }
}

Вот код из Activity (который использует соединение с БД различными способами)): -

public class MainActivity extends AppCompatActivity {

    ShoppingListDatabase mSL;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mSL = new ShoppingListDatabase(this);
        mSL.open();

        SO50312618();
        SO50312618_other();
        Cursor csr = mSL.getAllShoppingListAsCursor();
        while(csr.moveToNext()) {
            Log.d("SL INFO","Shopping List " + String.valueOf(csr.getPosition() + 1) +
                    " is " +
                    csr.getString(csr.getColumnIndex(ShoppingListDatabase.COL_SHOPPINGLIST_NAME))
            );
        }
    }

    private void SO50312618() {
        ShoppingListDatabase databaseConnection = new ShoppingListDatabase(this);
        SQLiteDatabase db = databaseConnection.open();
        databaseConnection.logDBTables();
    }

    private void SO50312618_other() {
        mSL.addShoppingList("List001");
        mSL.addShoppingList("List002");
    }
}

Результаты: -

05-13 05:36:24.290 4245-4245/soanswers.soanswers D/ShoppingListDatabase: Item android_metadata was created using CREATE TABLE android_metadata (locale TEXT)
    Item shoppinglist was created using CREATE TABLE shoppinglist(shoppinglist_name TEXT )
05-13 05:36:24.302 4245-4245/soanswers.soanswers D/SL INFO: Shopping List 1 is List001
    Shopping List 2 is List002
  • Все различные варианты использования будут использовать одно и то же соединение.
...