greendao源码解析

NOTE:greendao由Markus Junginger创建(markus另一项大作:eventbus)

新的手机数据库 ObjectBox,代替SQLite,据说快10倍,同时支持kottlin支持RxJava 2
http://greenrobot.org/greendao/documentation/objectbox-compat/

greeddao 实现流程图:


image

greendao特点:

1.版本3.0开始,通过注解自动化构建
2.性能好,参考http://greenrobot.org/greendao/features//
3.基于orm 设计,Query builder,方便书写代码。

自动化构建

  在实体的DAO中,您将发现包含数据库表名称的table name
  以及包含所有属性常量的内部类属性(field columnname)。
  能在编译时,就检查错误。

注:操作表的时候,表名和属性最好用常量,或者dao 里面去找。

以下涉及内容:缓存机制,预加载,api使用,线程安全等几个方面讨论。

why 性能好?

1.速度(核心):[inserted, updated and loaded at several thousand entities per second] 增删改查,每秒几千entities;加上cache,有2万多entities每秒。
对比查询,Greendao使用ID进行的加载操作根本不需要与数据库联系。对缓存实体使用加载操作会导致每秒超过100000个实体查找的速率。
http://greenrobot.org/open-source/current-performance-figures/
对比OrmLite, and ActiveAndroid
greenDAO inserts and updates entities around 2 times faster, and loads entities around 4 times faster than ORMLite

2.session cache 缓存机制
默认多次查询,返回相同的Java objcet。(由identity scope决定)
如果要清除缓存:

daoSession.clear();
只清空当前ID scope
noteDao = daoSession.getNoteDao();
noteDao.detachAll();

实现缓存的方法。dao::loadAllFromCursor()中的关键方法

    if (cursor.moveToFirst()) {
            if (identityScope != null) {
                identityScope.lock();
                identityScope.reserveRoom(count);
            }

            try {
                if (!useFastCursor && window != null && identityScope != null) {
                    loadAllUnlockOnWindowBounds(cursor, window, list);
                } else {
                    do {
                        list.add(loadCurrent(cursor, 0, false));
                    } while (cursor.moveToNext());
                }
            } finally {
                if (identityScope != null) {
                    identityScope.unlock();
                }
            }
        }

只要指针还有数据,往list的添加loadCurrent(cursor, 0, false)

final protected T loadCurrent(Cursor cursor, int offset, boolean lock) {
        if (identityScopeLong != null) {
            if (offset != 0) {
                // Occurs with deep loads (left outer joins)
                if (cursor.isNull(pkOrdinal + offset)) {
                    return null;
                }
            }

            long key = cursor.getLong(pkOrdinal + offset);
            T entity = lock ? identityScopeLong.get2(key) : identityScopeLong.get2NoLock(key);
            if (entity != null) {
                return entity;
            } else {
                entity = readEntity(cursor, offset);
                attachEntity(entity);
                if (lock) {
                    identityScopeLong.put2(key, entity);
                } else {
                    identityScopeLong.put2NoLock(key, entity);
                }
                return entity;
            }
        } else if (identityScope != null) {
            K key = readKey(cursor, offset);
            if (offset != 0 && key == null) {
                // Occurs with deep loads (left outer joins)
                return null;
            }
            T entity = lock ? identityScope.get(key) : identityScope.getNoLock(key);
            if (entity != null) {
                return entity;
            } else {
                entity = readEntity(cursor, offset);
                attachEntity(key, entity, lock);
                return entity;
            }
        } else {
            // Check offset, assume a value !=0 indicating a potential outer join, so check PK
            if (offset != 0) {
                K key = readKey(cursor, offset);
                if (key == null) {
                    // Occurs with deep loads (left outer joins)
                    return null;
                }
            }
            T entity = readEntity(cursor, offset);
            attachEntity(entity);
            return entity;
        }
    }

