CDS View

接上篇:HANA

文章目录

      • 3. CDS View
        • 3.1 Annotation注解
        • 3.2 Case 语句
        • 3.3 数学计算
        • 3.4 字符串表达
        • 3.5 货币和单位转换
        • 3.6 聚集语句
        • 3.7 Join
        • 3.8 Union
        • 3.9 参数
        • 3.10 作为子CDS View
        • 3.11 CDS View中的association关联
        • 3.12 扩展CDS View
        • 3.13 CDS权限
      • 4. AMDP

3. CDS View

3.1 Annotation注解

注解分为几大类:

  1. 视图注解:
    @AbapCatalog.sqlViewName: ‘ZBW_SQL_TEST’ (这个注解是强制性的,必须得有一个,而且是和CDS View名字不一样的)
    @EndUserText.label: ‘ZBW_CDS_TEST’
    就在define view的开头
    或者说这些注解:
    CDS View_第1张图片
    也就是这些注解去手动设置一些配置项。
    CDS View_第2张图片

  2. 元素注解:
    是view中间的元素
    @Sementics.amount.currencyCode:‘local_currency_key’
    @Semantics.currencyCode:true
    比如:
    你要把金额和单位关联起来,就像下面这样给这两个注解。
    CDS View_第3张图片

  3. 参数注解
    @Environment.systemField:#SYSTEM_DATE

  4. 扩展注解
    @AbapCatalog.sqlViewAppendName:‘Z_CDS_SQL’
    用来扩展CDSView的

  5. 功能注解
    在不同功能前的注解。
    Ctrl+空格键你能看到系统提供好多注解。
    CDS View_第4张图片

3.2 Case 语句

  1. 一般的case语句
    以关键字case开头判断紧跟在后,end 结尾,as别名在CDSview里面加一列。
    不满足判断条件就依据else的判断。
case smoker
  when 'x' then loccuram + 100
  else loccuram
end as inc_smoker_charge_1l

CDS View_第5张图片

多加一列,抽烟加100块钱
CDS View_第6张图片
2. 类似if/else的case语句
这个就很类似if语句,判断条件没有紧跟在后,而是在When里面进行判断。

case 
    when smoker = 'x' and currencytype = 'E' then loccuram + 100
    when smoker = 'x' and currencytype = 'C' then loccuram + 200
    else loccuram
end as inc_smoker_charge_2,
  1. 嵌套case语句
case smoker
  when ' ' then loccuram
  else case 
         when custtype = 'b' then loccuram + 100
         when custtype = 'p' then loccuram + 200
         else loccuram
       end 
   end as inc_smoker_charge_3,

3.3 数学计算

加减乘除肯定是支持的。那么剩下的就是一些内置的计算了。
ABS: 绝对值 abs(-1.7) = 1.7
FLOOR : 往低值取整 floor(-3.5) = -4 floor(3.5) = 3
CEIL: 往高值取整 ceil(-3.5) = -3 ceil(3.5) = 4
ROUND: 四舍五入 round(3.1415,2) = 3.14
CAST:这个关键字转换数据格式 CAST(操作数 AS 目标类型)
操作数:可以是数据源字段,数学表达式,case语句表达式,预定义功能
目标类型:可以是预定义的数据类型:ABAP int4、char
也可以是数据字典元素 S_CARR_ID

3.4 字符串表达

CONCAT(a,b) 连接字符,返回类型为CHAR或者STRING

REPLACE(a,b,c)这个意思是replace all occurrences of b in a with c 把a里面的所有b替换成c,返回类型是a的类型

SUBSTRING (a,pos,len)截取从pos后开始的len位。返回类型为a类型

LENGTH(a)取a的长度,返回值类型是个INT4型

3.5 货币和单位转换

就是说把源货币单位转换成目标货币单位,那么源单位和目标单位的转换关系就得有地方存。
unit_conversion(quantity=>a, source_unit=>b,target_unit=>c):这个转换规则一般通过CUNI这个tcode配置,保存在T006表里
CDS View_第7张图片

