Использование загрузчика курсора с данными BLOB

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

Я работаю с приложением, которое хранит данные BLOB в базе данных. Данные представляют собой картинку, очень маленькую (менее 2 КБ). Я решил добавить их прямо в базу данных из-за размера.

Я все еще изучаю программирование, и я уверен, что упускаю что-то простое, но вот моя проблема. В моей базе данных много текстовых полей, и я извлекаю из них информацию с помощью этого метода:

private void fillData() {

    // Pulls the fields from the database and assigns them to the columns

    String[] from = new String[] { BorrowMeTable.COLUMN_ITEM,
            BorrowMeTable.COLUMN_NAME, BorrowMeTable.COLUMN_DATE };

    // The label for each textview
    int[] to = new int[] { R.id.label, R.id.pName, R.id.pDate };

    getLoaderManager().initLoader(0, null, this);
    adapter = new SimpleCursorAdapter(this, R.layout.bm_row, null, from,
            to, 0);

    setListAdapter(adapter);
}

Мой класс реализует LoaderManager.LoaderCallbacks и сопутствующие методы:

public Loader<Cursor> onCreateLoader(int id, Bundle args) {

    // Remove to show all records - This sets up the list to filter based on
    // items returned
    String mSelection = BorrowMeTable.COLUMN_BORROW_FLAG + "=?";

    // SQL request for only returned items to be shown
    String[] selectionArgs = { "Borrowed" }; // The actual argument

    String[] projection = { BorrowMeTable.COLUMN_ID,
            BorrowMeTable.COLUMN_ITEM, BorrowMeTable.COLUMN_NAME,
            BorrowMeTable.COLUMN_DATE };
    // mSelection & selectionArgs can be removed and set to null to display
    // all records
    CursorLoader cursorLoader = new CursorLoader(this,
            BorrowMeContentProvider.CONTENT_URI, projection, mSelection,
            selectionArgs, null);
    return cursorLoader;
}

Когда я пытаюсь добавить столбец, в котором есть данные BLOB-объекта, я получаю сообщение об ошибке, в котором говорится, что я не могу преобразовать BLOB-объект в строку:

06-30 15:13:36.152: E/AndroidRuntime(842): android.database.sqlite.SQLiteException: unknown error (code 0): Unable to convert BLOB to string

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

Может ли кто-нибудь помочь мне в правильном направлении? Я не нашел много в Интернете, посвященного загрузчику курсора и просмотру списка.

Спасибо

Обновление. Я создал новый вид привязки для простого адаптера курсора, но по-прежнему получаю следующее сообщение об ошибке:

07-02 20:42:53.350: E/AndroidRuntime(798): FATAL EXCEPTION: main
07-02 20:42:53.350: E/AndroidRuntime(798): android.database.sqlite.SQLiteException:           unknown error (code 0): Unable to convert BLOB to string
07-02 20:42:53.350: E/AndroidRuntime(798):  at  android.database.CursorWindow.nativeGetString(Native Method)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.database.CursorWindow.getString(CursorWindow.java:434)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.database.AbstractWindowedCursor.getString(AbstractWindowedCursor.java:51)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.database.CursorWrapper.getString(CursorWrapper.java:114)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.SimpleCursorAdapter.bindView(SimpleCursorAdapter.java:150)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.CursorAdapter.getView(CursorAdapter.java:250)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.AbsListView.obtainView(AbsListView.java:2159)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.ListView.measureHeightOfChildren(ListView.java:1246)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.ListView.onMeasure(ListView.java:1158)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.View.measure(View.java:15518)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.RelativeLayout.measureChild(RelativeLayout.java:666)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.RelativeLayout.onMeasure(RelativeLayout.java:477)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.View.measure(View.java:15518)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:4825)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.View.measure(View.java:15518)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.LinearLayout.measureVertical(LinearLayout.java:847)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.LinearLayout.onMeasure(LinearLayout.java:588)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.View.measure(View.java:15518)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:4825)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
07-02 20:42:53.350: E/AndroidRuntime(798):  at com.android.internal.policy.impl.PhoneWindow$DecorView.onMeasure(PhoneWindow.java:2176)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.View.measure(View.java:15518)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:1874)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1089)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1265)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:989)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:4351)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.Choreographer$CallbackRecord.run(Choreographer.java:749)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.Choreographer.doCallbacks(Choreographer.java:562)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.Choreographer.doFrame(Choreographer.java:532)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:735)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.os.Handler.handleCallback(Handler.java:725)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.os.Handler.dispatchMessage(Handler.java:92)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.os.Looper.loop(Looper.java:137)
07-02 20:42:53.350: E/AndroidRuntime(798):  at android.app.ActivityThread.main(ActivityThread.java:5041)
07-02 20:42:53.350: E/AndroidRuntime(798):  at java.lang.reflect.Method.invokeNative(Native Method)
07-02 20:42:53.350: E/AndroidRuntime(798):  at java.lang.reflect.Method.invoke(Method.java:511)
07-02 20:42:53.350: E/AndroidRuntime(798):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
07-02 20:42:53.350: E/AndroidRuntime(798):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
07-02 20:42:53.350: E/AndroidRuntime(798):  at dalvik.system.NativeStart.main(Native Method)

