8. 让代码有判断力:分支结构

上一篇文章我们学了运算符——加加减减、比大小、判真假。但你有没有发现,我们写的程序还是一条直线走到底:从第一行执行到最后一行,从不拐弯。

真实世界里,决策无处不在:如果天冷就穿厚点,否则穿薄点;红灯停,绿灯行。程序要变得“聪明”,就必须具备这种判断力。C 语言给了我们两种主要的条件判断工具:if 系列switch。今天我们就让程序长出“大脑”,学会在不同条件下走不同的路。


一、if 语句:最简单的分岔口

最基本的 if 结构是这个样子的:

if (条件) {
    // 条件为真(非零)时执行的语句
}

条件可以是任何返回数值的表达式:如果结果是 0,就是“假”;非 0,就是“真”。关系表达式(如 a > b)返回 1 或 0,很适合放在这里。

一个简单的例子:判断一个人是否成年。

#include <stdio.h>

int main(void) {
    int age;
    printf("请输入你的年龄:");
    scanf("%d", &age);

    if (age >= 18) {
        printf("你已成年。\n");
    }

    printf("程序结束。\n");
    return 0;
}

如果输入的 age 大于等于 18,age >= 18 的值就是 1(真),花括号里的打印语句才会执行。如果小于 18,它就跳过那行,直接执行 程序结束

注意:条件一定要用圆括号括起来。如果花括号里只有一条语句,花括号可以省略,但不建议初学者省略——保留花括号能让逻辑一目了然,也避免以后加代码时出错。


二、if-else:二选一的岔路

如果既要处理“真”的情况,又要处理“假”的情况,就用 if-else

if (条件) {
    // 条件为真时执行
} else {
    // 条件为假时执行
}

改造刚才的例子:

if (age >= 18) {
    printf("你已成年。\n");
} else {
    printf("你未成年。\n");
}

两条路必走一条,不会同时走,也不会都不走。


三、else if:多岔路口

现实中的选择往往不止两个:分数分等级,菜单多选项。这时可以用 else if 链,它是 else 里再嵌套 if 的简写。

if (条件1) {
    // 条件1 真
} else if (条件2) {
    // 条件1 假 且 条件2 真
} else if (条件3) {
    // 条件1、2 假 且 条件3 真
} else {
    // 以上全假
}

举个例子:输入分数,输出等级。

#include <stdio.h>

int main(void) {
    int score;
    printf("请输入分数(0-100):");
    scanf("%d", &score);

    if (score >= 90) {
        printf("等级:A\n");
    } else if (score >= 80) {
        printf("等级:B\n");
    } else if (score >= 70) {
        printf("等级:C\n");
    } else if (score >= 60) {
        printf("等级:D\n");
    } else {
        printf("等级:F\n");
    }

    return 0;
}

执行逻辑:从上往下依次检查条件,第一个满足的被执行,后面的全部跳过。所以条件的顺序很重要——如果你把 score >= 60 放最前面,90 分也会掉进 D 档。务必把范围更小、更严格的条件放在前面


四、嵌套 if:岔路里还有岔路

你可以在 ifelse 的体内再写 if,形成嵌套结构。比如在一个成年人判断中,还分是否达到退休年龄:

if (age >= 18) {
    if (age >= 60) {
        printf("你已退休。\n");
    } else {
        printf("你已成年,还在工作。\n");
    }
} else {
    printf("你未成年。\n");
}

悬挂 else 问题:当嵌套层级多起来时,else 到底和哪个 if 配对?C 语言的规则是:else 和它前面最近的那个未配对的 if 结合。但为了人类阅读方便,请一定用缩进和花括号让意图清清楚楚。如果你不确定配对关系,用花括号明确指定:

if (条件1) {
    if (条件2) {
        // ...
    }
} else {   // 这个 else 绑定最外层的 if
    // ...
}

五、switch-case:多条等值分支的利器

当你要根据一个变量的多个具体值来选择不同路径时,if-else if 链写起来又长又重复。switch 就是为这种场景准备的。

基本结构:

switch (表达式) {
    case 常量1:
        // 语句
        break;
    case 常量2:
        // 语句
        break;
    // ...
    default:
        // 如果都不匹配,执行这里
        break;
}

一个菜单选择的例子:

#include <stdio.h>

int main(void) {
    int option;
    printf("请选择操作:1-加法 2-减法 3-退出\n");
    scanf("%d", &option);

    switch (option) {
        case 1:
            printf("你选择了加法。\n");
            break;
        case 2:
            printf("你选择了减法。\n");
            break;
        case 3:
            printf("程序退出。\n");
            break;
        default:
            printf("无效选项。\n");
            break;
    }

    return 0;
}

switch 的几个规则和陷阱

1. 表达式必须是整型(或能隐式转换成整型)

intcharenum 都可以,floatdouble、字符串不行。

2. case 后面必须是常量,不能是变量或范围

不允许 case a:(a 是变量),也不允许 case 1..10:。每个 case 是一个确定的值。

3. break 的作用:跳出开关

执行完一个 case 后,如果没有 break,程序会继续执行下一个 case 里的语句,这叫做“贯穿”(fall-through)。有时候故意利用它可以处理多个 case 共享同一段代码,但绝大多数情况你需要用 break 阻止它。

switch (grade) {
    case 'A':
    case 'B':
    case 'C':
        printf("及格\n");    // A、B、C 都会进到这里
        break;
    case 'D':
        printf("不及格\n");
        break;
}

4. default 是可选的

如果所有 case 都没匹配,就会执行 default(如果有的话)。一般建议写上,处理异常输入。


