OQL如何支持CASE WHEN?

今天,一个朋友问我,OQL可否支持CASE WHEN语句?他给的示例SQL如下:

  1. select HName,case when IsEnable=1 then '启用' else '停用' from tb_User

OQL是SOD框架ORM查询语言,它类似Linq,但是它诞生的历史比Linq早,并且更加接近SQL语法。所以,对SOD框架而言,对应ORM如何支持CASE WHEN,就等于是问OQL如何支持CASE WHEN了。

这个问题已经不止这一个朋友来问我了,我想了下,还是把这个问题的解决写一篇博客,给大家一种新的解决方案。
注意“新的方案”这个提法,我是不打算让OQL支持CASE WHEN这个特性的,为何要这样做呢?

  1. OQL只解决 80%的普通查询,其它复杂的查询,应该依托于其它技术,否则会增加OQL的复杂性;
  2. 复杂的查询,可以借助于SOD框架的SQL-MAP技术,将SQL语句写在SqlMap.config文件中;
  3. 可以采用数据库视图或者表的计算列,但有些数据库可能不支持计算列;

前面3种原因,第2,3条方法也可以看做是此问题的解决方案,但是它们都需要增加更多的工作量,如果OQL能够直接支持还是更方便些,所以,我今天在这里给大家第4种解决方案:

实体类的计算属性

废话不多说,先直接看代码:

  1. public class User:EntityBase
  2. {
  3. public User()
  4. {
  5. TableName="tb_User";
  6. }
  7.  
  8. public string HName{
  9. get{return getProperty<string>("HName");}
  10. set{setProperty<string>("HName",value,);}
  11. }
  12.  
  13. public bool IsEnable{
  14. get{return getProperty<bool>("IsEnable");}
  15. set{setProperty<bool>("IsEnable",value);}
  16. }
  17.  
  18. public string IsEnableDescrition
  19. {
  20. get{
  21. return IsEnable?"启用":"停用"
  22. }
  23. }
  24.  
  25. }

在这里,HName,IsEnable 这样的属性,SOD框架称为“持久化属性”,因为它MAP了Tb_User表的字段IsEnable ,该字段称为“持久化属性字段”。

持久化属性的get,set方法采用了SOD实体类特殊的方法 setProperty,getProperty 构造,所以属性IsEnableDescription 它不是持久化属性,但是它利用了IsEnable这个持久化属性,对持久化属性进行“计算”,因此,我们这里称呼这样的属性为实体类的“计算属性”。

大家看看,这个“计算属性”是不是很好的起到了 SQL的CASE WHEN效果?

只要忘记了数据库,不要遇到问题就去想如何用SQL语句解决,是不是思路豁然开朗?

使用“计算属性”来支持CASE WHEN效果

前面说过,实体类的“计算属性”本质上不是一个“持久化属性”,它是对持久化属性的计算处理,原理上非常类似SQLServer表上面的计算列。
因此,在SOD框架上使用“计算属性”,有一个必须注意的原则:“计算属性”不可以出现在OQL语句中。

具体举例来说,应该像下面的样子来使用包含计算列的实体类:

  1. User user=new User(){ HName="张三"};
  2. var q=OQL.From(user)
  3. .Select(user.HName,user.IsEnable)
  4. .Where(user.HName)
  5. .END;
  6.  
  7. User user2= EntityQuery<User>.QueryObject(q);
  8. string isEnableDesction =user2.isEnableDesction;

使用“ViewModel”来支持CASE WHEN效果

如果再仔细看看开篇的这个SQL语句,我们发现这种写法常常跟我们的界面查询有关,也就是这个查询要将原来的结果进行一下加工,以方便界面元素使用。对应这种需求,我们通常想到的办法是写一个ViewModel来定制这个查询结果。实际上,前面那个SOD实体类就是一种ViewModel,但它是基于实体类上的ViewModel,之外,SOD也支持直接将查询结果映射到ViewModel。

因此,前面的实体类需要改写成下面这个样子:

  1. public class User:EntityBase
  2. {
  3. public User()
  4. {
  5. TableName="tb_User";
  6. }
  7.  
  8. public string HName{
  9. get{return getProperty<string>("HName");}
  10. set{setProperty<string>("HName",value,);}
  11. }
  12.  
  13. public bool IsEnable{
  14. get{return getProperty<bool>("IsEnable");}
  15. set{setProperty<bool>("IsEnable",value);}
  16. }
  17.  
  18. }

这里去掉了前面的“计算列属性”,它将用一个匿名类型的属性来代替:

  1. User user=new User(){ HName="张三"};
  2. var q=OQL.From(user)
  3. .Select()
  4. .Where(user.HName)
  5. .END;
  6.  
  7. PWMIS.DataProvider.Data.AdoHelper db = PWMIS.DataProvider.Adapter.MyDB.GetDBHelper();
  8. EntityContainer ec = new EntityContainer(q, db);
  9. var list = ec.MapToList()(()=> return new
  10. {
  11. HName = user.HName,
  12. IsEnableDescription =user.IsEnable?"启用":"停用"
  13. });

这里的匿名类型中包含了 IsEnableDescription 一个属性,同时我们的OQL查询也不再需要在Select里面指定查询的属性,而是推迟到MapToList方法里面。

