PHP: «Примечание: неопределенная переменная», «Примечание: неопределенный индекс» и «Примечание: неопределенное смещение»

Я запускаю PHP-скрипт и продолжаю получать ошибки, например:

Примечание. Неопределенная переменная: my_variable_name в C: \ wamp \ www \ mypath \ index.php в строке 10

Примечание. Неопределенный индекс: my_index C: \ wamp \ www \ mypath \ index.php в строке 11

Строки 10 и 11 выглядят следующим образом:

echo "My variable value is: " . $my_variable_name; echo "My index value is: " . $my_array["my_index"]; 

Что означают эти ошибки?

Почему они появляются внезапно? Я использовал этот скрипт много лет, и у меня никогда не было проблем.

Что мне нужно сделать, чтобы исправить их?

Это общий справочный вопрос, для которого люди могут ссылаться как дубликаты, а не на необходимость повторять эту проблему снова и снова. Я считаю, что это необходимо, потому что большинство реальных ответов на этот вопрос очень специфичны.

Связанная мета-дискуссия:

  • Что можно сделать по повторяющимся вопросам?
  • Имеют ли смысл «справочные вопросы»?

Примечание. Неопределенная переменная

Из обширной мудрости руководства PHP :

Опора на значение по умолчанию неинициализированной переменной проблематично в случае включения одного файла в другой, который использует одно и то же имя переменной. Это также серьезный риск для безопасности при включении register_globals . Ошибка уровня E_NOTICE выдается в случае работы с неинициализированными переменными, но не в случае добавления элементов к неинициализированному массиву. isset () может быть использована для определения того, была ли уже инициализирована переменная. Кроме того, более идеальным является решение empty (), поскольку оно не генерирует предупреждение или сообщение об ошибке, если переменная не инициализирована.

Из документации PHP :

Предупреждение не генерируется, если переменная не существует. Это означает, что empty () по существу является кратким эквивалентом ! Isset ($ var) || $ var == false .

Это означает, что вы можете использовать только empty() чтобы определить, установлена ​​ли переменная, и, кроме того, она проверяет переменную на следующее: 0,"",null .

Пример:

 $o = []; @$var = ["",0,null,1,2,3,$foo,$o['myIndex']]; array_walk($var, function($v) { echo (!isset($v) || $v == false) ? 'true ' : 'false'; echo ' ' . (empty($v) ? 'true' : 'false'); echo "\n"; }); 

Протестируйте приведенный выше фрагмент в онлайн-редакторе 3v4l.org

Хотя PHP не требует объявления переменной, он рекомендует его, чтобы избежать некоторых уязвимостей или ошибок в системе, где можно было бы забыть присвоить значение переменной, которая будет использоваться позже в скрипте. Что делает PHP в случае необъявленных переменных, возникает ошибка очень низкого уровня, E_NOTICE , которая даже не сообщается по умолчанию, но Руководство советует разрешать во время разработки.

Способы решения проблемы:

  1. Рекомендуется: объявлять переменные, например, при попытке добавить строку к неопределенной переменной. Или используйте isset() / !empty() чтобы проверить, объявлены ли они перед их ссылкой, например:

     //Initializing variable $value = ""; //Initialization value; Examples //"" When you want to append stuff later //0 When you want to add numbers later //isset() $value = isset($_POST['value']) ? $_POST['value'] : ''; //empty() $value = !empty($_POST['value']) ? $_POST['value'] : ''; 

    Это стало намного чище с PHP 7.0, теперь вы можете использовать оператор null coalesce :

     // Null coalesce operator - No need to explicitly initialize the variable. $value = $_POST['value'] ?? ''; 
  2. Установите собственный обработчик ошибок для E_NOTICE и перенаправьте сообщения от стандартного вывода (возможно, в файл журнала):

     set_error_handler('myHandlerForMinorErrors', E_NOTICE | E_STRICT) 
  3. Отключить E_NOTICE от отчетов. Быстрый способ исключить только E_NOTICE :

     error_reporting( error_reporting() & ~E_NOTICE ) 
  4. Подавить ошибку с помощью оператора @ .