返回值类型为abap.quan
currency_conversion(amount=>a,source_currency=>b,target_currency=>c):这个同样的,得有换算比例,OB08设置在TCURR这个表里了。
返回值类型为abap.curr

这些自己做做例子。

CDS View_第8张图片CDS View_第9张图片

@AbapCatalog.sqlViewName: 'ZBW_SQL_TEST'
@AbapCatalog.compiler.compareFilter: true
@AbapCatalog.preserveKey: true
@AccessControl.authorizationCheck: #CHECK
@EndUserText.label: 'ZBW_CDS_TEST'
@AbapCatalog.buffering.status:#ACTIVE
@AbapCatalog.buffering.type: #SINGLE

define view ZBW_CDS_TEST 
as select from sbook
 {
   key sbook.carrid,
   key sbook.connid,
   fldate as flight_date,
   customid as custmer_id,
   custtype as custtype,
   smoker as smoker,
   @Semantics.amount.currencyCode: 'local_currency_key'
   loccuram as local_currency_amount,
   @Semantics.currencyCode: true
   loccurkey as local_currency_key,

   
   case smoker 
   when 'X' then loccuram + 100
   else loccuram
   end as inc_smoker_charge1,
   
   case 
      when smoker = 'X' and custtype = 'B'then loccuram + 200
      when smoker = 'X' and custtype = 'P'then loccuram + 100
      else loccuram
   end as inc_smoker_charge2,
   
   ceil(loccuram) as rounded_tohigh,
   cast(loccuram as abap.fltp)*0.85 as discounted,
   
   concat(carrid,connid) as mewid,
   length(concat(carrid,connid)) as length,
   
   currency_conversion(amount =>loccuram,
                       source_currency =>loccurkey,
                       target_currency=> cast('EUR'as abap.cuky( 5 )),
                       exchange_rate_date => fldate) as booking_amount
   
}

3.6 聚集语句

MIN(a) 取最小值 返回类型为a的类型
MAX(a)取最大值 返回类型为a的类型
SUM (a)取整 返回类型为a的类型
AVG(a)取平均 返回类型为floating
COUNT (*) 计算行数 返回类型为INT型
COUNT(DISTINCT a)计算不重复值得行数 返回类型为INT型

以上经常和GROUP BY或HAVING连用。
可以在select或者having语句中使用,select语句要给as别名
注意:当你使用聚集,那么没用聚集的那列要在group by里用,不然会报错。这时候Ctrl+1会自动提示
CDS View_第10张图片
只要你没用aggregation的,都会给你加在group by后面。
CDS View_第11张图片
having这样用:它就是用来限制group by返回值的。
CDS View_第12张图片

3.7 Join

对于Join有三种:内连,左外,右外。
这里有个要注意的点。
在ABAP的open SQL语法里,join是在字段后面。select A.a B.b from A inner join B on A.a = B.a
在eclipse里面我们是要把join放到字段选择之前,这样ADT语法编译可以自动帮我们补全查找错误之类的。

在CDS view里面用join的话,是不能用select *的。

  1. Inner Join
  2. Left Outer Join
  3. Right Outer Join
    CDS View_第13张图片
    CDS View_第14张图片
    嵌套Join也是可以的,直接在第一个select from 后面添加要嵌套的join,然后加括号把已经有的join放进去。
    这个join执行的时候会先执行括号里的join,然后把括号里join之后的结果作为外面的join的内容:
    CDS View_第15张图片

3.8 Union

在Union里面其实还有个Union ALL
这两个区别是:
Union不包含重复的行
Union ALL有可能包含重复行

在去Union的时候,有很多前提:

  1. 你union的表要有相同的字段个数。
  2. 在相同位置的字段,字段类型得能兼容
  3. 相同位置的字段名或者别名要相同
    CDS View_第16张图片union 就是两条,会把表里的重复行自动删除。
    CDS View_第17张图片
    union all就是所有
    CDS View_第18张图片
    另外,如果两个表的字段名不同,那就用个别名搞成一样的。

