查看“︁数据流分析”︁的源代码
←
数据流分析
跳转到导航
跳转到搜索
因为以下原因,您没有权限编辑该页面:
您请求的操作仅限属于该用户组的用户执行:
用户
您可以查看和复制此页面的源代码。
{{Copy edit|time=2023-05-08T06:06:27+00:00}} {{NoteTA |T=zh-cn:数据流分析;zh-tw:資料流分析;zh-hk:資料流分析; |G1=IT }} '''数据流分析''' 是一种用于收集[[计算机程序]]在不同点计算的值的信息的技术。一个程序的[[控制流圖]](control flow graph, CFG)被用来确定对变量的一次赋值可能传播到程序中的哪些部分。这些信息通常被[[编译器]]用来[[优化]]程序。数据流分析的一个典型的例子就是[[可到达定义]]的计算。 进行数据流分析的最简单的一种形式就是对控制流图的某个节点建立数据流方程,然后通过迭代计算,反复求解,直到到达[[不动点]]。这個方法是由[[蓋瑞·基爾多]]在[[海军研究生院]]任教时发明的。<ref>{{cite journal |last=Kildall |first=Gary |date=1973 |title=A Unified Approach to Global Program Optimization |journal=Proceedings of the 1st Annual ACM SIGACT-SIGPLAN Symposium on Principles of Programming Languages |url=http://portal.acm.org/citation.cfm?id=512945&coll=portal&dl=ACM |accessdate=2006-11-20 }}</ref> == 基本原理 == 数据流分析试图获得程序中每一点的特定信息。通常,在[[基本块]]的界限内就可以获得这些信息,因为很容易计算基本块中的信息。在前向流分析(forward flow analysis)中,一个块的结束状态是这个块起始状态的一个函数。函数由块内的语句的影响信息组成。一个块的开始状态是它的前驱的结束状态的函数。这就产生了一系列的数据流方程: 对于每一个块b: : <math> out_b = trans_{b}(in_b) </math> : <math> in_b = join_{p \in pred_b}(out_p) </math> 在这里,<math> trans_b </math> 是块<math>b</math>的 '''转移函数'''。它作用于入口状态<math>in_b</math>,并产生出口状态<math>out_b</math>。'''连接运算符''' <math>join</math>将块<math>b</math>的前驱节点<math>p \in pred_b</math>的出口状态联合起来,产生入口状态<math>b</math>。 在求解这一系列方程之后,块的入口和出口状态可以被用来获得程序在块内的属性。每条语句的转移函数可以被分别的用于获得在一个基本块内的某一点的信息。 每一个特定类型的数据流分析都有它自己的特定的转移函数和连接运算符。一些数据流问题需要后向数据流分析。和前向数据流分析类型相比,后向数据流分析使用的转移函数使用出口状态来产生入口状态,连接运算符作用于后继节点的入口状态以产生出口状态。 (在前向流分析中的)[[入口点]]起着重要的作用:因为它没有前驱节点,它的入口信息在分析开始时是明确的。比如,可以确定的局部变量的值的集合此时为空。如果控制流图并不包含迴圈(在程序中显性的或隐性的[[控制流程#迴圈|迴圈]]),只需直接求解数据流方程即可。此时可以对控制流图的基本块进行[[拓扑排序]];按照排序后的结果依次计算,则每个块的入口状态都可以在块起始处计算,因为此时块的所有前驱节点都已经计算过了,所以它们的出口状态是可以获得的。如果控制流图包含循环,那么就需要一个更高级的算法。 == 迭代算法 == 最常用的用于求解数据流方程的方法是使用迭代算法。它由每个块的近似入口状态信息出发。然后应用转移函数基于这些入口状态信息计算出口状态信息。然后,使用连接运算符更新入口状态信息。最后两步将一直进行下去直至到达'''不动点''': 即此时入口状态信息和出口状态信息都不再改变。 一个基本的求解数据流方程的算法是'''循环迭代算法''': :for ''i'' ← 1 to ''N'' ::''初始化节点i'' :while (''有集合发生了改变'') ::for ''i'' ← 1 to ''N'' :::''重新计算节点i处的集合'' === 收敛性分析 === 为了可用性的要求,迭代算法应该可以真正到达不动点。这可以通过对状态的值域的联合,转移函数以及连接运算符加上限制条件来保证。 值域应该是'''有界的 '''且有序。(比如,不存在一个无限的递增链<math>x_1</math> < <math>x_2</math> < ...)。转移函数和连接运算符应该是单调的。单调性保持了对值的每次迭代操作要么与之前一致,要么会变大,而有界性保证了它不会无限增大下去。因此最终我们会到达一个状态对于所有的x,有T(x) = x,此时即是不动点。 === 后向分析=== * '''活性变量''' {{en-link|活跃变量分析|Live-variable analysis}}计算每个程序点上的那些在重新定义前可以被潜在的使用的变量。这个的典型应用是用于[[死码删除]],移除那些给变量赋值但之后变量未被使用的语句。 块的入口状态是在上个块的末端仍然具有活性的变量的集合。 {| |----- | // out: {} b1: a = 3; b = 5; d = 4; if a > b then // in: {a,b,d} // out: {a,b} b2: c = a + b; d = 2; // in: {b,d} // out: {b,d} b3: endif c = 4; return b * d + c; // in:{} |} == 其它方法 == 在2002年,Mohnen描述了数据流分析的一个新方法,这种方法不需要显式的构造数据流图,<ref>{{cite journal |last=Mohnen |first=Markus |year=2002 |title=A Graph-Free Approach to Data-Flow Analysis |journal=Lecture Notes in Computer Science |volume=2304 |pages=185–213 |url=http://www.springerlink.com/content/a57r0a6q999p15yc/ |accessdate=2008-12-02 |author= |archive-url=https://web.archive.org/web/20110102204122/http://www.springerlink.com/content/a57r0a6q999p15yc/ |archive-date=2011-01-02 |dead-url=yes }}</ref>取代了依赖于程序的[[抽象解释]],并保持程序计数器工作列表集合的方法。在每个条件分支,对应的两条路径的目标都被加入工作列表中。每一条路径被尽可能多的指令跟随(直到程序结束点或直到没有变化),然后从工作列表中被移除并取回下一个程序计数。 == 位向量问题 == 上述例子中的问题是数据流的值是一个集合,比如[[可到达定义]]集(使用比特位来表示程序中定义的位置),或是活性变量的集合。这些集合可以通过'''位向量'''有效的表示,向量中的每一位表示一个特定元素是否属于集合。使用这种表示,连接运算和转移函数就可以通过按位逻辑运算实现。连接运算符是一个典型的取并集或取交集操作,可以通过位运算符''逻辑或''和''逻辑与''实现。每个块的转移函数可以被分解为''产生集''和''杀死集''。 举例来说,在活性变量分析中,连接运算符是取并集操作。''杀死集''就是那些在当前块中被重新定义的变量的集合,而''产生集''就是当前块中在重新定义变量值之前使用的变量的集合。数据流方程因此变为 <math> in_b = \bigcup_{s \in succ_b} out_s </math> <math> out_b = (in_b - kill_b) \cup gen_b </math> 在逻辑运算中,这可以看做是 :in(b) = 0 :for s in succ(b) ::in(b) = in(b) or out(s) :out(b) = (in(b) and not kill(b)) or gen(b) == 敏感性分析讨论 == 数据流分析本质上是流分析。数据流分析是典型的路径不敏感的,尽管可以定义数据流方程产生路径敏感的分析是可能的。 ''以下介绍的内容并不特定于数据流分析。'' * 一个'''流敏感'''的分析会考虑程序中语句的顺序。举例来说,一个流不敏感的指针分析可能认为"变量''x''和''y''可能指向了同一位置",而一个流敏感分析会认为"在语句20后,变量''x''和''y''可能指向了同一位置"。 * 一个'''路径敏感'''的分析计算了依赖于分支条件的谓词的不同的信息。比如,如果一个分支条件是''x>0'',那么在条件不满足的分支,分析会假设''x<=0'';而在满足条件的分支,会假设''x>0''确实成立。 * 一个'''上下文敏感'''的分析是一个''交互过程''分析,在分析目标函数的调用时它将考虑调用的信息。特别的,使用上下文信息,可以''回退''到原始的调用点,而如果没有这种信息,分析时就必须回退到所有可能的调用点,而丧失潜在的精度。 == 相关链接 == * [[可到达定义]] * [[活性分析]] * [[明确赋值分析]] ==备注== {{reflist}} == 补充书目、地址及网址 == *Aho, Alfred V. Sethi, Ravi. Ullman, Jeffrey D. ''Compilers: Principles, Techniques and Tools''. Addison Wesley. 1986. *Appel, Andrew W. ''Modern Compiler Implementation in ML''. Cambridge University Press. 1999. *Cooper, Keith D. and Torczon, Linda. ''Engineering a Compiler''. Morgan Kaufmann. 2005. *Muchnick, Steven S. ''Advanced Compiler Design and Implementation''. Morgan Kaufmann. 1997. *Hecht, Matthew S. ''Flow Analysis of Computer Programs''. Elsevier North-Holland Inc. 1977. [[Category:編譯器最佳化]] [[Category:静态程序分析]] {{编译器优化}}
该页面使用的模板:
Template:Cite journal
(
查看源代码
)
Template:Copy edit
(
查看源代码
)
Template:En-link
(
查看源代码
)
Template:NoteTA
(
查看源代码
)
Template:Reflist
(
查看源代码
)
Template:编译器优化
(
查看源代码
)
返回
数据流分析
。
导航菜单
个人工具
登录
命名空间
页面
讨论
不转换
查看
阅读
查看源代码
查看历史
更多
搜索
导航
首页
最近更改
随机页面
MediaWiki帮助
特殊页面
工具
链入页面
相关更改
页面信息