六、if vs switch:什么时候用哪个?

  • if:适合范围判断score >= 60)、复杂逻辑组合(a > 5 && b < 3)。
  • switch:适合离散的等值判断(菜单选择、按键码、星期几)。当分支超过三四个且是等值判断时,switch 更清晰,有时编译后的代码也更高效(编译器可能生成跳转表)。

七、条件运算符 ?: 的简洁用法

我们在上一篇运算符里提过一次,这里在分支结构里正式介绍。它是 C 语言里唯一的三目运算符:

变量 = 条件 ? 表达式1 : 表达式2;

如果条件为真,整个表达式的值是表达式1,否则是表达式2。它相当于一个简单的 if-else,但更简洁,适合条件二选一赋值的场景。

比如求两个数的最大值:

int a = 10, b = 20;
int max = (a > b) ? a : b;  // max = 20

再比如取绝对值:

int x = -5;
int abs_x = (x >= 0) ? x : -x;

不要滥用 ?: 写出多层嵌套,像 a ? b : c ? d : e 这样会让人抓狂。清晰永远比简短重要。


八、综合实战:简易计算器(用 switch 实现)

我们把上一篇文章末尾的简单计算器用 switch 改造一下,让结构更干净:

#include <stdio.h>

int main(void) {
    double num1, num2;
    char op;

    printf("请输入算式(例如 3 + 5):");
    scanf("%lf %c %lf", &num1, &op, &num2);

    switch (op) {
        case '+':
            printf("结果:%.2lf\n", num1 + num2);
            break;
        case '-':
            printf("结果:%.2lf\n", num1 - num2);
            break;
        case '*':
            printf("结果:%.2lf\n", num1 * num2);
            break;
        case '/':
            if (num2 != 0) {
                printf("结果:%.2lf\n", num1 / num2);
            } else {
                printf("错误:除数不能为零。\n");
            }
            break;
        default:
            printf("不支持的运算符。\n");
            break;
    }

    return 0;
}

你会发现 switch 比一堆 if-else if 更整洁,尤其是当分支较多时。


九、常见错误与防御建议

  1. if 条件里把 == 写成 =
    if (a = 5) 永远为真。用 if (5 == a) 可以避免,因为 5 = a 编译报错。

  2. switch 里忘记写 break
    这在逻辑上是合法的,但往往不是你想要的。养成给每个 casedefault 都写 break 的习惯。

  3. switch 的表达式用了浮点数
    switch (score / 10) 结果是浮点,不允许。必须显式转为整型。

  4. else 匹配错误
    用花括号明确划分边界,并且保持缩进一致。

  5. 布尔条件冗余
    if (flag == 1) 可以简写为 if (flag)if (flag == 0) 简写为 if (!flag)。简洁又清楚。


十、小结

今天你让程序长出了判断力。if 系列处理各种条件分支,switch 专精于等值多选,条件运算符提供了简洁的二选一。从现在起,你的程序不再是一条直愣愣的执行线,而是可以根据输入和状态走不同的路。

但这还不够——很多时候我们需要重复做一件事,而不是只选择一次。下一篇文章,我们就走进循环结构,让程序“勤劳”起来:forwhiledo-while,以及 breakcontinue 的控制。学会循环,你就能处理大量数据、实现复杂逻辑,真正感受到计算机“不知疲倦”的本领。


课后小练习

  1. 写一个程序,输入一个年份,判断它是不是闰年。闰年规则:能被 4 整除但不能被 100 整除,或者能被 400 整除。
  2. 输入三个整数,用 if 语句找出并输出最大值和最小值。
  3. switch 实现一个简单的星期提示器:输入 1-7 的数字,输出对应的星期几英文单词(Monday, Tuesday…),输入其他数字则输出“无效”。
  4. (陷阱挑战)阅读下面代码,你觉得输出是什么?然后实际运行验证,解释原因。
    #include <stdio.h>
    int main(void) {
        int x = 10;
        if (x > 5)
            printf("A");
        else if (x > 8)
            printf("B");
        else
            printf("C");
        return 0;
    }
    

我们下期见!

内容概要:本文详细记录了对一个Android ARM64静态ELF文件中字符串加密机制的逆向分析过程。该ELF文件的所有字符串均被加密,无法通过常规strings命令或IDA直接识别。作者通过分析发现,加密字符串存储在.rodata段,其解密所需信息(包括密文地址、长度和16位密钥)保存在.data.rel.ro段的40字节描述符中。核心解密函数sub_10F408采用自反的双pass流密码算法,结合固定密钥KEY_TERM(由.data段24字节数据计算得出),实现字节级非线性、位置与长度相关的加密。文章还复现了完整的Python解密脚本,并揭示了该保护机制的本质为代码混淆而非强加密,最终成功批量解密全部956条字符串,暴露程序真实行为,如shell命令模板、设备标识篡改、网络重置等操作。此外,文中还提及未启用的自定义壳框架及其反dump设计。; 适合人群:具备逆向工程基础的安全研究人员、二进制分析人员及对ELF保护技术感兴趣的开发者。; 使用场景及目标:①学习ELF二进制中字符串加密的典型实现方式与逆向突破口;②掌握从结构识别、函数追踪到算法还原的完整逆向流程;③理解“绑定二进制”的完整性校验设计及其局限性;④实践编写IDAPython脚本自动化提取与解密敏感数据。; 阅读建议:此资源以实战案例驱动,不仅展示技术细节,更强调逆向思维与验证方法,建议读者结合IDA调试环境,逐步跟随文中步骤进行动态分析与算法验证,深入理解每一步的推理依据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值