Примечание. Настоятельно рекомендуется реализовать только точку 1.

Примечание: Неопределенный индекс / Неопределенное смещение

Это уведомление появляется, когда вы (или PHP) пытаетесь получить доступ к неопределенному индексу массива.

Способы решения проблемы:

  1. Проверьте, существует ли индекс до того, как вы его получите. Для этого вы можете использовать isset() или array_key_exists() :

     //isset() $value = isset($array['my_index']) ? $array['my_index'] : ''; //array_key_exists() $value = array_key_exists('my_index', $array) ? $array['my_index'] : ''; 
  2. Список конструкции языка list() может генерировать это, когда он пытается получить доступ к индексу массива, который не существует:

     list($a, $b) = array(0 => 'a'); //or list($one, $two) = explode(',', 'test string'); 

Для доступа к двум элементам массива используются две переменные, но есть только один элемент массива, индекс 0 , поэтому он будет генерировать:

Примечание: Неопределенное смещение: 1

Переменная $_POST / $_GET / $_SESSION

Замечания выше появляются часто при работе с $_POST , $_GET или $_SESSION . Для $_POST и $_GET вам просто нужно проверить, существует ли индекс или нет, прежде чем использовать их. Для $_SESSION вам нужно убедиться, что сеанс запущен с session_start() и что индекс также существует.

Также обратите внимание, что все 3 переменные являются суперглобалями . Это означает, что они должны быть записаны в верхнем регистре.

Связанный:

  • Примечание. Неопределенная переменная
  • Примечание. Неопределенный индекс

Попробуйте эти

Q1: это уведомление означает, что $ varname не определено в текущей области действия сценария.

Q2: Использование условий isset (), empty () перед использованием любой подозрительной переменной хорошо работает.

 // recommended solution $user_name = $_SESSION['user_name']; if (empty($user_name)) $user_name = ''; OR // just define at the top of the script index.php $user_name = ''; $user_name = $_SESSION['user_name']; OR $user_name = $_SESSION['user_name']; if (!isset($user_name)) $user_name = ''; 

БЫСТРОЕ решение:

 // not the best solution, but works // in your php setting use, it helps hiding site wide notices error_reporting(E_ALL ^ E_NOTICE); 

Примечание о сеансах:

A (часто обескураженный) вариант – это оператор подавления ошибок @ . Это конкретная языковая конструкция, которая закрывает нежелательные уведомления и предупреждения, но должна использоваться с осторожностью.

Во-первых, это требует штрафа за isset за использование isset . Это не поддается измерению в приложениях реального мира, но их следует учитывать при сильных итерациях данных. Во-вторых, это может препятствовать отладке, но в то же время подавленные ошибки фактически передаются настраиваемым обработчикам ошибок (в отличие от выраженных выражений isset).

Это означает, что вы тестируете, оцениваете или печатаете переменную, которой еще не присвоили. Это означает, что у вас либо есть опечатка, либо вам нужно проверить, что переменная была сначала инициализирована. Проверьте логические пути, он может быть установлен в один путь, но не в другом.

Вообще из-за «плохого программирования» и возможности для ошибок сейчас или позже.

  1. Если это ошибка, сначала сделайте правильное назначение переменной: $ varname = 0;
  2. Если он действительно определен только иногда, проверьте его: if (isset ($ varname)) …. перед его использованием
  3. Если это из-за того, что вы неправильно это сделали, просто исправьте это
  4. Возможно даже поворот предупреждений в вас PHP-настройки

Я не хотел отключать уведомление, потому что это полезно, но хотелось избежать слишком большого количества ввода.

Моим решением была эта функция:

 function ifexists($varname) { return(isset($$varname)?$varname:null); } 

Поэтому, если я хочу ссылаться на $ name и echo, если существует, я просто пишу:

 <?=ifexists('name')?> 

Для элементов массива:

 function ifexistsidx($var,$index) { return(isset($var[$index])?$var[$index]:null); } 

На странице, если я хочу обратиться к $ _REQUEST ['name']:

 <?=ifexistsidx($_REQUEST,'name')?> 