上面这种查询方式,同样支持了我们标题的需求,可见,SOD对一个查询问题,往往能够提供多种解决方案,“条条道路通罗马”,这正是SOD的设计谏言。

---------------分界线--------------------------------------------------------------------------------

有关上面SOD框架查询使用的入门介绍,大家可以参考《.NET ORM 的 “SOD蜜”--零基础入门篇 》

感谢大家对SOD框架一直以来的支持,更多信息,请参考 PDF.NET SOD 开源框架红包派送活动 && 新手快速入门指引

开源项目需要大家更多的支持,SOD开源项目网站:http://pwmis.codeplex.com

如何用ORM支持SQL语句的CASE WHEN?的更多相关文章

  1. SQL语句中case函数

    case函数,严格的意义上来讲case函数已经试流程控制语句了,不是简单意义上的函数,不过为了方便,很多人将case函数称为流程控制函数. case函数的一般有两种用法:1.case expressi ...

  2. sql 语句之 case

    case语句语法: --简单Case函数 CASE sex WHEN '1' THEN '男' WHEN '2' THEN '女' ELSE '其他' END --Case搜索函数 CASE WHEN ...

  3. sql语句判断 case when用法

    sql语句判断方法之一 selectcase when t.bk1='on' then 1else 0 end  as 基础 ,case when t.bk2='on' then 1else 0 en ...

  4. 【Swift 4.0】扩展 WCDB 支持 SQL 语句

    前言 入坑 wcdb 有两个月了,整体来说还是很不错的,具体优点可以参考文档说明,由于官方明确说明不支持 SQL 只好自己写一个扩展支持一下了

  5. Oracle的sql语句中case关键字的用法 & 单双引号的使用

    关于sql中单引号和双引号的使用,来一点说明: 1. 查询列的别名如果含有汉字或者特殊字符(如以'_'开头),需要用双引号引起来.而且只能用双引号,单引号是不可以的. 2. 如果想让某列返回固定的值, ...

  6. 让EF支持sql语句

    BaseDal类: public class BaseDal : IDisposable { #region 事件 internal Func<bool> DBCommitting; in ...

  7. sql语句:CASE WHEN END 的用法

    select b,c, CASE a ' ' ' end from test1

  8. 条件分支SQL语句<一> Case When

    SELECT END ) AS MoneyIn, END ) AS MoneyOut, END ) AS BetMoney, END ) AS PctMoney, END ) AS WinMoney, ...

  9. SQL语句中case,when,then的用法

    用法如下bai: 复制代码 SELECT s.s_id, s.s_name, s.s_sex, CASE WHENs.s_sex='1'THEN'男' WHENs.s_sex='2'THEN'女' E ...

随机推荐

  1. windows命令——explorer

    转至http://www.cnblogs.com/ymind/archive/2012/03/30/explorer-command-args.html 今天才知道,explorer原来可以这样用, ...

  2. 我也来说说DDD~大话目录

    回到占占推荐博客索引 DDD之前没有接触过,但一但有了接触就一发不可收拾,他会带去进入一个全新的世界! DDD不是新技术,而是新思想,新模式,是软件开发领域的一次突破,它更接近于业务,对于业务的改动它 ...

  3. Cookie与Session

    再说Cookie与Session之前,先要了解一下http协议. 何为http协议: http协议即超文本传输协议,一种基于浏览器请求与服务器响应的协议,该协议主要的特点就是它是一种无状态的协议(只针 ...

  4. MySQL学习笔记十二:数据备份与恢复

    数据备份 1.物理备份与逻辑备份 物理备份 物理备份就是将数据库的数据文件,配置文件,日志文件等复制一份到其他路径上,这种备份速度一般较快,因为只有I/O操作.进行物理备份时,一般都需要关闭mysql ...

  5. 关于Checbox的操作,已选,未选,判断

    checkbox: $("#chk1").attr("checked",'');//不打勾                 $("#chk2" ...

  6. iOS开发之窥探UICollectionViewController(二) --详解CollectionView各种回调

    UICollectionView的布局是可以自己定义的,在这篇博客中先在上篇博客的基础上进行扩充,我们先使用UICollectionViewFlowLayout,然后好好的介绍一下UICollecti ...

  7. 恢复MySQL主从数据一致性的总结

    今日上午,同事告知,MySQL主从数据库的数据不一致,猜测备库在同步过程中出现了问题,于是,登上备库,使用 mysql> show slave status\G查看,果然,备库在insert语句 ...

  8. Using Headless Mode in the Java SE Platform--转

    原文地址: By Artem Ananiev and Alla Redko, June 2006     Articles Index This article explains how to use ...

  9. C++指针和动态内存分配

    指针和动态内存分配 数组与指针 数组 数组名是一个指针常量. 数组名传递数据时,传递的是地址. 数组作为函数参数时不指定第一维大小. 对象数组 A a[2] = {A(1,2)}; 执行时先调用有参数 ...

  10. js实现可拖拽的div

    前言 下午忙里偷闲想写一个可拖拽的例子,留在脑海里一直都是三个事件mouseDown,mouseUp,mouseMove, 但从没有动手实践过,今天想起了自己实践了并学习了张鑫旭的demo实现. 学习 ...