高效的Android ORM框架

作者: 骆驼骑士 | 来源:发表于2016-04-16 15:33 被阅读1447次

简要说明

YoDao是一个简单又高效的Android ORM开源框架。

项目主页:https://github.com/sandin/YoDao

项目作者:sandin

1.实现目标

项目受Hibernate等ORM框架启发,并实现了Java Persistence API标准,在大部分系统设计上学习了Spring Data JPA框架,所以在使用和其API上十分相似。

本项目和OrmLite等传统ORM框架不同的是,为了提高性能所以没有使用运行时的反射来解析注解,而使用daggerbutterknife一样的技术在编译时解析注解并自动生成代码,因此在使用ORM来提高开发效率的同时也没有牺牲运行时的性能和速度。

因此框架的目标是实现一个快速而轻量的ORM框架,即可以提高开发效率,但又不牺牲运行时的性能。

2.框架结构

整个框架和butterknife一样,分为三个子项目:

  1. 主项目,这是一个Android library项目,在使用的时候引用即可。
  2. 编译项目,这是一个编译工具,在IDE或gradle里配置即可,无需打包到APK里。
  3. 实例项目,DEMO演示。

3.加入项目

此项目为开源项目,因此欢迎任何有着开源精神的开发者参与进来,有意者可以联系作者,或直接在github上fork该项目。

使用说明

基本使用是和dagger一样,这里仅描述android studio的使用方式:

1. 修改gradle配置

build.gradle 文件加入一下配置:

apply plugin: 'com.neenbedankt.android-apt'

// ...

buildscript {
    dependencies {
        classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
    }
}

repositories {
    jcenter()
}

dependencies {
    compile 'com.lds:yodao:0.2.1'
    apt 'com.lds:yodao-compiler:0.2.1'
}

注意:目前仅上传到jcenter,还没有同步到Maven Center。

2. JavaBean增加注解

import com.yoda.yodao.annotation.Column;
import com.yoda.yodao.annotation.GeneratedValue;
import com.yoda.yodao.annotation.GenerationType;
import com.yoda.yodao.annotation.Id;

@Entity
@Table(name="user")
public class User implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;

    @Column
    private String name;

    @Column(name = "age")
    private int age;

    public User() {
    }

    // Getter & Setter
    // ...
}

这里使用的注解和JPA一模一样,只是因为android里面没有 javax.persistence包,所以所有的注解都放置在 com.yoda.yodao.annotation 包下,但注解类和使用方法是一样的,可以说完全兼容JPA标准的实体类。

这里注意Getter&Setter方法的命令必须完全遵循POJO的命名规范,否则会编译不过。并建议使用驼峰命名。

3. DAO的定义

这里设计和Spring Data JPA框架一样,只要定义个 interface ,并继承 YoDao 即完成了Dao的编码工作,框架会根据这个接口的定义自动生成其实现类的代码。

例如:

public interface UserDao extends YoDao<User, Long> {

    User findOneByName(String name);

    List<User> findListByNameAndAge(String name, int age);

    List<User> findListOrderByAge();

    List<User> findListOrderByAgeDesc();

    long countByAge(int age);

}

YoDao<T, ID> 的两个泛型:

  1. T,映射的实体类
  2. ID,主键PK的类

4. 在数据库里建表

框架已经将所有实体对应的table的建表SQL都生成了,只需要在建表的时候调用:

public class MySQLiteOpenHelper 
            extends android.database.sqlite.SQLiteOpenHelper {

    @Override
    public void onCreate(SQLiteDatabase db) {
        DaoFactory.create(HairDao.class, null).onCreateTable(db);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        DaoFactory.create(HairDao.class, null).onUpgradeTable(db, oldVersion, newVersion);
        onCreate(db);
    }

}

在修改实体类以后(如增加字段),只需要重新build一下,重建表的时候数据结构就会变化。

5. 使用你的Dao

UserDao userDao = DaoFactory.create(UserDao.class, mSQLiteOpenHelper);

// insert
User user = new User();
user.setName("Jack");
user.setAge(23);
userDao.save(user);

// update
user.setName("New Name");
userDao.update(user);

// query
List<User> users = userDao.findAll();
user = userDao.findOne(user.getId());

// delete
userDao.delete(user); 

注解说明

@Entity

使用在需要映射的实体类上,框架只会处理有此注解的实体类。

@Table(name='table_ame')

使用在需要映射的实体类上,属性name为表名,不提供name的时候将自动使用类名作为表名。(表名的命名规范是全部小写,下划线分割:如class名为 UserInfo 表名自动是 user_info)