Лучший способ получить входную строку :

 $value = filter_input(INPUT_POST, 'value'); 

Этот однострочный слой почти эквивалентен:

 if (!isset($_POST['value'])) { $value = null; } elseif (is_array($_POST['value'])) { $value = false; } else { $value = $_POST['value']; } 

Если вы абсолютно хотите значение строки , как:

 $value = (string)filter_input(INPUT_POST, 'value'); 

Это связано с тем, что переменная 'user_location' не определяется. Если вы используете какой-либо цикл if, внутри которого вы объявляете переменную $ user_location, тогда вы также должны иметь цикл else и определять то же самое. Например:

 $a=10; if($a==5) { $user_location='Paris';} else { } echo $user_location; 

Вышеприведенный код создаст ошибку, поскольку цикл if не выполняется, а в цикле else '$ user_location' не определен. Тем не менее PHP попросили выставить переменную. Поэтому, чтобы изменить код, вы должны сделать следующее:

 $a=10; if($a==5) { $user_location='Paris';} else { $user_location='SOMETHING OR BLANK'; } echo $user_location; 

В ответ на «Почему они появляются внезапно? Я использовал этот сценарий много лет, и у меня никогда не было проблем ».

Для большинства сайтов очень часто работает под сообщением об ошибках «по умолчанию» «Показать все ошибки, но не« уведомления »и« устаревшие ». Это будет установлено в php.ini и применимо ко всем сайтам на сервере. Это означает, что эти «уведомления», используемые в примерах, будут подавлены (скрыты), в то время как другие ошибки, считающиеся более критичными, будут показаны / записаны.

Другая критическая настройка – это ошибки, которые могут быть скрыты (например, display_errors установлен на «off» или «syslog»).

Что произойдет в этом случае, так это то, что либо error_reporting был изменен, чтобы также показывать уведомления (как в примерах), и / или параметры были изменены на display_errors на экране (в отличие от их подавления / регистрации).

Почему они изменились?

Очевидным / простым ответом является то, что кто-то настроил любой из этих параметров в php.ini, или обновленная версия PHP теперь использует другой php.ini из ранее. Это первое место для просмотра.

Однако также можно переопределить эти параметры в

  • .htconf (конфигурация веб-сервера, включая vhosts и подконфигурации) *
  • .htaccess
  • в самом php-коде

и любой из них также мог быть изменен.

Существует также дополнительное усложнение того, что конфигурация веб-сервера может включать / отключать директивы .htaccess, поэтому, если у вас есть директивы в .htaccess, которые внезапно запускают / останавливают работу, вам необходимо это проверить.

(.htconf / .htaccess предположим, что вы работаете как apache.Если запущена командная строка, это не будет применяться, если вы используете IIS или другой веб-сервер, тогда вам нужно будет проверить эти конфиги соответственно)

Резюме

  • Проверьте, что директивы php.ini error_reporting и display_errors php в файле php.ini не изменились или что вы еще не используете другой php.ini.
  • Проверить директивы error_reporting и display_errors php в .htconf (или vhosts и т. Д.) Не изменились
  • Проверить директивы error_reporting и display_errors php в .htaccess не изменились
  • Если у вас есть директива в .htaccess, проверьте, разрешены ли они в файле .htconf
  • Наконец, проверьте свой код; возможно, несвязанная библиотека; чтобы проверить, установлены ли там директивы php error_reporting и display_errors php.

быстрое исправление заключается в том, чтобы назначить вашу переменную нулевым в верхней части кода

 $user_location = null; 

Я проклинал эту ошибку, но может быть полезно напомнить вам, чтобы избежать ввода пользователя.

Например, если вы считаете, что это умный, сокращенный код:

 // Echo whatever the hell this is <?=$_POST['something']?> 

… Подумайте еще раз! Лучшее решение:

 // If this is set, echo a filtered version <?=isset($_POST['something']) ? html($_POST['something']) : ''?> 

(Я использую пользовательскую функцию html() для запуска символов, ваш пробег может отличаться)

Я использую всю свою собственную полезную функцию exst (), которая автоматически объявляет переменные.