再看有主键的情况, identityScopeLong.get2(key)

   public T get2(long key) {
        lock.lock();
        Reference<T> ref;
        try {
            ref = map.get(key);
        } finally {
            lock.unlock();
        }
        if (ref != null) {
            return ref.get();
        } else {
            return null;
        }
    }

public void put2(long key, T entity) {
        lock.lock();
        try {
            map.put(key, new WeakReference<T>(entity));
        } finally {
            lock.unlock();
        }
    }

数据缓存是通过identityScopeLong中成员变量map来获取的。现在看identityScopeLong是什么时候初始化的?

伪代码
xxEntityDaoConfig = daoConfigMap.get(xxEntityDao.class).clone();
        xxEntityDaoConfig.initIdentityScope(type);

   public void initIdentityScope(IdentityScopeType type) {
        if (type == IdentityScopeType.None) {
            identityScope = null;
        } else if (type == IdentityScopeType.Session) {
            if (keyIsNumeric) {
                identityScope = new IdentityScopeLong();
            } else {
                identityScope = new IdentityScopeObject();
            }
        } else {
            throw new IllegalArgumentException("Unsupported type: " + type);
        }
    }

由上可以看到,分三种identityScope = null;(identityScope==null)
若为主键,identityScope =IdentityScopeLong;若不为主键identityScope =IdentityScopeObject

参数IdentityScopeType是啥,默认创建的是IdentityScopeType.Session,也可以传NULL. 推荐默认IdentityScopeType.Session,NULL每次都要重新readEntity,创建新的entity,并将获取指针的内容放入entity中,不会获取缓存

    public DaoSession newSession() {
        return new DaoSession(db, IdentityScopeType.Session, daoConfigMap);
    }

    public DaoSession newSession(IdentityScopeType type) {
        return new DaoSession(db, type, daoConfigMap);
    }

3.eager loading 预加载
daomaster创建的时候,就registerDaoClass,创建各个dao daoconfig(),加载entity表结构。
之后每次新建daosession,就clone之前的dao配置(表结构等)
并初始化initIdentityScope。

  protected void registerDaoClass(Class<? extends AbstractDao<?, ?>> daoClass) {
        DaoConfig daoConfig = new DaoConfig(db, daoClass);
        daoConfigMap.put(daoClass, daoConfig);
    }
//daoconfig构造函数
 public DaoConfig(Database db, Class<? extends AbstractDao<?, ?>> daoClass) {
        this.db = db;
        try {
            this.tablename = (String) daoClass.getField("TABLENAME").get(null);
            Property[] properties = reflectProperties(daoClass);
            this.properties = properties;

            allColumns = new String[properties.length];

            List<String> pkColumnList = new ArrayList<String>();
            List<String> nonPkColumnList = new ArrayList<String>();
            Property lastPkProperty = null;
            for (int i = 0; i < properties.length; i++) {
                Property property = properties[i];
                String name = property.columnName;
                allColumns[i] = name;
                if (property.primaryKey) {
                    pkColumnList.add(name);
                    lastPkProperty = property;
                } else {
                    nonPkColumnList.add(name);
                }
            }
            String[] nonPkColumnsArray = new String[nonPkColumnList.size()];
            nonPkColumns = nonPkColumnList.toArray(nonPkColumnsArray);
            String[] pkColumnsArray = new String[pkColumnList.size()];
            pkColumns = pkColumnList.toArray(pkColumnsArray);

            pkProperty = pkColumns.length == 1 ? lastPkProperty : null;
            statements = new TableStatements(db, tablename, allColumns, pkColumns);

            if (pkProperty != null) {
                Class<?> type = pkProperty.type;
                keyIsNumeric = type.equals(long.class) || type.equals(Long.class) || type.equals(int.class)
                        || type.equals(Integer.class) || type.equals(short.class) || type.equals(Short.class)
                        || type.equals(byte.class) || type.equals(Byte.class);
            } else {
                keyIsNumeric = false;
            }

        } catch (Exception e) {
            throw new DaoException("Could not init DAOConfig", e);
        }
    }

