ctf show web入门 89

这是一道非常经典的 PHP 弱类型与正则表达式绕过的 CTF 题目。我们需要满足两个看似矛盾的条件来获取 $flag。

以下是完整的解题思路分析:

  1. 代码核心逻辑分析

我们先来看代码:

要拿到 flag,你的输入 ?num=xxx 必须同时满足以下两个条件:

条件一: preg_match("/0-9/", $num) 必须为 假 (False)。这意味着你的输入字符串中绝对不能包含任何 0 到 9 的数字,否则程序直接报错退出。

条件二: intval($num) 必须为 真 (True)。这意味着将你的输入转换成整数时,其值必须不等于 0(在 PHP 中,非 0 数字在 if 判断中会被视为 true)。

intval() 是 PHP 中的一个内置函数,全称是 integer value,它的核心功能是:获取变量的整数值。

简单来说,就是把各种类型的数据(字符串、浮点数、布尔值、数组等)强制转换成一个整数(Integer)。

  1. 解题思路(绕过方法)

不能包含数字,但转换成整数又不能是 0,这该怎么做?这就需要利用 PHP 的特性了。

原理: preg_match 的第一个参数预期接收的是一个字符串。如果你传入一个数组(例如 ?num\[\]=),preg_match 无法处理数组,会返回 false(或者报错,但不会触发 die)。

而在 PHP 中,intval(array()) 会根据数组是否为空来返回值。一个非空数组(例如 array(1) 或 array('a'))经 intval() 转换后,其结果是 1。

1 在 if 判断中代表 true,成功绕过!

Payload 构造:

?num\[\]=a

为什么?num=a不行

如果我们传入 ?num=a,确实可以通过第一个条件(因为 a 里面没有数字,preg_match 不会报错),但它会死在第二个条件上。在安全领域(如 CTF 或防 SQL 注入)中,intval() 对字符串的解析特性非常关键:

从左往右解析: intval() 会从字符串的左边第一个字符开始检查。

如果一开始就是数字,它会一直读到第一个非数字字符为止,然后把前面的数字转成整数。

如果一开始就不是数字(且不是正负号或空格),它会直接放弃,返回 0。

在第二关时,程序执行了 intval("a"):

intval() 看到字符串的第一个字符是 'a'。

因为 'a' 不是数字,也不是正负号,intval() 无法解析,于是直接摆烂,返回了 0。

此时代码相当于变成了:if(0)。

在 PHP 的逻辑判断中,0 代表 false(假)。

因为 if(false) 条件不成立,所以大括号里的 echo $flag; 不会被执行。页面最终会是一片空白,你什么也拿不到。

为什么 ?num\[\]=a (带数组)就可以?

既然单纯的字母不行,为什么加个方括号变成数组就可以呢?

echo intval(array()); // 空数组,输出 0

echo intval(array('a')); // 非空数组,输出 1

第一关: preg_match("/0-9/", $num)。前面提到过,preg_match 只能处理字符串。面对数组,它会直接报错并返回 false。因为返回的是 false,所以不会触发 if 里面的 die(),第一关通过。

第二关: if(intval($num))。根据 PHP 的规则,intval(一个非空数组) 的结果永远等于 1。

代码变成了 if(1)。在 PHP 中,1 代表 true(真),条件成功成立,顺利打印出 flag!

总结

?num=a \rightarrow intval("a") 变成了 0(假) \rightarrow 失败

?num\[\]=a \rightarrow intval(array("a")) 变成了 1(真) \rightarrow 成功

相关推荐
Cloud_Shy6186 分钟前
解读《Effective Python 3rd Edition》:从练气到老魔(第三章 Item 21 - 24)
开发语言·人工智能·笔记·python·迭代器模式
vim怎么退出6 分钟前
Dive into React——Diff 算法
前端·react.js·源码阅读
拾年2758 分钟前
别调 BERT 了:我用 Prompt 做了套 NLP 系统,20 分钟搞定
前端·人工智能
半个落月29 分钟前
别再死记变量提升了——从 V8 编译过程真正理解 JS 执行机制
前端
橘子星38 分钟前
别再懵圈!JS 执行机制的 “千层套路” 全揭秘
前端·javascript
GuWenyue38 分钟前
LeetCode 76 最小覆盖子串|JS 滑动窗口标准解法
前端·算法·面试
YHHLAI40 分钟前
前端 HTTP 请求 & LLM 接口开发
前端·网络协议·http
拾年27542 分钟前
__proto__ vs prototype:90% 的人分不清的 JavaScript 核心
前端·javascript·面试
国科安芯43 分钟前
国科安芯推出商业航天级抗辐照半双工 RS485 收发器 ASC485S2Y
前端·单片机·嵌入式硬件·架构·安全性测试
丑过三八线43 分钟前
Umi 运行时配置 app.tsx 详解
前端