PHP错误消息“注意:使用未定义的常量”是什么意思?


163

PHP正在日志中记录此错误:“注意:使用未定义的常量”。

日志错误:

PHP Notice:  Use of undefined constant department - assumed 'department' (line 5)
PHP Notice:  Use of undefined constant name - assumed 'name' (line 6)
PHP Notice:  Use of undefined constant email - assumed 'email' (line 7)
PHP Notice:  Use of undefined constant message - assumed 'message' (line 8)

相关代码行:

$department = mysql_real_escape_string($_POST[department]);
$name = mysql_real_escape_string($_POST[name]);
$email = mysql_real_escape_string($_POST[email]);
$message = mysql_real_escape_string($_POST[message]);

这是什么意思,为什么我会看到它?

Answers:


217

您应该引用数组键:

$department = mysql_real_escape_string($_POST['department']);
$name = mysql_real_escape_string($_POST['name']);
$email = mysql_real_escape_string($_POST['email']);
$message = mysql_real_escape_string($_POST['message']);

正如,有人找称为常量departmentnameemailmessage,等。当它没有找到这样的常量,PHP(奇怪)解释为一个字符串(“部门”等)。显然,如果以后再定义这样的常量,这很容易破坏(尽管使用小写常量是不好的样式)。


它是指$ _POST变量吗?
尼克,

3
@Col。弹片,我什么时候说需要引用数组键?他确实需要引用密钥,而不仅仅是引用。
马修·弗拉申

1
好吧,我的意思是不需要引用数组键。一个必须引用字符串,而不是数组键。密钥不需要特殊的引号。
您的常识

4
不是“ bizarre” ...是“向后兼容”。PHP最初允许甚至使用未加引号的字符串作为键进行升级。(好吧,也许它仍然是“ bizarre”。:-)
Brian White

1
@BrianWhite有趣的事实是,在提议弃用此功能时,我找不到任何证据表明它曾受到正式鼓励,并且仅PHP 3.0的beta版本包含了该行为,恕不另行通知,因此它似乎向后兼容该功能。从未发布。无论如何,只要它出现,它将在PHP 8.0中消失。
IMSoP

77

该错误消息是由于不幸的事实,即PHP将隐式声明一个未知令牌作为具有相同名称的常量字符串。

也就是说,它正在尝试对此进行解释(请注意缺少引号):

$_POST[department]

在PHP中这将是有效语法的唯一有效方法是,如果先前已department定义了常量。不幸的是,它并没有死于致命错误,而是发出了该通知,并且就像一个常量已经用相同的名称和值定义:

// Implicit declaration of constant called department with value 'department'
define('department', 'department');  

您可以通过多种方式获得此错误消息,但是它们都有相同的根本原因-一个可以是常量。

字符串缺少引号: $my_array[bad_key]

这就是您遇到的问题,这是因为您有未引用的字符串数组键。修复字符串键将修复该错误:

更改:

$department = mysql_real_escape_string($_POST[department]);
...(etc)...

至:

$department = mysql_real_escape_string($_POST['department']);
...(etc)...

可变的缺少美元符号: var_without_dollar

您可能会看到此错误消息的另一个原因是,您是$从变量还是$this->从成员中删除了。例如,以下任何一种都会导致类似的错误消息:

my_local;   // should be $my_local
my_member;  // should be $this->my_member

变量名称中的字符无效: $bad-variable-name

如果尝试在变量名中使用不允许的字符,则会导致类似但更微妙的问题- 常见的情况是连字符(-)而不是下划线_

例如,这样就可以了,因为变量名中可以使用下划线

if (123 === $my_var) {
  do_something();
}

但这不是:

if (123 === $my-var) {
  do_something();
}

它将被解释为:

if (123 === $my - var) {  // variable $my minus constant 'var'
  do_something();
}

在不指定类范围的情况下引用类常量

为了引用一个类常量,您需要用它指定类的范围::,如果您错过了这个范围,PHP会认为您在谈论全局变量define()

例如:

class MyClass {
  const MY_CONST = 123;

  public function my_method() {
    return self::MY_CONST;  // This is fine
  }


  public function my_method() {
    return MyClass::MY_CONST;  // This is fine
  }

  public function my_bad_method() {
    return MY_CONST;  // BUG - need to specify class scope
  }
}

使用此版本的PHP中未定义的常量,或未安装的扩展中定义的常量

有一些系统定义的常量只存在于PHP的较新版本,例如,模式选择常数round()PHP_ROUND_HALF_DOWN只存在于PHP 5.3或更高版本。

因此,如果您尝试在PHP 5.2中使用此功能,请说:

$rounded = round($my_var, 0, PHP_ROUND_HALF_DOWN);

您会收到以下错误消息:

使用未定义的常量PHP_ROUND_HALF_DOWN-假定为'PHP_ROUND_HALF_DOWN'警告(2):round()的参数计数错误


感谢分享。我错过了self::部分。现在正在工作。
moreirapontocom

毫无疑问,该答案比其他答案更全面,值得您最好的答案!我也太想念班级范围,最终无法获得一些有价值的信息。谢谢,约翰!
哈里斯ST

6

您可能忘记了使用""

举个例子:

$_array[text] = $_var;

改成:

$_array["text"] = $_var;



1
<?php 
  ${test}="test information";
  echo $test;
?>

注意:使用未定义的常数测试-在第3行测试信息的D:\ xampp \ htdocs \ sp \ test \ envoirnmentVariables.php中假定为“ test”


1

插入单引号。

$department = mysql_real_escape_string($_POST['department']);
$name = mysql_real_escape_string($_POST['name']);
$email = mysql_real_escape_string($_POST['email']);
$message = mysql_real_escape_string($_POST['message']); 

0

我不确定使用代码点火器是否存在任何区别,我使用“”作为名称,效果很好。

$department = mysql_real_escape_string($_POST["department"]);
$name = mysql_real_escape_string($_POST["name"]);
$email = mysql_real_escape_string($_POST["email"]);
$message = mysql_real_escape_string($_POST["message"]);

问候,

乔治。


1
在这种情况下,单引号和双引号之间没有区别。
罗比·阿弗里尔

1
@RobbieAverill实际上没有,因为单引号字符串按原样显示,双引号字符串将解析转义字符并评估变量,因此在这种情况下使用单引号字符串是一种很好的样式。
弗兰基(Frankie)2015年

0

看起来预定义的获取常量随MySQL扩展一起消失了,所以我们需要在第一个函数之前添加它们。

//预定义的提取常量

define('MYSQL_BOTH',MYSQLI_BOTH);
define('MYSQL_NUM',MYSQLI_NUM);
define('MYSQL_ASSOC',MYSQLI_ASSOC);

我测试并成功。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.