一、全局变量覆盖
当register_global=ON时,变量来源可能是各个不同的地方,比如页面的表单,Cookie等。

  1. <?php
  2. echo "Register_globals: ".(int)ini_get("register_globals")."<br/>";
  3. if ($auth){
  4. echo "private!";
  5. }
  6. ?>

当register_globals=OFF时,这段代码不会出问题。
但是当register_globals=ON时,提交请求URL:http://www.a.com/test.php?auth=1,变量$auth将自动得到赋值。得到的结果为
Register_globals:1

private!

小记:如果上面的代码中,已经对变量$auth赋了初始值,比如$auth=0,那么即使在URL中有/test.PHP?auth=1,也不会将变量覆盖,也就是说不会打印出private!

通过$GLOBALS获取的变量,也可能导致变量覆盖。

  1. <?php
  2. echo "Register_globals:".(int)ini_get("register_globals")."<br/>";
  3. if (ini_get('register_globals')) foreach($_REQUEST as $k=>$v) unset(${$k});
  4. print $a;
  5. print $_GET[b];
  6. ?>

变量$a未初始化,在register_globals=ON时,再尝试控制“$a”的值(http://www.a.com/test1.php?a=1&b=2),会因为这段代码而出错。
而当尝试注入“GLOBALS[a]”以覆盖全局变量时(http://www.a.com/test1.php?GLOBALS[a]=1&b=2),则可以成功控制变量“$a”的值。这是因为unset()默认只会销毁局部变量,要销毁全局变量必须使用$GLOBALS。
而在register_globals=OFF时,则无法覆盖到全局变量。
小记:register_globals的意思是注册为全局变量,所以当On的时候,传递过来的值会被直接注册为全局变量而直接使用,当为OFF的时候,就需要到特定的数组中去得到它。unset用于释放给定的变量
二、extract()变量覆盖

  1. <?php
  2. $auth = '0';
  3. extract($_GET);
  4. if($auth==1){
  5. echo "private!";
  6. }else{
  7. echo "public!";
  8. }
  9. ?>

假设用户构造以下链接:http://www.a.com/test1.php?auth=1
界面上会打印出private!

安全的做法是确定register_globals=OFF后,在调用extract()时使用EXTR_SKIP保证已有变量不会被覆盖。

小记:PHP extract() 函数从数组中把变量导入到当前的符号表中。对于数组中的每个元素,键名用于变量名,键值用于变量值。

三、遍历初始化变量

常见的一些以遍历的方式释放变量的代码,可能会导致变量覆盖。

  1. <?
  2. $chs = '';
  3. if($_POST && $charset != 'utf-8'){
  4. $chs = new Chinese('UTF-8', $charset);
  5. foreach($_POST as $key => $value){
  6. $$key = $chs->Convert($value);
  7. }
  8. unset($chs);
  9. }
  10. ?>

若提交参数chs,则可覆盖变量"$chs"的值。

小记:在代码审计时需要注意类似“$$k”的变量赋值方式有可能覆盖已有的变量,从而导致一些不可控制的结果。

四、import_request_variables变量覆盖

  1. <?php
  2. $auth = '0';
  3. import_request_variables('G');
  4. if($auth == 1){
  5. echo "private!";
  6. }else{
  7. echo "public!";
  8. }
  9. ?>

当用户输入http://www.a.com/test1.php?auth=1时,网页上会输出private!

import_request_variables('G')指定导入GET请求中的变量,从而导致变量覆盖。

小记:import_request_variables — 将 GET/POST/Cookie 变量导入到全局作用域中。如果你禁止了 register_globals,但又想用到一些全局变量,那么此函数就很有用。

五、parse_str()变量覆盖

  1. //var.php?var=new
  2. $var='init';
  3. parse_str($_SERVER['QUERY_STRING']);
  4. print $var;

与parse_str()类似的函数还有mb_parse_str()

小记:parse_str — 将字符串解析成多个变量,如果参数str是URL传递入的查询字符串(query string),则将它解析为变量并设置到当前作用域。

最新文章

  1. EF Power Tools参数不正确的解决方法
  2. 分布式架构 Hadoop 2.7.X 安装和配置
  3. C#面向对象中类的静态成员与非静态成员的区别
  4. BestCoder Round #73
  5. C# 面向对象之概念理解
  6. (转载)SQL语句,纵列转横列
  7. composer在ubuntu下安装
  8. IIS MIME的 映射 网站有些类型的文件不能通过网页访问
  9. SQL中的join连接查询
  10. 玩转FFmpeg的7个小技巧
  11. Spring Cloud实战的代码和视频位置
  12. 基础设施DevOps演进之路
  13. 2018-04-10 我的GitHub诞生的日子,欢迎大家吐槽批评
  14. linux 一些命令(2)
  15. Javaweb学习笔记——(二)——————CSS概述,进入JavaScript
  16. SharePoint 2010、2013多个域之间互信(Domain Trust)的设计与实施
  17. ios苹果机系统的1px显示解决方案
  18. java-jdk7-forkjoin带有返回值
  19. 【BZOJ4361】isn(动态规划,容斥)
  20. SCWS中文分词,安装说明(以:Win32环境、utf8字符集为例)

热门文章

  1. rpm封装包,只用于记录自己过程,不适合初学者看,请看参考链接
  2. [转载]Js小技巧||给input type=“password”的输入框赋默认值
  3. HDU 5410 CRB and His Birthday ——(完全背包变形)
  4. IOS系列swift语言之课时三
  5. try--catch--finally中return返回值执行的顺序(区别)
  6. error-2016-1-18
  7. [Swift] 疑难杂症
  8. search 搜索
  9. ubuntu pip 安装django报错解决
  10. iOS10 的适配问题,你遇到了吗?导航栏标题和返回按钮神奇的消失了