Ваш код будет –

 $greeting = "Hello, ".exst($user_name, 'Visitor')." from ".exst($user_location); /** * Function exst() - Checks if the variable has been set * (copy/paste it in any place of your code) * * If the variable is set and not empty returns the variable (no transformation) * If the variable is not set or empty, returns the $default value * * @param mixed $var * @param mixed $default * * @return mixed */ function exst( & $var, $default = "") { $t = ""; if ( !isset($var) || !$var ) { if (isset($default) && $default != "") $t = $default; } else { $t = $var; } if (is_string($t)) $t = trim($t); return $t; } 

На очень простом языке .
Ошибка в том, что вы используете переменную $user_location которая ранее не $user_location вами, и она не имеет никакого значения. Поэтому я рекомендую вам, пожалуйста, объявить эту переменную перед ее использованием , например:


$user_location = '';
Или
$user_location = 'Los Angles';


Это очень распространенная ошибка, с которой вы можете столкнуться. Так что не волнуйтесь, просто объявите переменную и наслаждайтесь кодированием .

В PHP 7.0 теперь можно использовать Null-коалесцирующий оператор:

 echo "My index value is: " . ($my_array["my_index"] ?? ''); 

Равно:

 echo "My index value is: " . (isset($my_array["my_index"]) ? $my_array["my_index"] : ''); 

PHP-руководство PHP 7.0

почему бы не держать вещи простыми?

 <?php error_reporting(E_ALL); // making sure all notices are on function idxVal(&$var, $default = null) { return empty($var) ? $var = $default : $var; } echo idxVal($arr['test']); // returns null without any notice echo idxVal($arr['hey ho'], 'yo'); // returns yo and assigns it to array index, nice ?> 

ПОЧЕМУ ЭТО ПРОИСХОДИТ?

Со временем PHP стал более ориентированным на безопасность языком. Настройки, которые по умолчанию были отключены по умолчанию, теперь включены по умолчанию. Прекрасным примером этого является E_STRICT , который по умолчанию включается с PHP 5.4.0 .

Кроме того, согласно документации PHP, defualt, E_NOTICE отключен в php.ini. PHP-документы рекомендуют включить его для целей отладки . Однако, когда я загружаю PHP из репозитория Ubuntu и из стека Windows BitNami, я вижу что-то еще.

 ; Common Values: ; E_ALL (Show all errors, warnings and notices including coding standards.) ; E_ALL & ~E_NOTICE (Show all errors, except for notices) ; E_ALL & ~E_NOTICE & ~E_STRICT (Show all errors, except for notices and coding standards warnings.) ; E_COMPILE_ERROR|E_RECOVERABLE_ERROR|E_ERROR|E_CORE_ERROR (Show only errors) ; Default Value: E_ALL & ~E_NOTICE & ~E_STRICT & ~E_DEPRECATED ; Development Value: E_ALL ; Production Value: E_ALL & ~E_DEPRECATED & ~E_STRICT ; http://php.net/error-reporting error_reporting = E_ALL & ~E_DEPRECATED & ~E_STRICT 

Обратите внимание, что error_reporting на самом деле задано производственным значением по умолчанию, а не значением по умолчанию. Это несколько запутывает и не документируется вне php.ini, поэтому я не подтвердил это в других дистрибутивах.

Однако, чтобы ответить на ваш вопрос, эта ошибка появляется, когда она не появилась раньше, потому что:

  1. Вы установили PHP, а новые настройки по умолчанию несколько плохо документированы, но не исключают E_NOTICE .

  2. Предупреждения E_NOTICE такие как неопределенные переменные и неопределенные индексы, действительно помогают сделать ваш код более чистым и безопасным. Могу вам сказать, что несколько лет назад E_NOTICE позволило мне объявить мои переменные. Это облегчило изучение C, где не объявление переменных намного больше неприятностей.