3.9 参数

也就是说给CDS View里面加个参数,CDS里面的参数是强制的。现在还没可选。你只要设置了参数,那就得输入。
参数就很简单了。
你要在一开始定义。
然后再在不同的地方调用。
直接看个例子:
CDS View_第19张图片
CDS View_第20张图片
以上参数,给类型的时候,可以参照数据元素,或者是给个预定义的类型。
调用的时候要用$parameters。 或者是前面加:调用。
但是最好是用第一种,因为ADT可以识别帮助。
在这里插入图片描述
直接定义和调用参数时,参数是必须输入的。
如果你借用系统数据,那就可以不必须输入。
比如下面这个,借用系统日期,需要加个注解的:
因为注解传递系统日期给参数sdate,所以无论你输不输这个参数,它都会填系统当前日期。
参数调用的地方多种多样,你可以用在case里,也可以用在where里,having里等等。
CDS View_第21张图片
CDS View_第22张图片

3.10 作为子CDS View

如果你要用已经存在的CDS view就很简单。就像用表一样。
如果这个要用到的CDS View有参数,那么用到的时候要把参数值直接给上,还要跟它本身参数定义顺序一致。
如果你有的参数是和外面的CDS View用的参数一样,那你去调用外面的CDS参数要给: 或者$parameters
CDS View_第23张图片到此都是在CDS View里面。
当然作为view也是可以在ABAP 程序里面调用的。
就是现在用新的OPEN SQL的语法。有些许不一样。
在ABAP 程序里调用有参数的CDS View:坑待填.

3.11 CDS View中的association关联

association 关联在数据库后台的表示和join没啥区别。
它可以提供一些简便的写法,还有加上基数。但是实际执行起来就是个join。
关联分三种:
Ad-hoc association: 定点关联,就是关联另外一个表或视图的一个或几个字段
exposed association: 全部关联,就是关联别人的整个表或者视图(前提条件是你这个关联的所有用来join的字段,都要展示在新定义的view里面)而且只有你这个整个表的字段真被用到了,才会在后台建View,不然后台是不会建的。
这个就和HANA里面的view有点像了,left outer join里面有一种情况是on demand,也就是只有你真用到了右表的数据,人家才会给你创建view,如果你搞半天只用到了左表的数据,他是不会给你建view的。
filtered associations: 就是关联里面加了个过滤,只是写法有些不一样。
CDS View_第24张图片
一般被关联的要名字前加_前短横。这样规定呢,估计是为了看着方便。
association后面会有中括号[0…1]这种就是映射基数。也就是说对于customid在_b中最少有0个,最多有1个对应的id。
最左边是最小对应,最后边是最大对应[1…*] *是无穷大。
如果缺省[1]就是把最小的0给省略了。
CDS View_第25张图片
这种右键看SQL creation
CDS View_第26张图片
一下就能看出来CDS view的association和open SQL的区别了。除了语法上的区别,基本和join没区别。
CDS View_第27张图片
下面去写全部关联:
如果这样写就错了。为啥呢,因为人家说了,全部关联的话,你的join的字段也要在view视图里。
CDS View_第28张图片
所以要改成这样:
最终创建只有连接的左表,只有你真实用到右表数据才会创建view。
CDS View_第29张图片
啥个意思呢?比如你再去建一个view,用到这个包含全部连接的view。然后用到全部连接的右表的字段。才会生成前面这个view。不用不生成。
用的时候,别忘了加全部连接的右表的名字:也就是._b这个叫做路径表达式。
CDS View_第30张图片
过滤关联就是字段里面给个过滤,这个过滤啥意思呢?
就是这个name,只有country=DE的时候才给name,其他时候是空值。
过滤只作用于这个name字段。过滤的写在中括号里。
CDS View_第31张图片