4.active entities.丰富的API,透明灵活的处理持久层操作,增删改查等。
5.支持数据库加密SQLCipher

DevOpenHelper helper = new DevOpenHelper(this, "notes-db-encrypted.db");
Database db = helper.getEncryptedWritableDb("<your-secret-password>");
daoSession = new DaoMaster(db).newSession();

测试加密方法:
.getWritableDb() instead of .getEncryptedWritableDb(<password>)
加密存在的问题:
1) SQLCipher用了更多的锁,增加了死锁的风险
2)抛出的异常为非android.database.SQLException
3)SQLCipher从3.5.0版本开始正式支持Android N

6.小于100k,对apk size不会产生大的影响
7.开源,社区活跃

为何方便书写

1.代码书写使用builder代替原生sql,具体见http://greenrobot.org/greendao/documentation/queries/
如:
First name is "Joe" AND (year of birth is greater than 1970 OR (year of birth is 1970 AND month of birth is equal to or greater than 10

QueryBuilder<User> qb = userDao.queryBuilder();
qb.where(Properties.FirstName.eq("Joe"),
qb.or(Properties.YearOfBirth.gt(1970),
qb.and(Properties.YearOfBirth.eq(1970), Properties.MonthOfBirth.ge(10))));
List<User> youngJoes = qb.list();
// order by last name
queryBuilder.orderAsc(Properties.LastName);
// in reverse
queryBuilder.orderDesc(Properties.LastName);
// order by last name and year of birth
queryBuilder.orderAsc(Properties.LastName).orderDesc(Properties.YearOfBirth);

** 可以添加个数的限制
[limit(int)]Limits the number of results returned by the query.

[offset(int)]Sets the offset for query results in combination with limit(int). The first offset results are skipped and the total number of results will be limited by limit(int). You cannot use offset withoutlimit(int).
**

2.类型,支持类型转换
默认支持的类型如下:

boolean, Boolean
int, Integer
short, Short
long, Long
float, Float
double, Double
byte, Byte
byte[]
String
Date

其他建议:
1)最好使用数据库原本支持的类型。
2)除了自增长的数据,@Entity最好都用String类型(String 映射到TEXT),否则date等类型在做转换的时候,有可能会出现异常。(或者Date用INTEGER或Long代替)
3)枚举类型映射到int,应该用int类型代替。

  1. 使用类型转换,尽量用透明的类型代替,如:
    boolean 映射 INTEGER(0,1)
    Date 映射 (long) INTEGER
@Entity
public class User {
    @Id
    private Long id;

    @Convert(converter = RoleConverter.class, columnType = Integer.class)
    private Role role;

    public enum Role {
        DEFAULT(0), AUTHOR(1), ADMIN(2);
        
        final int id;
        
        Role(int id) {
            this.id = id;
        }
    }

    public static class RoleConverter implements PropertyConverter<Role, Integer> {
        @Override
        public Role convertToEntityProperty(Integer databaseValue) {
            if (databaseValue == null) {
                return null;
            }
            for (Role role : Role.values()) {
                if (role.id == databaseValue) {
                    return role;
                }
            }
            return Role.DEFAULT;
        }

        @Override
        public Integer convertToDatabaseValue(Role entityProperty) {
            return entityProperty == null ? null : entityProperty.id;
        }
    }
}
  1. unique()和uniqueOrThrow()
    unique会返回结果或null,返回单一数据。
    uniqueOrThrow会抛出DaoException

4.list()listLazy() listLazyUncached() listIterator()
除了list(),其他的方法 必须关闭。正常的使用,
listLazy 使用内存缓存,内存有不会去查库。
listLazyUncached 不用内存缓存,会查库
listIterator,数据不会缓存,使用才会load data。

关闭原因及方法:
To load data on-demand, it holds a reference to a database cursor. This is the reason you must ensure to close the lazy lists and iterators (typically in a try/finally block).

