谓词记录

首先,我们需要知道何谓谓词,让我们看看官方的解释:

The NSPredicate class is used to define logical conditions used to constrain a search either for a fetch or for in-memory filtering.

NSPredicate类是用来定义逻辑条件约束的获取或内存中的过滤搜索。

可以使用谓词来表示逻辑条件,用于描述对象持久性存储在内存中的对象过滤。其实意思就是:我是一个过滤器,不符合条件的都滚开。

一、NSPredicate的基本语法

我们使用一门语言,无论是外语还是计算机语言,总是从语法开始的,这样我们才能正确的把握逻辑。所以我们从语法开始说起。在这部分我们仅关心其语法的使用

只要我们使用谓词(NSPredicate)都需要为谓词定义谓词表达式,而这个表达式必须是一个返回BOOL的值。

谓词表达式由表达式、运算符和值构成。

1.比较运算符

比较运算符如下

  • =、==:判断两个表达式是否相等,在谓词中=和==是相同的意思都是判断,而没有赋值这一说

1

2

3

4

5

NSNumber *testNumber = @123;

  NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF = 123"];

  if ([predicate evaluateWithObject:testNumber]) {

      NSLog(@"testString:%@", testNumber);

  }

我们可以看到输出的内容为:


1

2016-01-07 11:12:27.281 PredicteDemo[4130:80412] testString:123

  • >=,=>:判断左边表达式的值是否大于或等于右边表达式的值
  • <=,=<:判断右边表达式的值是否小于或等于右边表达式的值
  • >:判断左边表达式的值是否大于右边表达式的值
  • <:判断左边表达式的值是否小于右边表达式的值
  • !=、<>:判断两个表达式是否不相等
  • BETWEEN:BETWEEN表达式必须满足表达式 BETWEEN {下限,上限}的格式,要求该表达式必须大于或等于下限,并小于或等于上限

1

2

3

4

5

6

7

NSNumber *testNumber = @123;

NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF BETWEEN {100, 200}"];

  if ([predicate evaluateWithObject:testNumber]) {

      NSLog(@"testString:%@", testNumber);

  else {

      NSLog(@"不符合条件");

  }

输出结果为:


1

2016-01-07 11:20:39.921 PredicteDemo[4366:85408] testString:123

2.逻辑运算符

  • AND、&&:逻辑与,要求两个表达式的值都为YES时,结果才为YES。

1

2

3

4

NSArray *testArray = @[@1, @2, @3, @4, @5, @6];

  NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF > 2 && SELF < 5"];

  NSArray *filterArray = [testArray filteredArrayUsingPredicate:predicate];

  NSLog(@"filterArray:%@", filterArray);

输出结果为:


1

2

3

4

2016-01-07 11:27:01.885 PredicteDemo[4531:89537] filterArray:(

  3,

  4

)

  • OR、||:逻辑或,要求其中一个表达式为YES时,结果就是YES
  • NOT、 !:逻辑非,对原有的表达式取反

3.字符串比较运算符

  • BEGINSWITH:检查某个字符串是否以指定的字符串开头(如判断字符串是否以a开头:BEGINSWITH ‘a‘)
  • ENDSWITH:检查某个字符串是否以指定的字符串结尾
  • CONTAINS:检查某个字符串是否包含指定的字符串
  • LIKE:检查某个字符串是否匹配指定的字符串模板。其之后可以跟?代表一个字符和*代表任意多个字符两个通配符。比如"name LIKE ‘*ac*‘",这表示name的值中包含ac则返回YES;"name LIKE ‘?ac*‘",表示name的第2、3个字符为ac时返回YES。
  • MATCHES:检查某个字符串是否匹配指定的正则表达式。虽然正则表达式的执行效率是最低的,但其功能是最强大的,也是我们最常用的。

注:字符串比较都是区分大小写和重音符号的。如:café和cafe是不一样的,Cafe和cafe也是不一样的。如果希望字符串比较运算不区分大小写和重音符号,请在这些运算符后使用[c],[d]选项。其中[c]是不区分大小写,[d]是不区分重音符号,其写在字符串比较运算符之后,比如:name LIKE[cd] ‘cafe‘,那么不论name是cafe、Cafe还是café上面的表达式都会返回YES。

4.集合运算符

  • ANY、SOME:集合中任意一个元素满足条件,就返回YES。
  • ALL:集合中所有元素都满足条件,才返回YES。
  • NONE:集合中没有任何元素满足条件就返回YES。如:NONE person.age < 18,表示person集合中所有元素的age>=18时,才返回YES。
  • IN:等价于SQL语句中的IN运算符,只有当左边表达式或值出现在右边的集合中才会返回YES。我们通过一个例子来看一下

1

2

3

4

NSArray *filterArray = @[@"ab", @"abc"];

  NSArray *array = @[@"a", @"ab", @"abc", @"abcd"];

  NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (SELF IN %@)", filterArray];

  NSLog(@"%@", [array filteredArrayUsingPredicate:predicate]);

代码的作用是将array中和filterArray中相同的元素去除,输出为:


1

2

3

4

2016-01-07 13:17:43.669 PredicteDemo[6701:136206] (

  a,

  abcd

)

  • array[index]:返回array数组中index索引处的元素
  • array[FIRST]:返回array数组中第一个元素
  • array[LAST]:返回array数组中最后一个元素
  • array[SIZE]:返回array数组中元素的个数

5.直接量

在谓词表达式中可以使用如下直接量

  • FALSE、NO:代表逻辑假
  • TRUE、YES:代表逻辑真
  • NULL、NIL:代表空值
  • SELF:代表正在被判断的对象自身
  • "string"或‘string‘:代表字符串
  • 数组:和c中的写法相同,如:{‘one‘, ‘two‘, ‘three‘}。
  • 数值:包括证书、小数和科学计数法表示的形式
  • 十六进制数:0x开头的数字
  • 八进制:0o开头的数字
  • 二进制:0b开头的数字

6.保留字

下列单词都是保留字(不论大小写)

AND、OR、IN、NOT、ALL、ANY、SOME、NONE、LIKE、CASEINSENSITIVE、CI、MATCHES、CONTAINS、BEGINSWITH、ENDSWITH、BETWEEN、NULL、NIL、SELF、TRUE、YES、FALSE、NO、FIRST、LAST、SIZE、ANYKEY、SUBQUERY、CAST、TRUEPREDICATE、FALSEPREDICATE

注:虽然大小写都可以,但是更推荐使用大写来表示这些保留字

二、谓词的用法

1.定义谓词

一般我们使用下列方法来定义一个谓词


1

NSPredicate *predicate = [NSPredicate predicateWithFormat:];

下面我们通过几个简单的例子来看看它该如何使用:

首先我们需要定义一个模型,因为示例中需要用到它

ZLPersonModel.h


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

#import typedef NS_ENUM(NSInteger, ZLPersonSex) {

    ZLPersonSexMale = 0,

    ZLPersonSexFamale

};

@interface ZLPersonModel : NSObject

/** NSString 姓名 */

@property (nonatomic, copy) NSString *name;

/** NSUInteger 年龄 */

@property (nonatomic, assign) NSUInteger age;

/** ZLPersonSex 性别 */

@property (nonatomic, assign) ZLPersonSex sex;

+ (instancetype)personWithName:(NSString *)name age:(NSUInteger)age sex:(ZLPersonSex)sex;

@end

下面让我们进入正题

例一:(最简单的使用)


1

2

3

4

5

6

7

8

9

10

11

ZLPersonModel *sunnyzl = [ZLPersonModel personWithName:@"sunnyzl" age:29 sex:ZLPersonSexMale];

    ZLPersonModel *jack = [ZLPersonModel personWithName:@"jack" age:22 sex:ZLPersonSexMale];

    //  首先我们来看一些简单的使用

    //  1.判断姓名是否是以s开头的

    NSPredicate *pred1 = [NSPredicate predicateWithFormat:@"name LIKE ‘s*‘"];

    //  输出为:sunnyzl:1, jack:0

    NSLog(@"sunnyzl:%d, jack:%d", [pred1 evaluateWithObject:sunnyzl], [pred1 evaluateWithObject:jack]);

    //  2.判断年龄是否大于25

    NSPredicate *pred2 = [NSPredicate predicateWithFormat:@"age > 25"];

    //  输出为:sunnyzl的年龄是否大于25:1, jack的年龄是否大于25:0

    NSLog(@"sunnyzl的年龄是否大于25:%d, jack的年龄是否大于25:%d", [pred2 evaluateWithObject:sunnyzl], [pred2 evaluateWithObject:jack]);