ЧТО Я МОГУ СДЕЛАТЬ ЭТО?

  1. Отключите E_NOTICE , скопировав значение «Значение по умолчанию» E_ALL & ~E_NOTICE & ~E_STRICT & ~E_DEPRECATED и заменив его тем, что в настоящее время расколото после знака равенства error_reporting = . Перезапустите Apache или PHP, если вы используете CGI или FPM. Убедитесь, что вы редактируете «правильный» php.ini. Правильным будет Apache, если вы используете PHP с Apache, fpm или php-fpm, если используете PHP-FPM, cgi, если вы используете PHP-CGI и т. Д. Это не рекомендуемый метод, но если у вас есть устаревший код, который будет быть очень трудно редактировать, то это может быть вашим лучшим выбором.

  2. Отключите E_NOTICE на уровне файлов или папок. Это может быть предпочтительнее, если у вас есть какой-то унаследованный код, но в противном случае вы хотите сделать что-то «правильным». Для этого вам следует обратиться к Apache2, nginx или независимо от вашего сервера. В Apache вы должны использовать php_value внутри <Directory> .

  3. Перезапишите свой код, чтобы быть чище. Если вам нужно сделать это при переходе в производственную среду или не хотите, чтобы кто-то видел ваши ошибки, убедитесь, что вы отключите отображение ошибок и только регистрируете свои ошибки (см. log_errors и log_errors в php.ini и настройки сервера).

Чтобы расширить вариант 3: Это идеальный вариант. Если вы можете пойти по этому маршруту, вам нужно. Если вы не собираетесь на этом маршруте изначально, подумайте о том, чтобы переместить этот маршрут в конце концов, протестировав свой код в среде разработки. Пока вы на нем, избавитесь от ~E_STRICT и ~E_DEPRECATED чтобы понять, что может пойти не так в будущем. Вы увидите много незнакомых ошибок, но это остановит вас от каких-либо неприятных проблем, когда вам нужно обновить PHP в будущем.

ЧТО ОЗНАЧАЕТ ОШИБКИ?

Undefined variable: my_variable_name – Это происходит, когда переменная не была определена перед использованием. Когда скрипт PHP выполняется, он внутренне принимает только нулевое значение. Однако, в каком сценарии вам нужно было бы проверить переменную до ее определения? В конечном счете, это аргумент для «неаккуратного кода». Как разработчик, я могу сказать вам, что мне это нравится, когда я вижу проект с открытым исходным кодом, где переменные определены как высоко в своих областях, которые могут быть определены. Это облегчает определение того, какие переменные будут появляться в будущем, и облегчит чтение / изучение кода.

 function foo() { $my_variable_name = ''; //.... if ($my_variable_name) { // perform some logic } } 

Undefined index: my_index – Это происходит, когда вы пытаетесь получить доступ к значению в массиве и его не существует. Чтобы предотвратить эту ошибку, выполните условную проверку.

 // verbose way - generally better if (isset($my_array['my_index'])) { echo "My index value is: " . $my_array['my_index']; } // non-verbose ternary example - I use this sometimes for small rules. $my_index_val = isset($my_array['my_index'])?$my_array['my_index']:'(undefined)'; echo "My index value is: " . $my_index_val; 

Другой вариант – объявить пустой массив в верхней части вашей функции. Это не всегда возможно.

 $my_array = array( 'my_index' => '' ); //... $my_array['my_index'] = 'new string'; 

(дополнительный наконечник)

  • Когда я сталкивался с этими и другими проблемами, я использовал IDE NetBeans (бесплатно), и он дал мне множество предупреждений и уведомлений. Некоторые из них предлагают очень полезные советы. Это не является обязательным требованием, и я больше не использую IDE, кроме крупных проектов. В наши дни я больше vim человек :).

undefined index означает в массиве, который вы запросили для недоступного индекса массива, например

 <?php $newArray[] = {1,2,3,4,5}; print_r($newArray[5]); ?> 

undefined variable означает, что вы использовали полностью не существующую переменную или которая не определена или не инициализирована этим именем, например

 <?php print_r($myvar); ?> 

undefined offset в массиве, который вы запросили для не существующего ключа. И решение для этого – проверить перед использованием

 php> echo array_key_exists(1, $myarray); 

Относительно этой части вопроса:

Почему они появляются внезапно? Я использовал этот скрипт много лет, и у меня никогда не было проблем.