5.builder支持查询多次,复用

// fetch users with Joe as a first name born in 1970
Query<User> query = userDao.queryBuilder().where(
    Properties.FirstName.eq("Joe"), Properties.YearOfBirth.eq(1970)
).build();
List<User> joesOf1970 = query.list();
// using the same Query object, we can change the parameters
// to search for Marias born in 1977 later:
query.setParameter(0, "Maria");
query.setParameter(1, 1977);
List<User> mariasOf1977 = query.list();

6.Raw queries
1)WhereCondition.StringCondition

Query<User> query = userDao.queryBuilder().where(
  new StringCondition("_ID IN " +
    "(SELECT USER_ID FROM USER_MESSAGE WHERE READ_FLAG = 0)")
).build();

2) queryRaw() queryRawCreate() 区别及用法
List<T> queryRaw(String where, String... selectionArg)
Query<T> queryRawCreate(String where, Object... selectionArg)

Query<User> query = userDao.queryRawCreate(
  ", GROUP G WHERE G.NAME=? AND T.GROUP_ID=G._ID", "admin"
);

7.buildDelete()
返回 DeleteQuery,用于之后调用,针对相同规则的entities
注意,bulk deletes不会影响entities,可以恢复。

Keep in mind, that bulk deletes currently do not affect entities in the identity scope, for example you could “resurrect” deleted entities if they have been cached before and are accessed by their ID (load method).

8.调试
可协助打印出sql语句,在其他数据库工具里执行。

QueryBuilder.LOG_SQL = true;
QueryBuilder.LOG_VALUES = true;

9.join :如:一个user多个地址

QueryBuilder<User> queryBuilder = userDao.queryBuilder();
queryBuilder.join(Address.class, AddressDao.Properties.userId)
  .where(AddressDao.Properties.Street.eq("Sesame Street"));
List<User> users = queryBuilder.list();

其他技巧:
1)As a rule of thumb,用application scope操作,保存数据。
2)android sqlite 推荐主键用Long,
若创建唯一索(unique = true)

@Id
private Long id;

@Index(unique = true)
private String key;

3)multiple sessions refer to the same database connection

支持多线程操作

(仅仅指同一个daosession 的query builder以及dao的操作是线程安全的,daosession的创建必须保证时单例同步的,其他地方避免用synchronized)
核心:
i.query builder,通过同步锁和map实现,如果当前线程id
和map中存储的线程ID一致,则复制之前的参数;否则存储在map中。等待下次相同线程调用。(由于有同步锁的存在,建议dao.queryBuilder()最好在同一个方法区里。)
ii.数据缓存,通过dao中的IdentityScopeLong或者IdentityScopeObject来实现。具体通过ReentrantLock来确保线程安全。

  public IdentityScopeObject() {
        map = new HashMap<K, Reference<T>>();
        lock = new ReentrantLock();
    }

1)query builder在创建时会调用**forCurrentThread() **,然后在当前线程创建Query instance 。(这种设置比较安全,原因:
每次调用用forCurrentThread(),the parameters are set to the initial parameters at the time the query was built using its builder.
参数被设置到之前创建query的初始化参数里。)

2)避免用synchronized,竞争性交易(concurrent transactions)会导致死锁

//伪代码,接着看都发生了什么?
dao.queryBuilder()
.where(xx.eq(xx));
.list()

QueryBuilder::list

 public List<T> list() {
        return build().list();
    }

   /**
     * Builds a reusable query object (Query objects can be executed more efficiently than creating a QueryBuilder for
     * each execution.
     */
    public Query<T> build() {
        StringBuilder builder = createSelectBuilder();
        int limitPosition = checkAddLimit(builder);
        int offsetPosition = checkAddOffset(builder);

        String sql = builder.toString();
        checkLog(sql);

        return Query.create(dao, sql, values.toArray(), limitPosition, offsetPosition);
    }

  static <T2> Query<T2> create(AbstractDao<T2, ?> dao, String sql, Object[] initialValues, int limitPosition,
                                 int offsetPosition) {
        QueryData<T2> queryData = new QueryData<T2>(dao, sql, toStringArray(initialValues), limitPosition,
                offsetPosition);
        return queryData.forCurrentThread();
    }

