Transmisión de blob binario a través de un proveedor de contenido

Tengo un proveedor de contenido que es personalizado a mi conjunto de aplicaciones de Android, y una de las cosas que necesita exponer es una pequeña (20-30 KiB) matriz de bytes. El URI para estos blobs se parece a:

content://my.authority/blob/# 

Donde # es el número de fila; El cursor resultante tiene la columna _id estándar y una columna de datos. Estoy utilizando un MatrixCursor en el método query() del proveedor:

 byte[] byteData = getMyByteData(); MatrixCursor mc = new MatrixCursor(COLUMNS); mc.addRow(new Object[] { id, byteData }); 

Más tarde, en la aplicación que consume los datos, hago:

 Cursor c = managedQuery(uri, null, null, null, null); c.moveToFirst(); byte[] data = c.getBlob(c.getColumnIndexOrThrow("data")); 

Sin embargo, los datos no contienen el contenido de mi matriz de bytes original; Más bien, contiene algo así como [B@435cc518 , que se parece más a la dirección de la matriz que el contenido. He intentado envolver la matriz de bytes en una implementación de java.sql.Blob , calculando que podría estar buscando eso, ya que el subsistema de proveedores de contenido fue escrito para ser fácil de usar con SQLite, pero no ayudó.

¿Alguien ha conseguido que esto funcione? Si los datos estaban en el sistema de archivos, hay métodos en ContentProvider que podría usar para proporcionar un InputStream ordenado al cliente, pero los datos que estoy intentando devolver viven como un recurso en el APK del proveedor de contenido.

No podrá utilizar un MatrixCursor para enviar la matriz de bytes. Esto se debe a que depende del método AbstractCursor#fillWindow que llena el CursorWindow utilizando Object#toString . Así que lo que está sucediendo es que el método toString matriz de bytes está siendo llamado y está almacenando que en lugar del contenido de la matriz de bytes que es lo que desea. La única manera alrededor de esto que puedo ver es implementar su propio cursor que llenará el CursorWindow apropiadamente para una matriz del byte.

Obtenga las fuentes de MatrixCursor de Ice Cream Sandwich. Implementa getBlob correctamente.

El código fuente modificado está a continuación, si desea utilizarlo. Puedes usarlo en proyectos con Android 1.6+.

 /* * Copyright (C) 2007 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package yuku.androidsdk.android.database.icsmatrixcursor; import android.database.*; import java.util.*; /** * A mutable cursor implementation backed by an array of {@code Object}s. Use * {@link #newRow()} to add rows. Automatically expands internal capacity * as needed. */ public class MatrixCursor extends AbstractCursor { private final String[] columnNames; private Object[] data; private int rowCount = 0; private final int columnCount; /** * Constructs a new cursor with the given initial capacity. * * @param columnNames names of the columns, the ordering of which * determines column ordering elsewhere in this cursor * @param initialCapacity in rows */ public MatrixCursor(String[] columnNames, int initialCapacity) { this.columnNames = columnNames; this.columnCount = columnNames.length; if (initialCapacity < 1) { initialCapacity = 1; } this.data = new Object[columnCount * initialCapacity]; } /** * Constructs a new cursor. * * @param columnNames names of the columns, the ordering of which * determines column ordering elsewhere in this cursor */ public MatrixCursor(String[] columnNames) { this(columnNames, 16); } /** * Gets value at the given column for the current row. */ private Object get(int column) { if (column < 0 || column >= columnCount) { throw new CursorIndexOutOfBoundsException("Requested column: " + column + ", # of columns: " + columnCount); } if (mPos < 0) { throw new CursorIndexOutOfBoundsException("Before first row."); } if (mPos >= rowCount) { throw new CursorIndexOutOfBoundsException("After last row."); } return data[mPos * columnCount + column]; } /** * Adds a new row to the end and returns a builder for that row. Not safe * for concurrent use. * * @return builder which can be used to set the column values for the new * row */ public RowBuilder newRow() { rowCount++; int endIndex = rowCount * columnCount; ensureCapacity(endIndex); int start = endIndex - columnCount; return new RowBuilder(start, endIndex); } /** * Adds a new row to the end with the given column values. Not safe * for concurrent use. * * @throws IllegalArgumentException if {@code columnValues.length != * columnNames.length} * @param columnValues in the same order as the the column names specified * at cursor construction time */ public void addRow(Object[] columnValues) { if (columnValues.length != columnCount) { throw new IllegalArgumentException("columnNames.length = " + columnCount + ", columnValues.length = " + columnValues.length); } int start = rowCount++ * columnCount; ensureCapacity(start + columnCount); System.arraycopy(columnValues, 0, data, start, columnCount); } /** * Adds a new row to the end with the given column values. Not safe * for concurrent use. * * @throws IllegalArgumentException if {@code columnValues.size() != * columnNames.length} * @param columnValues in the same order as the the column names specified * at cursor construction time */ public void addRow(Iterable<?> columnValues) { int start = rowCount * columnCount; int end = start + columnCount; ensureCapacity(end); if (columnValues instanceof ArrayList<?>) { addRow((ArrayList<?>) columnValues, start); return; } int current = start; Object[] localData = data; for (Object columnValue : columnValues) { if (current == end) { // TODO: null out row? throw new IllegalArgumentException( "columnValues.size() > columnNames.length"); } localData[current++] = columnValue; } if (current != end) { // TODO: null out row? throw new IllegalArgumentException( "columnValues.size() < columnNames.length"); } // Increase row count here in case we encounter an exception. rowCount++; } /** Optimization for {@link ArrayList}. */ private void addRow(ArrayList<?> columnValues, int start) { int size = columnValues.size(); if (size != columnCount) { throw new IllegalArgumentException("columnNames.length = " + columnCount + ", columnValues.size() = " + size); } rowCount++; Object[] localData = data; for (int i = 0; i < size; i++) { localData[start + i] = columnValues.get(i); } } /** Ensures that this cursor has enough capacity. */ private void ensureCapacity(int size) { if (size > data.length) { Object[] oldData = this.data; int newSize = data.length * 2; if (newSize < size) { newSize = size; } this.data = new Object[newSize]; System.arraycopy(oldData, 0, this.data, 0, oldData.length); } } /** * Builds a row, starting from the left-most column and adding one column * value at a time. Follows the same ordering as the column names specified * at cursor construction time. */ public class RowBuilder { private int index; private final int endIndex; RowBuilder(int index, int endIndex) { this.index = index; this.endIndex = endIndex; } /** * Sets the next column value in this row. * * @throws CursorIndexOutOfBoundsException if you try to add too many * values * @return this builder to support chaining */ public RowBuilder add(Object columnValue) { if (index == endIndex) { throw new CursorIndexOutOfBoundsException( "No more columns left."); } data[index++] = columnValue; return this; } } // AbstractCursor implementation. @Override public int getCount() { return rowCount; } @Override public String[] getColumnNames() { return columnNames; } @Override public String getString(int column) { Object value = get(column); if (value == null) return null; return value.toString(); } @Override public short getShort(int column) { Object value = get(column); if (value == null) return 0; if (value instanceof Number) return ((Number) value).shortValue(); return Short.parseShort(value.toString()); } @Override public int getInt(int column) { Object value = get(column); if (value == null) return 0; if (value instanceof Number) return ((Number) value).intValue(); return Integer.parseInt(value.toString()); } @Override public long getLong(int column) { Object value = get(column); if (value == null) return 0; if (value instanceof Number) return ((Number) value).longValue(); return Long.parseLong(value.toString()); } @Override public float getFloat(int column) { Object value = get(column); if (value == null) return 0.0f; if (value instanceof Number) return ((Number) value).floatValue(); return Float.parseFloat(value.toString()); } @Override public double getDouble(int column) { Object value = get(column); if (value == null) return 0.0d; if (value instanceof Number) return ((Number) value).doubleValue(); return Double.parseDouble(value.toString()); } @Override public byte[] getBlob(int column) { Object value = get(column); return (byte[]) value; } @Override public boolean isNull(int column) { return get(column) == null; } } 

Necesita reemplazar fillWindow en su implementación de AbstractCursor. Aquí hay uno que funciona en un proveedor de contenido para JUST BLOBS

  public void fillWindow(int position, CursorWindow window) { if (position < 0 || position >= getCount()) { return; } window.acquireReference(); try { int oldpos = mPos; mPos = position - 1; window.clear(); window.setStartPosition(position); int columnNum = getColumnCount(); window.setNumColumns(columnNum); while (moveToNext() && window.allocRow()) { for (int i = 0; i < columnNum; i++) { byte [] field = getBlob(i); if (field != null) { if (!window.putBlob(field, mPos, i)) { window.freeLastRow(); break; } } else { if (!window.putNull(mPos, i)) { window.freeLastRow(); break; } } } } mPos = oldpos; } catch (IllegalStateException e){ // simply ignore it } finally { window.releaseReference(); } } 
  • ¿Cómo puedo crear una vista de contenedor personalizada en Android?
  • XML vs setImageDrawable / setImageBitmap
  • Captura de audio / video en PhoneGap / Córdoba utilizando un complemento de Java externo
  • Facturación de Android v3 - sin firma
  • Java / Eclipse - No más archivo R nunca
  • Descargar imagen de Android desde el servidor y guardar en sdcard sin utilizar BitmapFactory
  • Los datos permanecen en el almacén de datos local después de desenfilar
  • Biblioteca Java usando selectores css para analizar XML
  • ¿Cómo calcular el tiempo restante de la batería del teléfono Android?
  • Cómo configurar Proguard para mantener los métodos estáticos con diferentes tiempos de retorno
  • ¿Alguien tiene puntos de referencia (código y resultados) que comparan el rendimiento de las aplicaciones de Android escritas en Xamarin C # y Java?
  • FlipAndroid es un fan de Google para Android, Todo sobre Android Phones, Android Wear, Android Dev y Aplicaciones para Android Aplicaciones.