@Id

使用在需要映射成表主键的成员变量上,主键支持 int long String 类型。

@GeneratedValue

使用在需要映射成主键的成员变量上,

属性strategy为主键的生成策略:

  1. AUTO 主键自增
  2. UUID 主键使用UUID随机生成

@Column(name="column_ame")

使用在需要映射的成员变量上

属性name为表列名,不提供name的时候将默认使用变量名作为列名(一样会转成全部小写,下划线分割的命名规范)

属性unique对于SQL里面的UNIQUE

属性nullable对应SQL里面的NOT NULL

属性length对于SQL里面的长度限制

其他属性请参考JPA。

@NotColumn

使用在不需要映射的成员变量上,因为框架会将需要实体类的所有成员变量都进行映射,使用了该注解告诉框架这个字段不需要在表中持久化,则在建表时忽略该字段。

@Repository

使用在自己编写的DAO上面,表示该DAO是需要框架来生成实现类的代码。

@Query

使用在DAO的方法上面,表示这个方法不需要自动生成,自定义SQL查询。

属性value为自定义查询的SQL语句。

目前仅支持原生SQL,今后将支持在HSQL语法。

其他注解

今后将陆续开始支持 OneToOne OneToMany ManyToOne ManyToMany 等JPA的注解。

YoDao增删改查接口

save()

/**
 * Saves a given entity. Use the returned instance for further operations as
 * the save operation might have changed the entity instance completely.
 *
 * @param entity
 * @return the saved entity
 */
boolean save(T entity);

/**
 * Saves all given entities.
 *
 * @param entities
 * @return
 */
boolean save(List<T> entities);

update()

/**
    * update a row by primary key
    * 
    * @param entity
    * @return
    */
int update(T entity);

/**
 * update by fields
 * 
 * @param entity
 * @param whereClause
 * @param whereArgs
 * @return
 */
int updateByFields(T entity, String whereClause,
           String[] whereArgs);

/**
 * update some fields
 * 
 * @param values
 * @param whereClause
 * @param whereArgs
 * @return
 */
int updateByFields(ContentValues values, String whereClause,
           String[] whereArgs);

find()

/**
 * Retrives an entity by its primary key.
 *
 * @param id
 * @return the entity with the given primary key or {@code null} if none
 *         found
 * @throws IllegalArgumentException
 *             if primaryKey is {@code null}
 */
T findOne(ID id);

/**
 * Retrives an entity by its fields
 * 
 * @param selection
 * @param selectionArgs
 * @param groupBy
 * @param having
 * @param orderBy
 * @return
 */
T findOneByFields(String selection, String[] selectionArgs, String groupBy,
      String having, String orderBy);

/**
 * Retrives an entity by its fields
 * 
 * @param selection
 * @param selectionArgs
 * @param orderBy
 * @return
 */
T findOneByFields(String selection, String[] selectionArgs, String orderBy);

/**
 * Retrives an entity by sql
 * 
 * @param sql
 * @return
 */
T findOneBySql(String sql, String[] selectionArgs);

/**
 * Returns all instances of the type.
 *
 * @return all entities
 */
List<T> findAll();

/**
 * Find List By Fields
 * 
 * @param selection
 * @param selectionArgs
 * @param groupBy
 * @param having
 * @param orderBy
 * @return
 */
List<T> findListByFields(String selection, String[] selectionArgs,
      String groupBy, String having, String orderBy);

/**
 * Find List By Fields
 * 
 * @param selection
 * @param selectionArgs
 * @param orderBy
 * @return
 */
List<T> findListByFields(String selection, String[] selectionArgs,
      String orderBy);

/**
 * Find List By SQL
 * 
 * @param sql
 * @return
 */
List<T> findListBySql(String sql, String[] selectionArgs);

delete()

/**
 * Deletes the entity with the given id.
 * 
 * @param id
 */
int delete(ID id);

/**
 * Deletes a given entity.
 *
 * @param entity
 */
int delete(T entity);

/**
 * Deletes the given entities.
 *
 * @param entities
 */
int delete(List<T> entities);

/**
 * Deletes all entities managed by the repository.
 */
int deleteAll();

/**
 * Deletes entities by fields
 * 
 * @param selection
 * @param selectionArgs
 * @return
 */
int deleteByFields(String selection, String[] selectionArgs);

exists()