每次执行list()--build()--新建QueryData,最终调用QueryData .forCurrentThread()(如果是新的线程就创建Query,否则复制之前配的参数,并返回。通过synchronized,保证各自线程里的querydata互不干扰,从而达到querybuilder复用,且线程安全)

  /**
     * Note: all parameters are reset to their initial values specified in {@link QueryBuilder}.
     */
    Q forCurrentThread() {
        // Process.myTid() seems to have issues on some devices (see Github #376) and Robolectric (#171):
        // We use currentThread().getId() instead (unfortunately return a long, can not use SparseArray).
        // PS.: thread ID may be reused, which should be fine because old thread will be gone anyway.
        long threadId = Thread.currentThread().getId();
        synchronized (queriesForThreads) {
            WeakReference<Q> queryRef = queriesForThreads.get(threadId);
            Q query = queryRef != null ? queryRef.get() : null;
            if (query == null) {
                gc();
                query = createQuery();
                queriesForThreads.put(threadId, new WeakReference<Q>(query));
            } else {
                System.arraycopy(initialValues, 0, query.parameters, 0, initialValues.length);
            }
            return query;
        }
    }

接着看list();

   /** Executes the query and returns the result as a list containing all entities loaded into memory. */
    public List<T> list() {
        checkThread();//检查是否当前线程,否则抛DaoException
        Cursor cursor = dao.getDatabase().rawQuery(sql, parameters);
        return daoAccess.loadAllAndCloseCursor(cursor);
    }

这里的dao.getDatabase()是哪个数据库?回到最开始伪代码调用的地方 dao.queryBuilder()生成QueryBuilder,这里的dao 都是org.greenrobot.greendao.AbstractDao<T, K> ,最终发现QueryBuilder的成员变量dao就是AbstractDao, 也是QueryData的dao,并通过QueryData ::createQuery() 获取AbstractDao。

 public QueryBuilder<T> queryBuilder() {
        return QueryBuilder.internalCreate(this);
    }

  protected QueryBuilder(AbstractDao<T, ?> dao, String tablePrefix) {
        this.dao = dao;
        this.tablePrefix = tablePrefix;
        values = new ArrayList<Object>();
        joins = new ArrayList<Join<T, ?>>();
        whereCollector = new WhereCollector<T>(dao, tablePrefix);
        stringOrderCollation = " COLLATE NOCASE";
    }

 @Override
        protected Query<T2> createQuery() {
            return new Query<T2>(this, dao, sql, initialValues.clone(), limitPosition, offsetPosition);
        }

接着看AbstractDao是怎么来的,以及他的getDatabase()获取的数据库。可以从下面代码,看出是DaoConfig的db

 public AbstractDao(DaoConfig config, AbstractDaoSession daoSession) {
        this.config = config;
        this.session = daoSession;
        db = config.db;
        isStandardSQLite = db.getRawDatabase() instanceof SQLiteDatabase;
        identityScope = (IdentityScope<K, T>) config.getIdentityScope();
        if (identityScope instanceof IdentityScopeLong) {
            identityScopeLong = (IdentityScopeLong<T>) identityScope;
        } else {
            identityScopeLong = null;
        }
        statements = config.statements;
        pkOrdinal = config.pkProperty != null ? config.pkProperty.ordinal : -1;
    }
伪代码
        xxEntityDaoConfig = daoConfigMap.get(xxEntityDao.class).clone();
        xxEntityDaoConfig.initIdentityScope(type);