看到这里我们会发现evaluateWithObject:方法返回的是一个BOOL值,如果符合条件就返回YES,不符合就返回NO。而即使是最简单的使用也有一些大用处,比如以前我们写判断手机号码、邮编等等,像我就喜欢用John Engelhart大神的RegexKitLite,然而由于年代久远需要导入libicucore.dylib库(xcode7为libicucore.tbd)且由于是mrc又需要添加-fno-objc-arc,至此我们才能使用。然而使用谓词让我们可以用同样简洁的代码实现相同的功能

例二:判断手机号是否正确


1

2

3

4

5

6

 - (BOOL)checkPhoneNumber:(NSString *)phoneNumber

{

    NSString *regex = @"^[1][3-8]\\d{9}$";

    NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];

    return [pred evaluateWithObject:phoneNumber];

}

看到这里是不是感觉好爽,感觉以前所有的正则都可以这么匹配,但是谓词匹配正则时也是有缺点的,下面通过一个例子来看一下这个致命的缺点

例三:谓词匹配正则的缺点

(本意:检测字符串中是否有特殊字符)


1

2

3

4

5

6

- (BOOL)checkSpecialCharacter:(NSString *)string

{

    NSString *regex = @"[`[email protected]#$^&*()=|{}‘:;‘,\\[\\].<>/?~!@#¥……&*()——|{}【】‘;:”“‘。,、?]";

    NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];

    return [pred evaluateWithObject:string];

}

我们想要的效果是字符串中有特殊字符时就返回YES,然而梦想是美好的,现实是残酷的

让我们看看这悲催的结局


1

2

3

4

NSString *testString = @"!";

NSLog(@"是否含有特殊字符:%d", [self checkSpecialCharacter:testString]);

//  当testString为一个特殊字符时,我们惊喜的发现输出为

//  是否含有特殊字符:1

看到这里我们心里猛然一喜,这tmd根本没问题嘛

让我们修改下testString的值


1

2

3

4

NSString *testString = @"!~";

NSLog(@"%d", [self checkSpecialCharacter:testString]);

//  我们会发现悲催的结局来了输出为

//  是否含有特殊字符:0

再次修改testString的值


1

2

3

4

NSString *testString = @"abc!~d";

NSLog(@"%d", [self checkSpecialCharacter:testString]);

//  我们会发现输出为

//  是否含有特殊字符:0

这总与我们的想法事与愿违,看到这里我们会发现谓词对正则并不像我们使用NSRegularExpression时匹配的那么好,究其原因是为什么呢?我们用NSRegularExpression时会发现匹配到一个结果时就会存入数组,再从匹配到的位置继续向下匹配。