Это код, который я использовал для своего связующего представления:

public class MyViewBinder implements ViewBinder {

@Override
public boolean setViewValue(View view, Cursor cursor, int columnIndex) {
    // TODO Auto-generated method stub

    int viewId = view.getId();

    Log.i("ViewBinder: view", Integer.toString(viewId));
    Log.i("ViewBinder: name",cursor.getString(2));
    Log.i("ViewBinder: email",cursor.getString(3));
    Log.i("ViewBinder: photo",cursor.getBlob(4)==null?"NO Photo":"Has photo");

    switch(viewId){

    case R.id.pic_of_item:

        ImageView storedItem = (ImageView) view;

        byte[] blob = cursor.getBlob(columnIndex);

        // If the blob is null then pull it from database
        if (blob != null){

            // Decode the bitmap from the blob
            storedItem.setImageBitmap(BitmapFactory.decodeByteArray(blob, 0, blob.length));

            Log.i("RETURN TRUE BLOB",cursor.getString(3));

        } else {

            // If nothing is found use the camera icon from drawable 
            storedItem.setImageResource(R.drawable.take_camera_small);

            Log.i("RETURN TRUE NO BLOB",cursor.getString(3));

            return true;
        }

    }

    Log.i("RETURN FALSE",cursor.getString(3));

    return false;
}

}


person Vince    schedule 30.06.2013    source источник


Ответы (1)


SimpleCursorAdapter использует cursor.getString для извлечения значения из базы данных, как показано в исходный код. Для изображений он предполагает, что полученное значение String представляет собой целое число, указывающее на конкретный ресурс изображения (т. е. вызывает v.setImageResource(Integer.parseInt(value))). Следовательно, SimpleCursorAdapter не понимает, как обращаться с данными больших двоичных объектов, и вам нужно будет использовать свой собственный класс CursorAdapter, который выполняет правильное преобразование больших двоичных объектов в Bitmap для установки на ваш ImageView, вероятно, используя что-то вроде следующего кода в вашем bindView:

ImageView imageView = // get your ImageView
byte[] blob = cursor.getBlob(cursor.getColumnIndex(BorrowMeTable.COLUMN_ITEM));
Bitmap bitmap = BitmapFactory.decodeByteArray(blob, 0, blob.length);
imageView.setImageBitmap(bitmap);
person ianhanniballake    schedule 30.06.2013
comment
Мне удалось создать свой собственный видоискатель, используя шаги из статьи здесь но я все равно получаю сообщение об ошибке String to Blob - person Vince; 03.07.2013
comment
@ Винс, так ты настроил ViewBinder, у которого есть переключатель, ищущий BorrowMeTable.COLUMN_ITEM? getString вызывается только внутри bindView, если ваш ViewBinder возвращает false. Подтвердили ли вы (через операторы Log или отладку), что ваш ViewBinder обрабатывает изображение и возвращает true? - person ianhanniballake; 03.07.2013
comment
Я подтвердил через операторы журнала, что он возвращает истину, а затем ложь сразу после этого. Я разместил свой код View Binder выше - person Vince; 03.07.2013
comment
Еще раз спасибо за любую помощь, которую вы можете предоставить - я все еще новичок и пытаюсь разобраться в этом. - person Vince; 03.07.2013
comment
@Vince - похоже, вы возвращаете true только тогда, когда blob==null (ваш оператор else) - переместите этот оператор return true; из оператора else, чтобы оба пути возвращали true. - person ianhanniballake; 03.07.2013
comment
Вы, сэр, гений. Большое спасибо за помощь. - person Vince; 03.07.2013