接着看daoConfigMap

  public DaoSession newSession() {
        return new DaoSession(db, IdentityScopeType.Session, daoConfigMap);
    }

    public DaoSession newSession(IdentityScopeType type) {
        return new DaoSession(db, type, daoConfigMap);
    }

源头daomaster:AbstractDaoMaster

    public AbstractDaoMaster(Database db, int schemaVersion) {
        this.db = db;
        this.schemaVersion = schemaVersion;

        daoConfigMap = new HashMap<Class<? extends AbstractDao<?, ?>>, DaoConfig>();
    }

    protected void registerDaoClass(Class<? extends AbstractDao<?, ?>> daoClass) {
        DaoConfig daoConfig = new DaoConfig(db, daoClass);
        daoConfigMap.put(daoClass, daoConfig);
    }

DaoMaster构造函数会注册各个@entity,生成各种entity的DaoConfig。当每次新建daosession都会新建各种dao,再通过DaoConfig .clone(),复制一份新的相同的DaoConfig。(所以这就是为什么greendao的特性第3点:eager loading,预加载)

最开始要查找的db,即DaoMaster的db

public DaoMaster(android.database.sqlite.SQLiteDatabase db) {
        this(new StandardDatabase(db));
    }
//StandardDatabase是org.greenrobot.greendao.database.Database db的默认实现代理类
 public DaoMaster(org.greenrobot.greendao.database.Database db) {
        super(db, SCHEMA_VERSION);
        registerDaoClass(xxEntityDao.class);
        registerDaoClass(yyDao.class);
      .........
    }

所以最终querybuild::list()中要找的db即是StandardDatabase

    public List<T> list() {
        checkThread();//检查是否当前线程,否则抛DaoException
        Cursor cursor = dao.getDatabase().rawQuery(sql, parameters);
        return daoAccess.loadAllAndCloseCursor(cursor);
    }

StandardDatabase:: rawQuery

    public StandardDatabase(SQLiteDatabase delegate) {
        this.delegate = delegate;
    }
   @Override
    public Cursor rawQuery(String sql, String[] selectionArgs) {
        return delegate.rawQuery(sql, selectionArgs);
    }

delegate是android.database.sqlite.SQLiteDatabase
通过DaoMaster.OpenHelper的实现类来生成。

    public SQLiteDatabase getWritableDatabase() {
        synchronized (this) {
            return getDatabaseLocked(true);
        }
    }

DaoMaster.OpenHelper extends DatabaseOpenHelper extends SQLiteOpenHelper
最终是android.database.sqlite.SQLiteOpenHelper生成的SQLiteDatabase

    public Cursor rawQuery(String sql, String[] selectionArgs) {
        return rawQueryWithFactory(null, sql, selectionArgs, null, null);

  public Cursor rawQueryWithFactory(
            CursorFactory cursorFactory, String sql, String[] selectionArgs,
            String editTable, CancellationSignal cancellationSignal) {
        acquireReference();
        try {
            SQLiteCursorDriver driver = new SQLiteDirectCursorDriver(this, sql, editTable,
                    cancellationSignal);
            return driver.query(cursorFactory != null ? cursorFactory : mCursorFactory,
                    selectionArgs);
        } finally {
            releaseReference();
        }
    }
    }

接着看list()方法。中daoAccess.loadAllAndCloseCursor(cursor);

