昨日万圣节ABAP怪兽级代码谜团,公布答案啦
首先非常感谢大家在周末还抽出宝贵的时间耗在Jerry昨天发布的文章 一段让人瑟瑟发抖的ABAP代码 上面。
虽然Jerry在文末开玩笑的声称,只有文章阅读量上千或者评论数超过50,才公布答案。其实这只是Jerry的玩笑,因为正如Jerry在之前一篇文章 5000粉丝数达成,感谢大家一如既往的支持 里提到,在微信自媒体泛滥的今天,大家能够在众多公众号中关注汪子熙公众号,并抽出时间来阅读Jerry或一本正经或胡说八道的文字,Jerry真的很感激,谢谢大家。
在分析ABAP代码谜题之前,Jerry还想讲一个故事。Jerry 2007年加入SAP成都研究院时,我老板的老板是E君,当时就已经是在SAP界打拼很多年的老江湖了。E君平时表情严肃的时候居多,做事雷厉风行,很有领导的范,包括Jerry在内的很多刚毕业入职的新人们都很害怕他(当然Jerry现在不怕了,哈哈)。
2008年的时候,Jerry还是标准的ABAP菜鸟一枚,有一天学到了如何使用ABAP发送邮件给一个Distribution List,并任意指定邮件的SendTo字段。Jerry那时觉得这个技巧很酷(当时的确够菜的 -_-),正好当时我在开发一个工具,需要向整个团队的DL汇报进度。于是Jerry在用邮件汇报进度的时候,没有采用在Outlook里编写邮件然后发送的方式,而是写了一段ABAP代码,把邮件发送给了整个团队。很多同事收到邮件后,因为SendTo字段为空,所以不知道这封邮件是谁发的。当时Jerry觉得这很酷。
后来E君把我叫到他的办公室谈工作上的事情时,特意提到了这封邮件,他说他不用猜都知道一定是我发的,然后问我为什么要这样做,听完我的解释之后,先说了句:“以后别这样。”然后给我分析了原因。从那以后,Jerry慢慢地开始懂得,作为一个SAP应用开发人员,再新再酷再吸引眼球的技术,如果不能为业务服务,不能为客户服务,那也make no sense at all.
回到题目本身。这道题不过是用于万圣节搞怪消遣的产物罢了,相信没有任何ABAP开发顾问会在实际工作者去模仿这种风格来编码。
Jerry给大家介绍这个谜题,目的不是在炫耀ABAP这门语言的一些奇技淫巧,而是觉得我们仍然可以从谜题本身找到一些积极的因素,比如借此弄清楚一些平时掌握得似是而非的ABAP语言特性。
(1) 从评论区能看出,很多朋友都找到了谜题的突破口,即 NOT=>NOT( NOT ). 这是典型的ABAP类静态方法调用的语法,因此说明,在名为NOT的INCLUDE里,包含了一个名为NOT的ABAP类,有一个名为NOT的静态方法。同时,这个静态方法调用的前面出现了OR这个布尔逻辑运算符,只能有一种情况才能通过语法检查,就是NOT静态方法的输出参数为RETURNING类型,然后该输出参数作为OR的操作数。
值得一提的是,很多其他编程语言都禁止使用关键字保留字来命名标识符或者类,而ABAP却没有这个限制,显得有点特立独行。
(2) 也有朋友在评论区提到,代码可执行部分以IF开头,但是却没有以ENDIF结尾。唯一的解释,就是在NOT这个include里,声明了包含ENDIF语句的宏,并把宏的名称取名为NOT.
(3) ABAP里感叹号的用法。
ABAP帮助文档里说的很清楚,!作为ABAP里的转义字符,能够告诉ABAP编译器,!后面紧跟的并不是关键字,而是普通的ABAP标识符。
给出的例子也很清晰,如果有人非要用CHANGING和USING作为形式参数的名称,只需要在前面加上感叹号即可。
而如果感叹号后面跟的并不是真的ABAP关键字,而是普通的标识符,那又会如何呢?
答案是,此时感叹号会直接被忽略。看下面的例子,加上感叹号的效果和不加一致。
(4) 现在我们已经知道了,题目中的!NOT,暗示大家在NOT include里,还定义了一个名为NOT的变量。
首先我们把谜题里迷惑人眼球的障眼法全部拿掉。
在有NOT参与的ABAP逻辑判断语句里,出现偶数个NOT,相当于一个NOT也未出现过(类似负负得正的原理),出现奇数个NOT,只相当于出现一个NOT.
所以题目中那多余的一系列NOT,就像《笑傲江湖》中衡山掌门莫大先生那套“衡山百变千幻云雾十三式”中的虚招一样,能让不明就里的对手眼花缭乱。
人称“琴中藏剑,剑发琴音”的莫大,凭借这手如梦如幻的剑法,在衡山城外击杀了嵩山十三太保之一的“大嵩阳手”费彬。即使武功强如费彬,也没弄看透莫大剑法中的虚招。而亲爱的ABAP顾问们,这道谜团中重复的NOT虚招,大家看透了没?
仔细观察代码中所有出现!NOT的地方,按照上述法则去除掉多余的NOT之后,能够提取出两个规律:
a. !NOT 前面至少有一个IF,OR或者AND
b. !NOT 后面直接结束,并未出现 IS INITIAL或者 > XXX, <> XXX等判断语句。
什么样的ABAP变量类型允许这种操作呢?
整型不行:
字符串类型不行:
而SELECTION-OPTIONS就可以。
这个SELECTION-OPTIONS是ABAP古董级的功能了,在SAPGUI下做Dynpro开发的顾问们会经常用,而SAP Cloud Platform ABAP编程环境下已经不再支持了。
上述四个语法点逐一突破后,如何编写NOT include的源码,思路也就清晰了。
源代码如下:
如果想复制粘贴这段代码,可以访问Jerry的github:
https://github.com/i042416/KnowlegeRepository/blob/master/ABAP/backup/NOT.include.abap
感谢阅读。
要获取更多Jerry的原创文章,请关注公众号"汪子熙":
昨日万圣节ABAP怪兽级代码谜团,公布答案啦的更多相关文章
- abap调用代码块
1:abap 调用代码块. *&---------------------------------------------------------------------* *& Re ...
- 如何减小ABAP业务代码的复杂度
在程序开发的过程中,相同的功能往往有不同的实现方式.对于可以实现同样功能的不同代码,复杂度是用于比较其质量优劣的重要指标. 在本文中,代码复杂度是指代码被理解/修改的难易程度.越容易被理解.修改的代码 ...
- 业界良心:Square开源Viewfinder,25万行代码全公布!
http://www.csdn.net/article/2014-05-06/2819633-square-open-sources-viewfinder https://github.com/vie ...
- Java,JavaScript和ABAP通过代码取得当前代码的调用栈Callstack
Java StackTraceElement stack[] = Thread.currentThread().getStackTrace(); System.out.println("Ca ...
- ABAP Util代码
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
- 搜索ABAP程序代码中的字符串
标准程序名:RPR_ABAP_SOURCE_SCAN /BEV1/NERM07DOCS
- 使用ABAP操作Excel的几种方法
这篇文章本来不在我计划之内,因为最近一个朋友微信上问到我这个问题,但我平时在SAP研究院工作中从没遇到过需要用ABAP操作Excel的需求,因此也没有太多技术实现细节可以分享给大家,只能泛泛写一些. ...
- SAP错误消息调试之七种武器:让所有的错误消息都能被定位
目录 长生剑 - SAPGUI Where Used List 碧玉刀 - ABAP调试器观察点 霸王枪 - ABAP调试器动态断点 多情环 - ABAP代码静态扫描 孔雀翎 - SAT 离别钩 - ...
- 【C语言探索之旅】 第二部分第九课: 实战"悬挂小人"游戏 答案
内容简介 1.课程大纲 2.第二部分第九课: 实战"悬挂小人"游戏 答案 3.第二部分第十课预告: 安全的文本输入 课程大纲 我们的课程分为四大部分,每一个部分结束后都会有练习题, ...
随机推荐
- MetaPruning: Meta Learning for Automatic Neural Network Channel Pruning
MetaPruning: Meta Learning for Automatic Neural Network Channel Pruning 2019-08-11 19:48:17 Paper: h ...
- odoo开发笔记 -- 模型(类)继承的几种机制
1. 类继承 2. 原型继承 3. 委托继承 待完善 https://www.cnblogs.com/chenshuquan/p/10523626.html
- Qt开发经验小技巧1-10
当编译发现大量错误的时候,从第一个看起,一个一个的解决,不要急着去看下一个错误,往往后面的错误都是由于前面的错误引起的,第一个解决后很可能都解决了. 定时器是个好东西,学会好使用它,有时候用QTime ...
- LinkedBlockingQueue与ArrayBlockingQueue
阻塞队列与普通的队列(LinkedList/ArrayList)相比,支持在向队列中添加元素时,队列的长度已满阻塞当前添加线程,直到队列未满或者等待超时:从队列中获取元素时,队列中元素为空 ,会将获取 ...
- 调用k8s api遇到CERTIFICATE_VERIFY_FAILED的问题解决方法
前言 python3.5.6版本,django1.1.12版本 最近要调用k8s接口,k8s接口的使用方法请参考官网的说明:戳我 调用k8s接口时,遇到 CERTIFICATE_VERIFY_FAIL ...
- Spring MVC -- 转换器和格式化
在Spring MVC -- 数据绑定和表单标签库中我们已经见证了数据绑定的威力,并学习了如何使用表单标签库中的标签.但是,Spring的数据绑定并非没有任何限制.有案例表明,Spring在如何正确绑 ...
- [LeetCode] 231. Power of Two 2的次方数
Given an integer, write a function to determine if it is a power of two. Example 1: Input: 1 Output: ...
- [LeetCode] 801. Minimum Swaps To Make Sequences Increasing 最少交换使得序列递增
We have two integer sequences A and B of the same non-zero length. We are allowed to swap elements A ...
- 寄存器vs缓存vs硬盘
对于多核cpu来说(一个处理器cpu上有多个核),L1/L2是各个核独自的,L3是多个核共享的 如下配置:一个处理器cpu,六个核.处理器速度为2.2GHz即电流每秒钟可以振荡22亿次.二级缓存256 ...
- Python进阶之面向对象
新式类与旧式类 区别: 在2.2版本之前所有的类都是旧式类,3.x版本已取消旧式类 旧式类一般的写法,不继承任何父类 class Person: def __init__(self, name): s ...