GreenDao3.0 源码分析-DaoMaster和DaoSeesion

AbstractDaoMaster


从图中我们知道,DaoMaster和AbstractDaoMaster是父子类的关系,我们先看AbstractDaoMaster:


从上面类图中,我们知道daoMaster维护一个daoConfigMap以Dao.class为key维护一个映射关系,而且通过调用


我们可以知道 AbstractDaoMaster的功能主要职责之一是创建注册DaoConfig,并且维护他和dao之间的关系。

实现类DaoMaster的功能

功能一:创建所有表和删除所有表


通过调用XXXDAO的静态方法来进行建表和删表。

功能二:创建Session会话

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

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

两个创建的默认数据缓存的类型,一个是随我们定义的类型,IdentityScopeType定义了两个类型,后面再说。

功能三:注册Config:

public DaoMaster(Database db) {
        super(db, SCHEMA_VERSION);
        registerDaoClass(NoteDao.class);
    }

创建DaoMaster对象的时候以  Class<?> 来作为key,对DaoConfig进行配置缓存绑定。

功能四:定义内部类OpenHelper和DevOpenHelper。


Dao配置类DaoConfig

 源码对DaoConfig是这样评价的:在GreenDao内部使用,通过AbstractDaoMaster创建和持有其对象,存储一些DAO实体所必须的数据。这个类实例能通过以Dao class为Key ,使用map维护,通过key取出必要的信息。

也就是说,DaoConfig就是用来存储dao,必要的一些数据的。

DaoConfig,维护了以下的字段:


上图源码可知,DaoConfig 维护了数据库对象、数据库名称、Dao实体类表所有的属性对应关系等,并且这些字段都是在这个类进行初始化的,核心功能有两个。

根据xxxDao.class反射获得属性对象:

 private static Property[] reflectProperties(Class<? extends AbstractDao<?, ?>> daoClass)
            throws ClassNotFoundException, IllegalArgumentException, IllegalAccessException {
        Class<?> propertiesClass = Class.forName(daoClass.getName() + "$Properties");
        Field[] fields = propertiesClass.getDeclaredFields();

        ArrayList<Property> propertyList = new ArrayList<Property>();
        //根据反射的字段 拿出静态和公开的属性对象
        final int modifierMask = Modifier.STATIC | Modifier.PUBLIC;
        for (Field field : fields) {
            //有些工具可能会引入其他字段,可以忽略它们
            if ((field.getModifiers() & modifierMask) == modifierMask) {
                Object fieldValue = field.get(null);
                if (fieldValue instanceof Property) {
                    propertyList.add((Property) fieldValue);
                }
            }
        }
        //列表转成数组,并通过序号进行排序
        Property[] properties = new Property[propertyList.size()];
        for (Property property : propertyList) {
            if (properties[property.ordinal] != null) {
                throw new DaoException("Duplicate property ordinals");
            }
            properties[property.ordinal] = property;
        }
        return properties;
    }

初始化缓存的策略

public enum IdentityScopeType {
    Session, None
}

数据库缓存的种类有两种。Session,就是使用内存缓存,None就是不使用内存缓存。

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);
        }
    }

可以看到,当我们创建Dao之前会调用这个方法,就说明,如果type==IdentityScopeType.None没有内存缓存,反之会根据主键的类型,赋予内存缓存的管理对象。

还有其他功能就是对TableStatements初始化,判断主键是否是数字类型,分拣主键的数据和非主键的数组,记录所有属性行等大家可以看源码,都比较简单。

Property、TableStatements是两个非常重要的对象,留待后面我们再讲。

AbstractDaoSession

    AbstractDaoSession维护是的Dao.calss对应AbstractDao的关系,AbstractDao是所有实体Dao对象的父类,AbstractDaoSession里面就是实现一个从映射表中取出AbstractDao对象来进行增删改查的操作,也提供了runInTx(Runnable runnable)在事务中做任务不需要返回结果,和callInTx(Callable<V> callable)在事务中做任务并且等待结果等一些方法。

注:一些关于Rx和异步的操作留待后面统一再说。

DaoSession

DaoSession是一个Dao会话,首先我们看构造方法中的初始化

    
  public DaoSession(Database db, IdentityScopeType type, Map<Class<? extends AbstractDao<?, ?>>, DaoConfig>
            daoConfigMap) {
        super(db);

        //根据Class<?>取出配置信息
        noteDaoConfig = daoConfigMap.get(NoteDao.class).clone();
        //根据Session的缓存类型初始化定制的缓存范围
        noteDaoConfig.initIdentityScope(type);
        //根据配置初始化得到一个NoteDao实体
        noteDao = new NoteDao(noteDaoConfig, this);
        //注册到映射表中
        registerDao(Note.class, noteDao);
    }
    

由注释可知,DaoSession会创建我们需要的Dao对象并注册到映射表中管理起来。

每个Dao还会生成这样的格式的语句,所以我们可以根据Session对象直接获取到XXDao:

 public XXDao getxxDao() {
        return xxDao;
    }

还有一个清除所有内存缓存的方法:

 public void clear() {
        noteDaoConfig.clearIdentityScope();
    }

猜你喜欢

转载自blog.csdn.net/u010782846/article/details/80043762
今日推荐