24

我正在查询 CallLog 内容提供程序并需要检测列类型。

在 Honeycomb 和更新版本(API 级别 11+)中,您可以通过调用Cursor.getType(int columnIndex)返回以下类型之一的方法来获取列首选数据类型:

  • FIELD_TYPE_NULL (0)
  • FIELD_TYPE_INTEGER (1)
  • FIELD_TYPE_FLOAT (2)
  • FIELD_TYPE_STRING (3)
  • FIELD_TYPE_BLOB (4)

如何在预 Honeycomb <11 设备上完成此操作?

我尝试了以下方法:

for ( int i = 0; i < cursor.getColumnCount(); i++ ) {    

    int columnType = -1;
    try {
        cursor.getInt( i );
        columnType = Cursor.FIELD_TYPE_INTEGER;

    } catch ( Exception ignore ) {

        try {
            cursor.getString( i );
            columnType = Cursor.FIELD_TYPE_STRING;

        } catch ( Exception ignore1 ) {

            try {
                cursor.getFloat( i );
                columnType = Cursor.FIELD_TYPE_FLOAT;

            } catch ( Exception ignore2 ) {

                try {                                             
                  cursor.getBlob( i );
                  columnType = Cursor.FIELD_TYPE_BLOB;

                } catch ( Exception ignore3 ) {

                     columnType = Cursor.FIELD_TYPE_NULL;
                }
           }
       }
   }

}

但是,不会抛出异常。数据始终以您要检查的第一种类型进行转换,在本例中为 getInt()。这意味着,如果列类型为Integer但所有其他类型为0 ,我会得到正确的值。

为什么我不查看文档来检查存储的类型?这些列因设备制造商而异,并非所有列都记录在案,请参阅此问题:如何处理 ContentProviders 中与制造商相关的差异?

有任何想法吗?

4

4 回答 4

12

扩展 Juan 的答案,这里是我对 API 11 方法 Cursor.getType(int i) 的替换 - 用于由 SQL 查询重新调整的游标

public class DbCompat {

    protected static final int FIELD_TYPE_BLOB = 4;
    protected static final int FIELD_TYPE_FLOAT = 2;
    protected static final int FIELD_TYPE_INTEGER = 1;
    protected static final int FIELD_TYPE_NULL = 0;
    protected static final int FIELD_TYPE_STRING = 3;

    static int getType(Cursor cursor, int i) throws Exception {
        SQLiteCursor sqLiteCursor = (SQLiteCursor) cursor;
        CursorWindow cursorWindow = sqLiteCursor.getWindow();
        int pos = cursor.getPosition();
        int type = -1;
        if (cursorWindow.isNull(pos, i)) {
            type = FIELD_TYPE_NULL;
        } else if (cursorWindow.isLong(pos, i)) {
            type = FIELD_TYPE_INTEGER;
        } else if (cursorWindow.isFloat(pos, i)) {
            type = FIELD_TYPE_FLOAT;
        } else if (cursorWindow.isString(pos, i)) {
            type = FIELD_TYPE_STRING;
        } else if (cursorWindow.isBlob(pos, i)) {
            type = FIELD_TYPE_BLOB;
        }

        return type;
    }
}

要点:https ://gist.github.com/kassim/c340cbfc5243db3a4826

于 2013-12-19T15:31:04.507 回答
10

当光标位于有效行中时,您可以使用此代码:

CursorWrapper cw = (CursorWrapper)cursor;

Class<?> cursorWrapper = CursorWrapper.class;
Field mCursor = cursorWrapper.getDeclaredField("mCursor");
mCursor.setAccessible(true);
AbstractWindowedCursor abstractWindowedCursor = (AbstractWindowedCursor)mCursor.get(cw);
CursorWindow cursorWindow = abstractWindowedCursor.getWindow();
int pos = abstractWindowedCursor.getPosition();
for ( int i = 0; i < cursor.getColumnCount(); i++ ) {
    String type = null;
    if (cursorWindow.isNull(pos, i)) {
        type = "Cursor.FIELD_TYPE_NULL";
    } else if (cursorWindow.isLong(pos, i)) {
        type = "Cursor.FIELD_TYPE_INTEGER";
    } else if (cursorWindow.isFloat(pos, i)) {
        type = "Cursor.FIELD_TYPE_FLOAT";
    } else if (cursorWindow.isString(pos, i)) {
        type = "Cursor.FIELD_TYPE_STRING";
    } else if (cursorWindow.isBlob(pos, i)) {
        type = "Cursor.FIELD_TYPE_BLOB";
    }
}