然而NSPredicate并不会做这样的自动操作,我们最终发现在NSPredicate输入[`[email protected]#$^&*()=|{}‘:;‘,\[\].<>/?~!@#¥……&*()——|{}【】‘;:”“‘。,、?]正则表达式时和写成^[`[email protected]#$^&*()=|{}‘:;‘,\[\].<>/?~!@#¥……&*()——|{}【】‘;:”“‘。,、?]$的效果是一样的。

所以通过这个例子我们总结出来,只有在正则表达式为^表达式$时才使用谓词,而不是所有情况都使用。

那么我们是不是因为这一点就摒弃它了呢,答案是否定的。因为虽然NSPredicate有这么一点瑕疵,但是它总体带给我们的便利其实除了正则表达式匹配时的这个问题外是更多的。

2.使用谓词过滤集合

此部分是我们需要掌握的重点,因为从这里我们就可以看到谓词的真正的强大之处

其实谓词本身就代表了一个逻辑条件,计算谓词之后返回的结果永远为BOOL类型的值。而谓词最常用的功能就是对集合进行过滤。当程序使用谓词对集合元素进行过滤时,程序会自动遍历其元素,并根据集合元素来计算谓词的值,当这个集合中的元素计算谓词并返回YES时,这个元素才会被保留下来。请注意程序会自动遍历其元素,它会将自动遍历过之后返回为YES的值重新组合成一个集合返回。

其实类似于我们使用tableView设置索引时使用的下段代码


1

2

3

4

- (NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView

{

    return [self.cityGroup valueForKey:@"title"];

}

中的[self.cityGroup valueForKey:@"title"]。它的作用是遍历所有title并将得到的值组成新的数组。

  • NSArray提供了如下方法使用谓词来过滤集合

- (NSArray*)filteredArrayUsingPredicate:(NSPredicate *)predicate:使用指定的谓词过滤NSArray集合,返回符合条件的元素组成的新集合

  • NSMutableArray提供了如下方法使用谓词来过滤集合

- (void)filterUsingPredicate:(NSPredicate *)predicate:使用指定的谓词过滤NSMutableArray,剔除集合中不符合条件的元素

  • NSSet提供了如下方法使用谓词来过滤集合

- (NSSet*)filteredSetUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0):作用同NSArray中的方法

  • NSMutableSet提供了如下方法使用谓词来过滤集合

- (void)filterUsingPredicate:(NSPredicate *)predicate NS_AVAILABLE(10_5, 3_0):作用同NSMutableArray中的方法。

通过上面的描述可以看出,使用谓词过滤不可变集合和可变集合的区别是:过滤不可变集合时,会返回符合条件的集合元素组成的新集合;过滤可变集合时,没有返回值,会直接剔除不符合条件的集合元素

下面让我们来看几个例子:

例一:


1

2

3

4

5

6

7

8

9

10

11

12

13

NSMutableArray *arrayM = [@[@20, @40, @50, @30, @60, @70] mutableCopy];

    //  过滤大于50的值

    NSPredicate *pred1 = [NSPredicate predicateWithFormat:@"SELF > 50"];

    [arrayM filterUsingPredicate:pred1];

    NSLog(@"arrayM:%@", arrayM);

    NSArray *array = @[[ZLPersonModel personWithName:@"Jack" age:20 sex:ZLPersonSexMale],

                       [ZLPersonModel personWithName:@"Rose" age:22 sex:ZLPersonSexFamale],

                       [ZLPersonModel personWithName:@"Jackson" age:30 sex:ZLPersonSexMale],

                       [ZLPersonModel personWithName:@"Johnson" age:35 sex:ZLPersonSexMale]];

    //  要求取出包含‘son’的元素

    NSPredicate *pred2 = [NSPredicate predicateWithFormat:@"name CONTAINS ‘son‘"];

    NSArray *newArray = [array filteredArrayUsingPredicate:pred2];

    NSLog(@"%@", newArray);

输出为


1

2

3

4

5

6

7

8

2016-01-07 16:50:09.510 PredicteDemo[13660:293822] arrayM:(

    60,

    70

)

2016-01-07 16:50:09.511 PredicteDemo[13660:293822] (

    "[name = Jackson, age = 30, sex = 0]",

    "[name = Johnson, age = 35, sex = 0]"

)

从这个例子我们就可以看到NSPredicate有多么强大,如果让我们用其他的方法来实现又是一大堆if...else。

让我们来回顾一下上面的从第二个数组中去除第一个数组中相同的元素

例二:


1

2

3

4

NSArray *filterArray = @[@"ab", @"abc"];

    NSArray *array = @[@"a", @"ab", @"abc", @"abcd"];

    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"NOT (SELF IN %@)", filterArray];

    NSLog(@"%@", [array filteredArrayUsingPredicate:predicate]);

输出为:


1

2

3

4

2016-01-07 13:17:43.669 PredicteDemo[6701:136206] (

    a,

    abcd

)

如果我们不用NSPredicate的话,肯定又是各种if...else,for循环等等。可以看出NSPredicate的出现为我们节省了大量的时间和精力。

3.在谓词中使用占位符参数

我们上面所有的例子中谓词总是固定的,然而我们在现实中处理变量时决定了谓词应该是可变的。下面我们来看看如果让谓词变化起来。

首先如果我们想在谓词表达式中使用变量,那么我们需要了解下列两种占位符:

  • %K:用于动态传入属性名
  • %@:用于动态设置属性值

其实相当于变量名与变量值

除此之外,还可以在谓词表达式中使用动态改变的属性值,就像环境变量一样


1

NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF CONTAINS $VALUE"];

上述表达式中,$VALUE是一个可以动态变化的值,它其实最后是在字典中的一个key,所以可以根据你的需要写不同的值,但是必须有$开头,随着程序改变$VALUE这个谓词表达式的比较条件就可以动态改变。