public final class InternalQueryDaoAccess<T> 
  public List<T> loadAllAndCloseCursor(Cursor cursor) {
        return dao.loadAllAndCloseCursor(cursor);
    }
   protected List<T> loadAllAndCloseCursor(Cursor cursor) {
        try {
            return loadAllFromCursor(cursor);
        } finally {
            cursor.close();
        }
    }

  /** Reads all available rows from the given cursor and returns a list of entities. */
    protected List<T> loadAllFromCursor(Cursor cursor) {
        int count = cursor.getCount();
        if (count == 0) {
            return new ArrayList<T>();
        }
        List<T> list = new ArrayList<T>(count);
        CursorWindow window = null;
        boolean useFastCursor = false;
        if (cursor instanceof CrossProcessCursor) {
            window = ((CrossProcessCursor) cursor).getWindow();
            if (window != null) { // E.g. Robolectric has no Window at this point
                if (window.getNumRows() == count) {
                    cursor = new FastCursor(window);
                    useFastCursor = true;
                } else {
                    DaoLog.d("Window vs. result size: " + window.getNumRows() + "/" + count);
                }
            }
        }

        if (cursor.moveToFirst()) {
            if (identityScope != null) {
                identityScope.lock();
                identityScope.reserveRoom(count);
            }

            try {
                if (!useFastCursor && window != null && identityScope != null) {
                    loadAllUnlockOnWindowBounds(cursor, window, list);
                } else {
                    do {
                        list.add(loadCurrent(cursor, 0, false));
                    } while (cursor.moveToNext());
                }
            } finally {
                if (identityScope != null) {
                    identityScope.unlock();
                }
            }
        }
        return list;
    }

上面最后一段主要是判断是否是跨进程指针,当然一般我们不会有这种情况。
重点看identityScope.lock()和 identityScope.unlock();
identityScope是DaoConfig的成员变量IdentityScope,是由创建daosession的时候和dao一起初始化的。具体如何实现加锁,可参考ReentrantLock的实现原理。

*遗留问题*:greendao的缓存,只是大批量查询时,对象复用,不重新创建。

但是cursor指针还是会遍历,只是少了获取指针上面的数据。
具体如下:

   do {
                        list.add(loadCurrent(cursor, 0, false));
                    } while (cursor.moveToNext());

//loadCurrent, 有主键的情况
 T entity = lock ? identityScopeLong.get2(key) : identityScopeLong.get2NoLock(key);
            if (entity != null) {
                return entity;
            } else {
                entity = readEntity(cursor, offset);
                attachEntity(entity);
                if (lock) {
                    identityScopeLong.put2(key, entity);
                } else {
                    identityScopeLong.put2NoLock(key, entity);
                }
                return entity;
            }
//xxEntityDao
public void readEntity(Cursor cursor, xxEntity entity, int offset) {
        entity.setxx(cursor.isNull(offset + 0) ? null : cursor.getLong(offset + 0));
        entity.setyy(cursor.isNull(offset + 1) ? null : cursor.getString(offset + 1));
    ...
     }

贴下cursor.getString方法。这里的cursor是

    public SQLiteOpenHelper(Context context, String name, CursorFactory factory, int version,
            int minimumSupportedVersion, DatabaseErrorHandler errorHandler) {
        if (version < 1) throw new IllegalArgumentException("Version must be >= 1, was " + version);

        mContext = context;//一般为application context
        mName = name;  // 默认数据库名字,可以带路径
        mFactory = factory;  //默认为null,可以自定义
        mNewVersion = version;
        mErrorHandler = errorHandler;
        mMinimumSupportedVersion = Math.max(0, minimumSupportedVersion);
    }

    public SQLiteDatabase getReadableDatabase() {
        synchronized (this) {
            return getDatabaseLocked(false);
        }
    }

 db = mContext.openOrCreateDatabase(mName, mEnableWriteAheadLogging ?
                                Context.MODE_ENABLE_WRITE_AHEAD_LOGGING : 0,
                                mFactory, mErrorHandler);