请注意,Cursor.FIELD_TYPE_* 常量值是从 HONEYCOMB 开始定义的。

于 2013-09-24T00:45:15.483 回答
0

有些东西可能会起作用:http: //developer.android.com/reference/android/database/DatabaseUtils.html cursorRowToContentValues

将复制 ContentValues 对象中的行。然后,您可以调用 ContentValues.get(),它会为您提供一个对象。然后您可以查看该对象的类。

编辑

根据 DatabaseUtils 的源代码,对象要么是 blob,要么是字符串。

编辑 2

但是,如果您的光标是 WindowedCursor,则它具有了解对象类型的方法。(isBlob、isString、isLong...)

于 2012-08-23T14:34:05.297 回答
0

我过去也遇到过同样的问题。我用一个非常好的解决方案解决了这个问题。将此与您的需求相匹配。就我而言,我有大量不同的对象,它们都同步到云中的服务器。它们都有共同的属性,因此它们都继承自一个共同的 BaseObject。这个对象有一个方法,它接受一个游标作为参数并返回一个相同类型的新对象,因此从它继承的每个对象都用它的扩展属性覆盖这个方法。

*请注意,这种方法不需要继承对象。这只是一种更聪明的方法。只要您在需要采用 DB 形式的所有对象中都有相同的方法,这将起作用,因为您最终将能够看到。

让我说明一下:

我们的基础对象。

public class BaseObject{

    protected int number;
    protected String text;

    public <T extends BaseObject> T setObject(Cursor c) {
        number = c.getInt(cur.getColumnIndexOrThrow(COLUMN_NAME_FOR_NUMBER));
        text = c.getString(cur.getColumnIndexOrThrow(COLUMN_NAME_FOR_TEXT));

        return (T) this;
    }
}

从第一个继承的新对象。

public class Contact extends BaseObject{

    private String name;

    @Override
    public <T extends BaseObject> T setObject(Cursor c) {

        super.setObject(c);

        name = c.getString(cur.getColumnIndexOrThrow(COLUMN_NAME_FOR_NAME));

        return (T) this;
    }
}

最后,在您的数据库中,通过调用通用方法“getAllObjects”并传递您想要查询的类类型以及查询的其他参数,就可以很容易地询问您想要的数据:

public synchronized <T extends BaseObject> ArrayList<T> getObjectsForClass(final Class<T> classType,
        String selection, String[] selectionArgs, String sort, String limit) {

    ArrayList<T> objects = null;

    if (db == null || !db.isOpen()) {
        db = getWritableDatabase();
    }

    objects = new ArrayList<T>();

    Cursor c = null;
    T object;
    try {
        object = classType.newInstance();

        String table = object.getTable();

        StringBuilder tableSb = new StringBuilder();
        tableSb.append(table).append(" INNER JOIN ").append(Constants.DB_BASE_OBJECT_TABLE)
                .append(" ON ").append(table).append(".").append(BaseObject.DB_OBJECT_ID_KEY).append(" = ")
                .append(Constants.DB_BASE_OBJECT_TABLE).append(".")
                .append(BaseObject.DB_ID_KEY);

        c = db.query(tableSb.toString(), null, selection, selectionArgs, null, null, sort, limit);

        if (c.getCount() > 0) {
            c.moveToFirst();
            while (!c.isAfterLast()) {

                object = classType.newInstance();
                object.setObject(c);
                objects.add(object);

                c.moveToNext();
            }
        }

    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }
    c.close();

    return objects;
}

你去吧。一种从数据库中获取任何对象并在运行时成功地将其转换为对象或对象数组的通用方法。

笔记:

  • 每个对象都应该有一个方法 getTable() 以便能够查询正确的表
  • 如您所见,在这种方法中还有一个 OODB 连接。您可以使用相同的方法,只需查询所有项目 (SELECT * FROM...)

希望能帮助到你。回答问题或疑问。

于 2012-08-27T13:02:38.960 回答