/**
 * Returns whether an entity with the given id exists.
 *
 * @param id
 * @return true if an entity with the given id exists, alse otherwise
 * @throws IllegalArgumentException
 *             if primaryKey is {@code null}
 */
boolean exists(ID id);

count()

/**
 * Returns the number of entities available.
 *
 * @return the number of entities
 */
long count();

/**
 * Returns the number of entities with selections available.
 * 
 * @param selections
 * @param selectionArgs
 * @return
 */
long countByFields(String selections, String[] selectionArgs);

onCreateTable()

这个方法在SQLOpenHelper中的onCreate()中调用,调用后才会执行建表语句。

onUpgradeTable()

这个方法在SQLOpenHelper中的onUpgrade()中调用,调用后才会执行删除表语句。

DaoFactory工厂类

自动生成的DAO

实现类

默认情况下,框架会根据你写的dao的包名和类名来生成实现类的包名和类名。

例如,你的Dao为:

com.your.test.dao.UserDao

那么生成的实现类名为:

com.your.test.dao.impl.UserDaoImpl

在android studio里,自动的生成的代码在项目根目录下的build/source/apt/debug目录里 ,这个目录由插件决定,eclipse上可以自行配置。

实例化DAO

框架推荐不直接调用和实例化自动生成的Impl类,而应该使用 DaoFactory 来实例化Dao,例如:

UserDao userDao = DaoFactory.create(UserDao.class, mSQLiteOpenHelper);

今后将支持通过注解来定义生命周期以支持对于指定Dao可选性的单例模式。目前所有Dao都默认非单例。

DAO的写法

自定义Dao

一般情况下,自定义的 dao 只需要写一个空的 interface 然后继承 YoDao接口即够用了,因为接口里实现了大量增删改查的API。

例如:

public interface UserDao extends YoDao<User, Long> {

}

注意:两个泛型不要搞反了,第一个是实体类,第二个是主键。今后框架将会校验,目前还没有。

方法的定义

对于接口无法满足的时候,则可以通过定义接口的方法来实现,这里必须按照指定的约定来写方法名,框架则会按照方法的定义来帮你实现。

如果你写的方法名不被支持,则框架在编译时会提示哪一个方法定义出现了问题,请查询相关文档来修改方法的定义。

以下列出所有支持的方法:

WHERE

User findOneByName(String name);
// SQL: select * from user where name = ? limit 1

// AND
List<User> findListByNameAndAge(String name, int age);
// SQL: select * from user where name = ? and age = ?

// OR
List<User> findListByNameOrAge(String name, int age);
// SQL: select * from user where name ? or age = ?

GROUP BY

List<User> findListByNameGroupByAge(String name);
// SQL: select * from user where name = ? group by age;

// HAVINGS
// 暂不支持

ORDER BY

List<User> findListOrderByAge();
// SQL: select * from user order by age

List<User> findListOrderByAgeDesc();
// SQL: select * from user order by age DESC

UPDATE

int updateByName(User user, String name);
// SQL: update user ... where name = ?

DELETE

int deleteByName(String name);
// SQL: delete from user where name = ?

COUNT

long countByAge(int age);

原生SQL

@Query(value= "select * from user where name = ?")
User findByUsername(String username);

关键要点:

  1. 所有SQL的关键字都有意义,不能随便将其放置在方法名中,如 BY Order Where 等。
  2. 注意字段名的命名规范,这里会将字段名转成和字段名一样的命名规范,即全部小写,下划线分词,例如 findByUserName 那么则直接使用 where user_name = ?的SQL来查找,并且会使用 setUserName() 和 getUserName() 方法来设值和取值。Java的字段名和方法名都保持驼峰命名法,数据库的表名和列名都按全部小写,下划线分割的命名法。

方法的参数

方法的参数的个数和顺序必须和方法定义的查询保持一致。

方法的返回值

方法的返回值类型只能是:

  1. T find语句
  2. List<T> find语句
  3. boolean exists语句
  4. long count语句
  5. int delete语句

相关文章

网友评论

  • 02ff3d94682c:我也正打算写一款基于编译时注解的ORM框架
  • 02ff3d94682c:跟我的想法不谋而合
  • 8d7cc6bc70d2:greendao有用反射吗?
    骆驼骑士:@aduwan greendao确实没使用反射,这点写错了,不好意思,我稍后会改正过来,和greendao的区别主要在于不需要再用另外一个工具来生成代码。

本文标题:高效的Android ORM框架

本文链接:https://www.haomeiwen.com/subject/uwxolttx.html