下面我们通过一个例子来看看这三个重要的占位符应该如何使用

例一:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

NSArray *array = @[[ZLPersonModel personWithName:@"Jack" age:20 sex:ZLPersonSexMale],

                     [ZLPersonModel personWithName:@"Rose" age:22 sex:ZLPersonSexFamale],

                     [ZLPersonModel personWithName:@"Jackson" age:30 sex:ZLPersonSexMale],

                     [ZLPersonModel personWithName:@"Johnson" age:35 sex:ZLPersonSexMale]];

  //  定义一个property来存放属性名,定义一个value来存放值

  NSString *property = @"name";

  NSString *value = @"Jack";

  //  该谓词的作用是如果元素中property属性含有值value时就取出放入新的数组内,这里是name包含Jack

  NSPredicate *pred = [NSPredicate predicateWithFormat:@"%K CONTAINS %@", property, value];

  NSArray *newArray = [array filteredArrayUsingPredicate:pred];

  NSLog(@"newArray:%@", newArray);

  

  //  创建谓词,属性名改为age,要求这个age包含$VALUE字符串

  NSPredicate *predTemp = [NSPredicate predicateWithFormat:@"%K > $VALUE", @"age"];

  // 指定$SUBSTR的值为 25    这里注释中的$SUBSTR改为$VALUE

  NSPredicate *pred1 = [predTemp predicateWithSubstitutionVariables:@{@"VALUE" : @25}];

  NSArray *newArray1 = [array filteredArrayUsingPredicate:pred1];

  NSLog(@"newArray1:%@", newArray1);

  

  //  修改 $SUBSTR的值为32,  这里注释中的SUBSTR改为$VALUE

  NSPredicate *pred2 = [predTemp predicateWithSubstitutionVariables:@{@"VALUE" : @32}];

  NSArray *newArray2 = [array filteredArrayUsingPredicate:pred2];

  NSLog(@"newArray2:%@", newArray2);

输出为


1

2

3

4

5

6

7

8

9

10

11

2016-01-07 17:28:02.062 PredicteDemo[14542:309494] newArray:(

  "[name = Jack, age = 20, sex = 0]",

  "[name = Jackson, age = 30, sex = 0]"

)

2016-01-07 17:28:02.063 PredicteDemo[14542:309494] newArray1:(

  "[name = Jackson, age = 30, sex = 0]",

  "[name = Johnson, age = 35, sex = 0]"

)

2016-01-07 17:28:02.063 PredicteDemo[14542:309494] newArray2:(

  "[name = Johnson, age = 35, sex = 0]"

)

从上例中我们主要可以看出来%K和$VALUE的含义。

时间: 2024-10-07 03:59:31

谓词记录的相关文章

【异常记录(五)】C# 无法发送具有此谓词类型的内容正文错误

今天请求接口直接调了以前写好的方法,结果报了(405)不支持方法的错误,一看是GET写成POST了,改成GET之后,又报了无法发送具有此谓词类型的内容正文错误的错误 原来之前的方法里面有GetRequestStream(), GET请求并不支持它. 把GetRequestStream()和相应的代码注释掉就OK了 // Stream outStream = myRequest.GetRequestStream(); // outStream.Write(arrB, 0, arrB.Length)

谓词函数、函数对象

从概念上讲,函数对象用作函数的对象:但是从实现上来说,函数对象时实现了 operate()的类的对象. 虽然函数和函数指针也可以归为函数对象,但实现了operate()的类的对象才能保存状态,才能用于STL. 我们直接看定义: 一元函数:接受一个参数的函数,如f(x). 一元谓词函数:如果一元函数返回一个BOOL类型的值,则该函数称为谓词. 二元函数:接受2个参数的函数,如f(x,y). 二元谓词函数:如果二元函数返回一个BOOL值,则该函数称为二元谓词. 之所以给返回布尔类型的函数对象专门命名

Oracle查询转换之连接谓词推入

