同时具有Hibernate 优点 & Mybatis优点功能,适用于承认以SQL为中心,同时又需求工具能自动能生成大量常用的SQL的应用。
List<User> list = SqlManager.getSQLScript("selectUser").select(paras,User.class); User user = SqlManager.getSql(User.class,SELECT_BY_ID).unque(id);
selectUser === select * from user where 1=1 @if(user.age==1){ and age = #user.age# @} selectAll === select * from user @use("selectWhere"); selectWhere === where age = #age#
BeetlSQL集中管理SQL语句,SQL 可以按照业务逻辑放到一个文件里,文件可以按照模块逻辑放到一个目录下。文件格式抛弃了XML格式,采用了Markdown,原因是
目前SQL文件格式如下
文件一些说明,放在头部可有可无,如果有说明,可以是任意文字 SQL标示 === SQL语句 SQL标示2 === SQL语句 2
所有SQL文件建议放到一个db目录,db目录有多个子目录,表示数据库类型,比如db下有common,这是公共SQL语句 还有mysql,oralce。当程序获取SQL语句得时候,先会根据数据库找特定数据库下的sql语句,如果未找到,会寻找common下的。如下代码
SqlScript sql = SqlManager.getSql("sys.user.update");
SqlManager 会根据当前使用的数据库,先找db/mysql/sys/user.md 文件,确认是否有update语句,如果没有,则会寻找db/common/sys/user.md
BeetlSql 适合各种类型的引用,对于大中小型应用,模型通常是Pojo,这样易于维护和与三方系统交互,对于特小型项目,往往不需要严格的模型,表示业务实体通常是Map/List 组合。SQL语句的输入可以是Pojo或者Map,SQL语句执行结果也可以映射到Pojo和Map。
SqlScript sql = SqlManager.getSql("user.update"); int result = sql.update(user); SqlScript sql = SqlManager.getSql("user.select"); List<User> list = sql.select(user,User.class); SqlScript sql = SqlManager.getSql("user.select"); Map paras = new HashMap(); paras.put("age",11); User user = sql.single(paras,User.class); //or Map user = sql.single(paras,Map.class);
SQL语句可以动态生成,基于Beetl语言,这是因为
beetl执行效率业界出名的高效 ,因此对于基于模板的动态sql语句,采用beetl非常合适
beetl 语法简单易用,可以通过半猜半式的方式实现,杜绝myBatis这样难懂难记得语法。BeetlSql学习曲线几乎没有
利用beetl可以定制定界符号,完全可以将sql模板定界符好定义为数据库sql注释符号,这样容易在数据库中测试,如下也是sql模板(定义定界符为"--" 和 "null",null是回车意思);
selectByCond === select * form user where 1=1 --if(age!=null) age=#age# --}
beetl 错误提示非常友好,减少写SQL脚本编写维护时间
beetl 能容易与本地类交互(直接访问Java类),能执行一些具体的业务逻辑 ,也可以直接在sql模板中写入模型常量,即使sql重构,也会提前解析报错
beetl语句易于扩展,提供各种函数,比如分表逻辑函数,跨数据库的公共函数等
BeetlSql虽然不是一个O/R Mapping 工具,但能根据默认约定,生成大量常用SQl而几乎不需要注解(对于oralce,需要注解SeqId(name="seqName")),BeetlSql 根据输入的class,自动能生成如下sql
select_by_id : 根据主键查询
select_by_template: 将实例变量作为模板查询,如果其变量的属性为空,则不计入查询条件
update_by_id: 根据主键更新
update_by_template:根据模板更新
delete_by_id: 根据主键查询
delete_by_template:根据模板查询
insert: 自动插入,如果没有主键Annotaion,则会寻找数据库找到主键,并认为是自增主键。总共有如下三种主键Annotation
SeqId 用于oralce AutoId,用于自增。这是主键默认设置 AssignId,代码指定主键
如前所述,BeetlSql 可以通过sql文件的管理和搜索来支持跨数据库开发,如前所述,先搜索特定数据库,然后再查找common。另外BeetlSql也提供了一些夸数据库解决方案
BeetlSql可以在执行sql前后执行一系列的Intercetor,从而有机会执行各种扩展和监控,这比已知的通过数据库连接池做Interceptor更加容易。如下Interceptor都是有可能的
BeetlSql管理数据源,如果只提供一个数据源,则认为读写均操作此数据源,如果提供多个,则默认第一个为写库,其他为读库。用户在开发代码的时候,无需关心操作的是哪个数据库,因为调用sqlScrip 的 select相关api的时候,总是去读取从库,add/update/delete 的时候,总是读取主库。
sqlScript.insert(user) // 操作主库,如果只配置了一个数据源,则无所谓主从 sqlScript.selectById(id,User.class) //读取从库
当然,也可以根据自己具体逻辑来确定如果选择主从库,只需要扩展BeetlSql,这一切对开发者是透明的
开发者也可以通过在Sql 模板里完成分表逻辑而对使用者透明,如下sql语句
insert into #text("log"+date())# values () ... 注:text函数直接输出表达式到sql语句,而不是输出?。
log表示按照一定规则分表,table可以根据输入的时间去确定是哪个表
select * from #text("log"+log.date)# where 注:text函数直接输出表达式到sql语句,而不是输出?。
同样,根据输入条件决定去哪个表,或者查询所有表
@ var tables = getLogTables(); @ for(table in tables){ select * from #text(table)# @ if(!tableLP.isLast) print("union"); @} where name = #name#
BeetlSQL 仍然还在开发过程中,因为业余时间有限,原本计划的1个月开发完毕,已经耽误了数周了,但相信8月底或者9月初就会有一个版本出来,希望beetl粉丝继续支持
原文:http://my.oschina.net/xiandafu/blog/493438