这里的contenxt如果是application,则对象是ContextImpl实例

 @Override
    public SQLiteDatabase openOrCreateDatabase(String name, int mode, CursorFactory factory) {
        return openOrCreateDatabase(name, mode, factory, null);
    }

    @Override
    public SQLiteDatabase openOrCreateDatabase(String name, int mode, CursorFactory factory,
            DatabaseErrorHandler errorHandler) {
        checkMode(mode);
        File f = getDatabasePath(name);
        int flags = SQLiteDatabase.CREATE_IF_NECESSARY;
        if ((mode & MODE_ENABLE_WRITE_AHEAD_LOGGING) != 0) {
            flags |= SQLiteDatabase.ENABLE_WRITE_AHEAD_LOGGING;
        }
        if ((mode & MODE_NO_LOCALIZED_COLLATORS) != 0) {
            flags |= SQLiteDatabase.NO_LOCALIZED_COLLATORS;
        }
        SQLiteDatabase db = SQLiteDatabase.openDatabase(f.getPath(), factory, flags, errorHandler);
        setFilePermissionsFromMode(f.getPath(), mode, 0);
        return db;
    }
//SQLiteDatabase 
    public static SQLiteDatabase openDatabase(String path, CursorFactory factory, int flags,
            DatabaseErrorHandler errorHandler) {
        SQLiteDatabase db = new SQLiteDatabase(path, flags, factory, errorHandler);
        db.open();
        return db;
    }

由以上代码,可以找到,最终查询操作的cursor是SQLiteOpenHelper获取的SQLiteDatabase中调用rawquery生成的,如果SQLiteOpenHelper::oncreat中没有用CursorFactory,则SQLiteDirectCursorDriver中可知,cursor是SQLiteCursor

  public Cursor query(CursorFactory factory, String[] selectionArgs) {
        final SQLiteQuery query = new SQLiteQuery(mDatabase, mSql, mCancellationSignal);
        final Cursor cursor;
        try {
            query.bindAllArgsAsStrings(selectionArgs);

            if (factory == null) {
                cursor = new SQLiteCursor(this, mEditTable, query);
            } else {
                cursor = factory.newCursor(mDatabase, this, mEditTable, query);
            }
        } catch (RuntimeException ex) {
            query.close();
            throw ex;
        }

        mQuery = query;
        return cursor;
    }

接着往下着,
SQLiteCursor extends AbstractWindowedCursor extends AbstractCursor


  public String getString(int columnIndex) {
        checkPosition();
        return mWindow.getString(mPos, columnIndex);
    }
//CursorWindow mWindow
    public String getString(int row, int column) {
        acquireReference();
        try {
            return nativeGetString(mWindowPtr, row - mStartPos, column);
        } finally {
            releaseReference();
        }
    }

cursor.getString()最终调用的是本地方法。

总结:greendao,所谓的性能好,只是在高并发,大数据量的时候少了创建各种对象的开销,以及省略了调用每列的指针的本地方法(第一列的数据获取,还是会调用指针的本地方法。)
如果连生成指针对象的这部操作也可以省掉,即list()方法中的rawquery以及遍历指针;肯定性能还会提升。但是这样的话,要注意保持key值不能重复,且唯一,这可能也是创建者用指针key表示的原因

附加:listLazy()流程:核心是LazyList,和list不同的是,list是循环遍历指针;LazyList是调用get的时候,才移动一下指针,并且获取值。其他流程和list一致(默认如果有缓存,则使用缓存entity)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容

  • rljs by sennchi Timeline of History Part One The Cognitiv...
    sennchi阅读 7,317评论 0 10
  • pyspark.sql模块 模块上下文 Spark SQL和DataFrames的重要类: pyspark.sql...
    mpro阅读 9,449评论 0 13
  • 世界这么大,你说我们要多有缘, 才会认识,才会遇见。 一个人久了,心也就麻木了。 有时,总会觉得很累,想休息,想旅...
    思考Ajax阅读 209评论 0 0
  • 《一本正经》读书会第一次分享记录——孟德 今天参加了《一本正经》第一次读书会。我分享了《颠覆者》这本书,这是一本互...
    天山云读书会阅读 273评论 0 1
  • 六日 树犹如此 风暴过后,许多树都断成两截。释迦摩尼总以树取喻,从其种子的抽芽说起,再到根部的延展深入,再到枝干的...
    宝宝脚抽筋阅读 229评论 0 0