连接谓词推入(Join Predicate  Pushdown)是优化器处理带视图的目标SQL的一种优化手段,它是指虽然优化器会把该SQL中视图的定义SQL语句当作一个独立单元来单独执行,但此时优化器会把原本处于该视图外部查询中和该视图之间的连接条件推入到该视图的定义SQL语句内部,这样是为了能使用上该视图内部相关基表上的索引,进而能走出基于索引的嵌套循环连接. 连接谓词推入所带来的基于索引的嵌套循环连接并不一定能走出更高效的执行计划,因为当做了连接谓词推入后,原目标SQL中的视图就和外部查询产

c++编程规范的纲要和记录 (转)

这是一本好书, 可以让你认清自己对C++的掌握程度. 看完之后,给自己打分,我对C++了解多少? 答案是不足20分. 对于我自己是理所当然的问题, 就不提了, 记一些有启发的条目和细节: (*号表示不能完全理解,实力升级了之后回头看) 一般性问题: 1. 不要在注释中重复写代码语义,这样很容易产生不一致. 应该编写的是解释方法和原理的说明性注释. 2. 不要对每个项目每个文件进行强制的排版格式规定, 在同一个文件中风格一致即可. 3. 匈牙利记法是混用了类型不安全语言中的设施, 在面向对象语言中

MySQL(三)--函数与谓词

前文已有涉及,这里作为总结重新整理一下. 一.函数 1.算术函数 NUMERIC 是大多数 DBMS 都支持的一种数据类型,通过 NUMBERIC ( 全体位数, 小数位数 ) 的形式来指定数值的大小. -- +-*/ --余数 SELECT n, p, MOD(n, p) AS mod_col FROM SampleMath; --绝对值 SELECT m, ABS(m) AS abs_col FROM SampleMath; --四舍五入 SELECT m, n, ROUND(m, n) A

oracle开发之&lt;&lt;SQL Cookbook&gt;&gt;学习笔记整理:第一章 检索记录

1.写程序时列出需要的列比直接写SELECT *  更直观.采用这种方式,行数据检索性能相同,列数据检索性能提高:选取部分需要的列时,降低IO和网络传输时间,提高性能. 2.WHERE子句筛选记录配合=,<,>,<=,>=,!,<>和AND,OR等谓词. 3.NULL值用IS NULL和IS NOT NULL来判断. 4.为列取别名,在子查询中特别重要. 5.使用||来拼接列的值. 6.使用case when end子句来实现if-else判断. 7.取top N时,需

SQL基础学习_05_函数、谓词、CASE表达式

函数 算术函数 1. 四则运算: +.-.*./? 2. ABS:求绝对值, ABS(数值) 3. MOD: 求余,MOD(被除数,除数) 4. ROUND:四舍五入,ROUND(对象数值,保留小数的位数) 字符串函数 1. 字符串拼接: ||, str1 || str2 || str3 || - || strn ?AS str_all 2. 字符串长度:LENGTH,LENGTH(str1) AS len_str1 3. 小写转换:LOWER,LOWER(str1) AS low_str 4.

SAP+DB2 糟糕的报表查询『ZCOR0015』 优化全程记录

ZCOR0015的优化全过程记录文档 2015年3月,今天无意翻到这篇写于2010年7月的文档,回想那时的工作,毕业3年初出茅庐的我面对接触不多的SAP+DB2竟敢操刀动斧,自信满满. 虽然这过程一路坎坷,数次判断几乎全都被打脸验证,看着如小强般坚毅的我,哈哈~ 文档里带着情绪的措词十分逗笑,可以洞悉那时的心情. ——如今5年过去,却久未闻硝烟,怀念ing~ 目录: ZCOR0015的优化全过程记录文档... 1 内容说明:1 1.寻找问题点,切入分析... 2 2.再次碰到问题,转折点.. 6

代码混淆之道——控制流扁平与不透明谓词理论篇

控制流是指代码执行时指令的执行顺序.在各种控制逻辑的作用下,程序会沿着特定的逻辑顺序执行.一般控制逻辑包括有无条件分支.循环.函数调用等. 本文原创作者:i春秋签约作家--penguin_wwy 一.扁平化的定义 本篇讲代码混淆的一个重要手段,控制流扁平化. 所谓控制流是指代码执行时指令的执行顺序.在各种控制逻辑的作用下,程序会沿着特定的逻辑顺序执行.一般控制逻辑包括有\无条件分支.循环.函数调用等.在正常情况下程序的逻辑非常好理解(代码逻辑不好的程序员都死了...),开发过程中有各种人为的行为