CDS View_第32张图片
在ABAP 程序里调用有关联的CDS View:坑待填.

3.12 扩展CDS View

就是给CDS View来个append。但是这里涉及到一些注解。
注解里可以说给这个append一个带union的,带参数的等等的view。
CDS View_第33张图片
CDS View_第34张图片
扩展了之后这个新view就是7列。
CDS View_第35张图片
CDS View_第36张图片
去看它的SQLview,有个append。不一样的是,如果你去append一个数据字典表,那么有命名规则是ZZ或YY开头的字段,这里没有规定。
CDS View_第37张图片
回到被extend的view能看到个小圈圈,表示被扩展了。我下图有个警告,实际上是个像箭靶的小圈圈。
在这里插入图片描述

3.13 CDS权限

权限管理这是个比较大的话题。
但是这里只讲简单的CDS View的权限管理。

一般在BW里面,权限管理通过权限相关对象,权限profile,到角色,再到用户。

而在ABAP里面,传统的权限控制是这样的:

loop at lt_tab into ls_tab.
authority-check object 'ZS_carrid'
                        ID 'CARRID' FIELD ls_tab-carrid
                        ID 'ACTVT' FIELD '03'.
           if sy-subrc <> 0.
           delete lt_tab index sy-tabix.
           endif.
endloop.

这种要从数据库先大量查找,然后删除。
但是现在我们是代码下推到数据库层,也就是只有需要的数据量才会被带到application server层。所以原先的逻辑就变了。
现在是咋了呢?
就是直接在一个CDS view上建一个DCL的role,即时权限控制。我控制你到这个CDS View的权限,只能看这个CDS View里面的哪些哪些数据。
但是有一点就是,我觉得是个bug。如果我从open SQL直接访问这个CDS View那么权限控制是有用的。如果我从其他的CDS View来访问这个CDS View那么在这个CDS View上面的权限控制就不起作用了。

关键字就是 define role, grant select on,等等
CDS View_第38张图片
这边你建了role,那么CDS View里面就可以给一个注解了:
在这里插入图片描述
那么要讲建role。哎,这也是个大话题。反正在BW里面能牵扯到很多。
不过先简单来看看,从SU21开始。先建一个权限对象。
CDS View_第39张图片
class相当于文件夹,文件夹里面有权限对象,权限对象里面有权限字段。这个所有组成了SAP_ALL这个profile。
如果你建了一个新对象,给了SAP_ALL这个profile,那他就有所有的权限对象的权限了。
CDS View_第40张图片
也就是常见的role下面的权限对象和权限字段。
CDS View_第41张图片

再往权限字段里面填值,就是它允许的用来限制权限的值了。
CDS View_第42张图片
建了role之后,要生成profile才有用啊。因为实际上是profile来控制的。有效方向盘的就是自建的role生成的profile。profile就是你的权限对象和权限字段及权限字段给的值得文档记录。
CDS View_第43张图片
CDS View_第44张图片
如果我们自己建也是可以的,就是在SU21下面建权限对象class,然后建权限对象,然后填入你的权限字段。最后填允许的值。
再接着呢,就给它加到role里面。再赋给用户。生成role的时候必须生成profile,也就是把profile给了用户。
CDS View_第45张图片
这个就是简单的给权限字段限制。
比如你在这里设置CDS View的权限相关字段,那就给个比如CARRID这个字段。然后只给‘AA’的值。最后再建role,然后赋给用户。

到此并没有结束:
还要建个CDS Access Control
CDS View_第46张图片
CDS View_第47张图片
这下等你去在ABAP 程序里跑CDS View的时候就只能跑你限定的值。以上是个例子,carrid是你建的权限对象里的字段(同时是CDSView里面想要限定的值),并且在role里面给值。

4. AMDP

重启下一篇。
AMDP.

你可能感兴趣的:(BW4HANA,HANA,其他)