hisql ORM 查询语句使用教程
HiSql 提供一个可以适合多种数据库的中间查询语法,不需要关注各个数据库的语法特性,通过HiSql写语句可以在常用的不同类型数据库中执行,且语法与Sql语境类似一看就懂一学就会
HiSql 源码(github) https://github.com/tansar/HiSql
git clone https://github.com/tansar/HiSql.git
::: tip
HiSql 查询方式提供两种查询方式
- 链式写法
- HiSql 语句 (非原生数据库的语句,但类似于SQL的语法可以跨库执行) 一套中间SQL语句 详请见
HiSql语法文档
HiSql主要的方向是低代码平台(低代码平台的特点是用户可以建表,如果建一张表要对应一个实体是行不通的) 所以暂时不支持Lambda表达式的写法,写也是HiSql与其它ORM框架的区别之一
:::
单表查询
HiSql生成的语句
通过ToSql可以查看当前操作生成的底层Sql语句 注:不同的数据库生成的语句会有差别
::: tip
HiSql底层对于SQL语句进行了AST语法,语义解析能智能识别语法错误
:::
//返回当前连接数据库类型的SQL查询语句
string sql1=sqlClient.Query("Hi_TabModel").Field("*").ToSql();
//返回当前连接数据库类型的SQL查询语句
// 注意这不是原生的sql语句 是HiSql定义的一套跨数据库的中间SQL语句 详请见[HiSql语法] 文档
string sql2=sqlClient.HiSql("select * from Hi_TabModel").ToSql();
在执行ToSql() HiSql会检测查询的表及字段的合法性,只要检测通过了才会编译成对应数据库的Sql查询语句
生成的SQL语句如下(SqlServer为例)
select [Hi_TabModel].[TabName],[Hi_TabModel].[TabReName],[Hi_TabModel].[TabDescript],[Hi_TabModel].[TabStoreType],[Hi_TabModel].[TabType],[Hi_TabModel].[TabCacheType],[Hi_TabModel].[TabStatus],[Hi_TabModel].[IsSys],[Hi_TabModel].[IsEdit],[Hi_TabModel].[IsLog],[Hi_TabModel].[LogTable],[Hi_TabModel].[LogExprireDay],[Hi_TabModel].[CreateTime],[Hi_TabModel].[CreateName],[Hi_TabModel].[ModiTime],[Hi_TabModel].[ModiName] from [Hi_TabModel] as [Hi_TabModel]
可能有人会问为什么*号会带出来所有的字段是不是生成的语句有问题?
::: tip
当字段使用*号时会查询表中所有的字段,如果字段的顺序要调整请修改系统表Hi_FieldModel中的SortNum的值,值越小排在越前面(当物理表中的字段顺序不是想要的但又不想修改物理表可以通过该方式调整顺序)
为了可以自定义排序所以会带出字段,不过没有关系这里不影响性能
:::
单表查询结果
hisql查询的查询语法与sql类似
提供了四种查询结果的返回类型
ToTable()返回一个DataTable表结果ToJson()返回一个Json结果集ToList<T>返回一个实体类集合 T 的类的属性必须与返回的结果集有字段相匹配ToDynamic()返回一个动态类集 HiSql.TDynamic 可以查看文档ToEObjectAsync()返回ExpandObject类集 (1.0.2.9及以上版本才支持)
//查询表Hi_TabModel的所有字段 并返回一个 DataTable的结果集
DataTable dt1= sqlClient.Query("Hi_TabModel").Field("*").ToTable();
//查询表Hi_TabModel的所有字段 并返回一个 DataTable的结果集
// 注意这不是原生的sql语句 是HiSql定义的一套跨数据库的中间SQL语句 详请见[HiSql语法] 文档
DataTable dt2= sqlClient.HiSql("select * from Hi_TabModel").ToTable();
查询指定字段
Field()方法支持多参数(params string[] )的字段 注意不要带数据库底层的符号如sqlserver [字段名] ,hisql在生成sql时将会自动处理
//仅查询表Hi_TabModel的部分字段 并返回一个 DataTable的结果集
DataTable dt= sqlClient.Query("Hi_TabModel").Field("TabName","TabReName").ToTable();
//仅查询表Hi_TabModel的部分字段 并返回一个 DataTable的结果集
// 注意这不是原生的sql语句 是HiSql定义的一套跨数据库的中间SQL语句 详请见[HiSql语法] 文档
DataTable dt= sqlClient.HiSql("select TabName,TabReName from Hi_TabModel").ToTable();
生成的SQL语句如下(SqlServer为例)
select [Hi_TabModel].[TabName],[Hi_TabModel].[TabReName] from [Hi_TabModel] as [Hi_TabModel]
单表分页查询
Skip表示取的页码数Take表示每页获取数量Sort可以支持多种方式如Sort("createtime desc")注意这个适用于Hisql支持的所有库(不要用底层库的语法不然Hisql检测时会抛出异常)Sort("createtime desc","TabName asc")多个排序条件时可以这样写Sort(new SortBy { { "createtime",SortType.DESC} })也支持这种写法Sort(new SortBy { { "createtime",SortType.DESC} , { "TabName", SortType.ASC } })多个参数写法
//方式1
DataTable dt1= sqlClient.Query("Hi_TabModel").Field("*")
.Sort("createtime desc"). Skip(1).Take(100).ToTable();
//方式2
DataTable dt2 = sqlClient.Query("Hi_TabModel").Field("*")
.Sort(new SortBy { { "createtime",SortType.DESC} }).Skip(1).Take(100).ToTable();
//如果需要返回当前查询条件的记录总数
int _total=0;
DataTable dt3 = sqlClient.Query("Hi_TabModel").Field("*")
.Sort(new SortBy { { "createtime",SortType.DESC} }).Skip(1).Take(100).ToTable(ref _total);
DataTable dt4= sqlClient.HiSql("select * from Hi_TabModel").Skip(1).Take(100).ToTable();
//如果需要返回当前查询条件的记录总数
int _total=0;
DataTable dt5 = sqlClient.HiSql("select * from Hi_TabModel").Skip(1).Take(100).ToTable(ref _total);
::: tip
在HiSql底层会根据页码生在不同的分页方式以追求性能极致
:::
Where条件查询
HiSql提供了多种查询语句的写法总有一款适合你
HiSql查询条件的特点是可以动态拼接(非生成数据库SQL),这是其它ORM框架完全不具备的用Lambda表达式的方式如果要拼动态条件试过的人都知道多痛苦,如果用原生SQL是无法跨库支持的且完全性不可控(sql注入),HiSql在底层编译器解析成不同的数据库语句。有兴趣的可以试一下ToSql()方法查看一下编译出来的SQL
可能有些人会问?用表达式这种强类型的不好吗? 当然好还是那句话如果你的项目不涉及到动态的或与低代码相关的建议你用传统的ORM工具,如果你的项目想做成高度配置型的HiSql将是不错的选择
//单值
DataTable dt1 = sqlClient.Query("Hi_FieldModel").Field("*").Where(new Filter { { "TabName", OperType.EQ, "HTest01" } }).ToTable();
//多值查询 或查询
DataTable dt2 = sqlClient.Query("Hi_FieldModel").Field("*").Where(new Filter {
{ "TabName", OperType.EQ, "HTest01" },
{ LogiType.OR},
{ "FieldName", OperType.EQ, "UName" }
}).ToTable();
DataTable dt3 = sqlClient.Query("Hi_FieldModel").Field("*").Where(new Filter { { "TabName", OperType.EQ, "HTest01" } }).ToTable();
//注意这不是原生的sql语句 是HiSql定义的一套跨数据库的中间SQL语句 详请见[HiSql语法] 文档
//hisql语法
DataTable dt4 = sqlClient.HiSql("select * from Hi_FieldModel as a where a.TabName='HTest01'").ToTable();
DataTable dt5 = sqlClient.HiSql("select * from Hi_FieldModel where TabName='HTest01'").ToTable();
DataTable dt6 = sqlClient.HiSql("select * from Hi_FieldModel where (TabName='HTest01' or FieldName='UName')").ToTable();
DataTable dt7 = sqlClient.Query("Hi_FieldModel").Field("*").Where("TabName='HTest01'").ToTable();
DataTable dt8 = sqlClient.Query("Hi_FieldModel").Field("*").Where("(TabName='HTest01' or FieldName='UName')").ToTable();
子查询
上面介绍的查询方式大家可能都看得出来查询是比较简单的,如果涉及到子查询怎么样写呢?下来就来演示一下
DataTable dt1= sqlClient.Query("Hi_FieldModel").Field("*").Where(new Filter { { "TabName", OperType.IN,
sqlClient.Query("Hi_TabModel").Field("TabName").Where(new Filter { {"TabName",OperType.IN,new List<string> { "HTest01", "Hi_TabModel" } } })
} }).ToTable();
//注意这不是原生的sql语句 是HiSql定义的一套跨数据库的中间SQL语句 详请见[HiSql语法] 文档
DataTable dt2 = sqlClient.HiSql("select * from Hi_FieldModel where TabName in (select TabName from Hi_TabModel where TabName in ('HTest01','Hi_TabModel'))").ToTable();
生成的SQL语句如下(SqlServer为例)
select [Hi_FieldModel].[TabName],[Hi_FieldModel].[FieldName],[Hi_FieldModel].[FieldDesc],[Hi_FieldModel].[IsIdentity],[Hi_FieldModel].[IsPrimary],[Hi_FieldModel].[IsBllKey],[Hi_FieldModel].[FieldType],[Hi_FieldModel].[SortNum],[Hi_FieldModel].[Regex],[Hi_FieldModel].[DBDefault],[Hi_FieldModel].[DefaultValue],[Hi_FieldModel].[FieldLen],[Hi_FieldModel].[FieldDec],[Hi_FieldModel].[SNO],[Hi_FieldModel].[SNO_NUM],[Hi_FieldModel].[IsSys],[Hi_FieldModel].[IsNull],[Hi_FieldModel].[IsRequire],[Hi_FieldModel].[IsIgnore],[Hi_FieldModel].[IsObsolete],[Hi_FieldModel].[IsShow],[Hi_FieldModel].[IsSearch],[Hi_FieldModel].[SrchMode],[Hi_FieldModel].[IsRefTab],[Hi_FieldModel].[RefTab],[Hi_FieldModel].[RefField],[Hi_FieldModel].[RefFields],[Hi_FieldModel].[RefFieldDesc],[Hi_FieldModel].[RefWhere],[Hi_FieldModel].[CreateTime],[Hi_FieldModel].[CreateName],[Hi_FieldModel].[ModiTime],[Hi_FieldModel].[ModiName] from [Hi_FieldModel] as [Hi_FieldModel]
where [Hi_FieldModel].[TabName] in (select [Hi_TabModel].[TabName] from [Hi_TabModel] as [Hi_TabModel]
where [Hi_TabModel].[TabName] in ('HTest01','Hi_TabModel')
)
虽然两种不同的写法,但是生成的语句是一模一样的,HiSql内置了HiSql语句编译器有兴趣的可以看一下源码,实现原理与其它ORM框架有着本质匹别解析性能比Lambda表达式方式要强
分类汇总查询
HiSql支持max,min,avg,sum,count 五种常用聚合函数
DataTable dt = sqlClient.Query("Hi_FieldModel").Field("FieldName", "count(FieldName) as CHARG_COUNT").Group(new GroupBy { { "FieldName"} }).ToTable();
DataTable dt2 = sqlClient.HiSql("select FieldName,count(FieldName) as CHARG_COUNT from Hi_FieldModel group by FieldName").ToTable();
生成的sql如下(以sqlserver为例)
select [Hi_FieldModel].[FieldName],count(*) as CHARG_COUNT from [Hi_FieldModel] as [Hi_FieldModel]
group by [Hi_FieldModel].[FieldName]
对分类汇总不跳号排名
相有多个值同时则并列排名
DataTable dt_dens= sqlClient.Query("Hi_FieldModel").Field("FieldName", "count(FieldName) as CHARG_COUNT")
.Group(new GroupBy { { "FieldName" } })
.WithRank(DbRank.DENSERANK, DbFunction.COUNT, "FieldName", "rowidx1", SortType.DESC).ToTable();
生成的sql如下 (sqlserver)
select [Hi_FieldModel].[FieldName],count(*) as CHARG_COUNT,
dense_rank() over( order by COUNT([FieldName]) DESC) as rowidx1
from [Hi_FieldModel] as [Hi_FieldModel]
group by [Hi_FieldModel].[FieldName]
查询结果如下
并列第一的都是第1名 但第二名这个号没有被占用,这种叫不跳号排名