Нет определенных ответов, но вот некоторые возможные объяснения того, почему настройки могут «внезапно» меняться:

  1. Вы обновили PHP до более новой версии, которая может иметь другие значения по умолчанию для error_reporting, display_errors или других соответствующих параметров.

  2. Вы удалили или внесли какой-либо код (возможно, в зависимости), который устанавливает соответствующие параметры во время выполнения, используя ini_set() или error_reporting() (ищите их в коде)

  3. Вы изменили конфигурацию веб-сервера (предполагая здесь apache): файлы .htaccess и конфигурации vhost также могут управлять настройками php.

  4. Обычно уведомления не отображаются / не сообщаются (см. Руководство по PHP ), поэтому возможно, что при настройке сервера файл php.ini не может быть загружен по какой-либо причине (разрешения на файлы?), И вы были в настройках по умолчанию , Позже «ошибка» была решена (случайно), и теперь он может загрузить правильный файл php.ini с набором error_reporting, чтобы показывать уведомления.

Другая причина, по которой будет вызываться неопределенное уведомление индекса, будет заключаться в том, что столбец был исключен из запроса базы данных.

То есть:

 $query = "SELECT col1 FROM table WHERE col_x = ?"; 

Затем попытаемся получить доступ к большему количеству столбцов / строк внутри цикла.

То есть:

 print_r($row['col1']); print_r($row['col2']); // undefined index thrown 

или в цикле while:

 while( $row = fetching_function($query) ) { echo $row['col1']; echo "<br>"; echo $row['col2']; // undefined index thrown echo "<br>"; echo $row['col3']; // undefined index thrown } 

Что-то еще, что нужно отметить, это то, что на ОС * NIX и Mac OS X все зависит от регистра.

Проконсультируйтесь со следующими вопросами и ответами на Stack:

  • Являются ли имена таблиц в регистре MySQL чувствительными к регистру?

  • Имена таблиц, чувствительных к mysql в запросах

  • MySql – Случайная проблема с таблицами на разных серверах

Если вы работаете с классами, вам нужно убедиться, что вы ссылаетесь на переменные-члены, используя $this :

 class Person { protected $firstName; protected $lastName; public function setFullName($first, $last) { // Correct $this->firstName = $first; // Incorrect $lastName = $last; // Incorrect $this->$lastName = $last; } } 

Вероятно, вы использовали старую версию PHP до и теперь обновили PHP, поэтому причина, по которой она работала без каких-либо ошибок, с годами. пока PHP4 не возникнет ошибки, если вы используете переменную без ее определения, но начиная с PHP5 она вызывает ошибки для таких кодов, как упомянутые в вопросе.

При работе с файлами требуется надлежащий enctype и POST-метод, который вызывает неопределенное уведомление индекса, если они не включены в форму.

В руководстве описывается следующий базовый синтаксис:

HTML

 <!-- The data encoding type, enctype, MUST be specified as below --> <form enctype="multipart/form-data" action="__URL__" method="POST"> <!-- MAX_FILE_SIZE must precede the file input field --> <input type="hidden" name="MAX_FILE_SIZE" value="30000" /> <!-- Name of input element determines name in $_FILES array --> Send this file: <input name="userfile" type="file" /> <input type="submit" value="Send File" /> </form> 

PHP

 <?php // In PHP versions earlier than 4.1.0, $HTTP_POST_FILES should be used instead // of $_FILES. $uploaddir = '/var/www/uploads/'; $uploadfile = $uploaddir . basename($_FILES['userfile']['name']); echo '<pre>'; if (move_uploaded_file($_FILES['userfile']['tmp_name'], $uploadfile)) { echo "File is valid, and was successfully uploaded.\n"; } else { echo "Possible file upload attack!\n"; } echo 'Here is some more debugging info:'; print_r($_FILES); print "</pre>"; ?> 

Справка:

если вы хотите остановить некоторое уведомление, используйте @ выше переменной, как

 @$abc 

выше метод не является хорошим, но решение проблем.

Используйте isset, например:

 if(isset($abc)){ }