分类汇总跳号排名
DataTable dt_rank = sqlClient.Query("Hi_FieldModel").Field("FieldName", "count(FieldName) as CHARG_COUNT")
.Group(new GroupBy { { "FieldName" } })
.WithRank(DbRank.RANK, DbFunction.COUNT, "FieldName", "rowidx2", SortType.DESC).ToTable();
生成的sql如下(sqlserver)
select [Hi_FieldModel].[FieldName],count(*) as CHARG_COUNT,
rank() over( order by COUNT([FieldName]) DESC) as rowidx2
from [Hi_FieldModel] as [Hi_FieldModel]
group by [Hi_FieldModel].[FieldName]
查询结果如下
当有几个并列第一时则跳过这此号 称为跳号排名

分类汇总结果增加行号
var dt1 = sqlClient.Query(
sqlClient.Query("Hi_FieldModel").Field("*").WithLock(LockMode.ROWLOCK).Where(new Filter { { "TabName", OperType.IN,
sqlClient.Query("Hi_TabModel").Field("TabName").Where(new Filter { {"TabName",OperType.IN,new List<string> { "Hone_Test", "H_TEST" } } })
} }),
sqlClient.Query("Hi_FieldModel").WithLock(LockMode.ROWLOCK).Field("*").Where(new Filter { { "TabName", OperType.EQ, "DataDomain" } }),
sqlClient.Query("Hi_FieldModel").Field("*").Where(new Filter { { "TabName", OperType.EQ, "Hi_FieldModel" } })
)
.Field("TabName", "count(*) as CHARG_COUNT")
.WithRank(DbRank.DENSERANK, DbFunction.NONE, "TabName", "rowidx1", SortType.ASC)
//.WithRank(DbRank.ROWNUMBER, DbFunction.COUNT, "*", "rowidx2", SortType.ASC)
//以下实现组合排名
.WithRank(DbRank.ROWNUMBER, new Ranks { { DbFunction.COUNT, "*" }, { DbFunction.COUNT, "*", SortType.DESC } }, "rowidx2")
.WithRank(DbRank.RANK, DbFunction.COUNT, "*", "rowidx3", SortType.ASC)
.Group(new GroupBy { { "TabName" } }).ToTable();
多表查询
join 查询
大家可以对比与其它框架的写法,HiSql的语法更贴近于数据库的SQL语句,学习成本更低,且其它ORM框架由于受泛型类的限制,对Join表的数量是有限制的。
DataTable dt1 = sqlClient.Query("Hi_FieldModel", "A").Field("A.FieldName as Fname")
.Join("Hi_TabModel").As("B").On(new JoinOn { { "A.TabName", "B.TabName" } })
.Where(new Filter { { "A.TabName",OperType.EQ, "HTest01" } })
.ToTable();
DataTable dt2 = sqlClient.HiSql("select A.FieldName as Fname from Hi_FieldModel as A inner join Hi_TabModel as B on A.TabName = B.TabName where A.TabName='HTest01'").ToTable();
生成的sql如下 (sqlserver)
select [A].[FieldName] as [Fname] from [Hi_FieldModel] as [A]
inner join [Hi_TabModel] as [B] on [A].[TabName]=[B].[TabName]
where [A].[TabName] = 'HTest01'
hisql ORM 查询语句使用教程的更多相关文章
- 用lambda构建ORM查询语句
本文介绍如何解析lambda表达式来获取一个满足条件的查询语句. 先看个截图 通过设置实体对象Article_Content的查询表达式,就可以获取对应的参数化SQL语句,使用起来很方便,减少了代码 ...
- Java开源协同办公项目:数据中心,自定义查询语句使用教程
O2OA提供的数据管理中心,可以让用户通过配置的形式完成对数据的汇总,统计和数据分组展现,查询和搜索数据形成列表数据展现.也支持用户配置独立的数据表来适应特殊的业务的数据存储需求.本文主要介绍如何在O ...
- hisql orm update表数据更新文档
更新 HiSql数据更新 HiSql 提供了好几种数据更新的方式下面一一介绍一下 如果你的表中增加了这四个字段 字段 描述 类型 CreateTime 创建时间 DateTime CreateName ...
- ORM查询条件
模板: from django.db import models class Article(models.Model): title = models.CharField(max_length=20 ...
- Django(17)orm查询操作
前言 查找是数据库操作中一个非常重要的技术.查询一般就是使用filter.exclude以及get三个方法来实现.我们可以在调用这些方法的时候传递不同的参数来实现查询需求.在ORM层面,这些查询条件都 ...
- hisql orm 框架insert数据写入教程
hisql.net 官网(文档编写中) HiSql 源码(github) https://github.com/tansar/HiSql git clone https://github.com/ta ...
- 第二十篇ORM查询与SQL语句
ORM查询与SQL语句 多表操作 创建模型 实例:我们来假定下面这些概念,字段和关系 作者模型:一个作者有姓名和年龄. 作者详细模型:把作者的详情放到详情表,包含生日,手机号,家庭住址等信息.作者详情 ...
- SQL Server-简单查询语句,疑惑篇(三)
前言 对于一些原理性文章园中已有大量的文章尤其是关于索引这一块,我也是花费大量时间去学习,对于了解索引原理对于后续理解查询计划和性能调优有很大的帮助,而我们只是一些内容进行概括和总结,这一节我们开始正 ...
- Django 源码小剖: Django ORM 查询管理器
ORM 查询管理器 对于 ORM 定义: 对象关系映射, Object Relational Mapping, ORM, 是一种程序设计技术,用于实现面向对象编程语言里不同类型系统的数据之间的转换.从 ...
随机推荐
- ZooKeeper 06 - ZooKeeper 的常用命令
目录 1 - 服务端常用命令 2 - 客户端常用命令 3 - 常用四字命令 4 - ZooKeeper 日志的可视化 版权声明 若要部署 ZooKeeper 单机环境,请查看此篇:https://ww ...
- 沉淀vue相关知识(主要还是个人积累用)
路由懒加载的配置: const Home= () =>import('../components/Home') //使用ES6中的路由懒加载的方式 const About= () =>im ...
- 【Fastjson】Fastjson反序列化由浅入深
Fastjson真-简-介 fastjson是由alibaba开发并维护的一个json工具,以其特有的算法,号称最快的json库 fastjson的使用 首先先创一个简单的测试类User public ...
- Linux目录终章,单用户模式修改密码、环境变量、第三方软件安装
目录 今日内容概要 内容详细 解析映射文件 磁盘挂载文件 开机加载脚本 系统启动级别 使用单用户模式修改密码 变量加载文件 登录提示信息 第三方软件安装目录(编译安装目录) 系统日志目录 保存系统运行 ...
- Nginx日志分析脚本
目录 一.简介 二.脚本 一.简介 运维工作是一个比较复杂的工作,有时候面对上万条的日志,如何作分析?难道一条条的分析? 这估计看两眼就要打哈欠了吧?聪明的人会选择脚本,这就是为什么现在提倡自动化运维 ...
- Android: Client-Server communication by JSON
Refer to: http://osamashabrez.com/client-server-communication-android-json/ This is a sequel to my l ...
- Linux 01 计算机硬件之冯诺依曼体系
1. 计算机硬件软件体系 1.1 冯诺依曼体系结构 (1) 计算机处理的数据和指令用二进制表示 (2) 按顺序执行指令 (3) 计算机硬件:运算器.控制器.储存器.输入设备和输出设备. 1.2 计算机 ...
- AJAX get和post请求
<!DOCTYPE html><html><head> <meta charset="UTF-8"> <title>&l ...
- JAVA获取指定日期是星期几
/** * 获取指定日期是星期几<br> * * @param date * @return 指定日期是星期几 */ public static String getWeekOfDate( ...
- Mysq索引优化(什么情况创建索引,什么情况不创建索引)
一.以下情况需要创建索引 1.主键自动建立唯一索引 2.频繁作为查询条件的字段应该创建索引 3.查询中与其他表关联的字段,外键关系建立索引 4.单键/组合索引的选择问题,组合索引性价比更高 5.查询中 ...