Будьте осторожны с PHP empty при ее использовании в шорткодах Wordpress

Очень часто на веб-ресурсах по изучению и разработке плагинов для Wordpress можно встретить примеры, где предлагают использовать PHP empty для проверки атрибутов шорткодов. Но давайте разберемся, как работает эта функция и какие ошибки могут возникать при неправильном ее использовании.

Особенности работы функции empty(), о которых нужно знать

Согласно мануалу на php.net, эта функция проверяет пуста ли переменная.

Если переменная не существует, то empty() не выдаст ошибку. Например, пусть переменная $foo не установлена:

if( empty($foo) ){ echo "variable is empty"; } //выведет "variable is empty"

Следовательно нужна дополнительная проверка функцией isset():

$foo = 1;
if( isset($foo) && !empty($foo) ){ echo "variable = ".$foo; } // выведет "variable = 1"

Значения, которые PHP empty() считает пустыми, это:

  • "" (строки - strlen('') == 0 );
  • 0 (целые числа - (int)0 );
  • 0.0 (числа с плавающей точкой - (float)0.0 );
  • "0" (строки - strlen("0") == 1 );
  • NULL;
  • FALSE;
  • array() ( пустые массивы - count( array( ) ) == 0 ).
$string_1 = '';
echo strlen($string_1); //выведет 0
if( empty( $string_1 ) ){ echo 'string_1 is empty'; } // выведет "string_1 is empty"

$string_2 = '0';
echo strlen($string_2); //выведет 1
if( empty( $string_2 ) ){ echo 'string_2 is empty'; } // выведет "string_2 is empty"

Строку ($string_2) длиной в один символ со строчным нулем ('0') функция empty() тоже считает пустой.

Но строку с одним пробелом не считает пустой:

$string_3 = ' ';
echo strlen($string_3); //выведет 1
if( empty( $string_3 ) ){ echo 'string_3 is empty'; } // ничего не выведет

Давайте рассмотрим практическое применение PHP empty и увидим, как эта функция может привести к неожиданному результату.

Применение функции empty() для проверки атрибутов шорткода в плагинах Wordpress

Например, вы разрабатываете плагин для Wordpress, в котором с помощью шорткодов будут выводится статьи разных авторов. На странице аккаунта каждого автора будут показаны его собственные статьи ("self") и статьи других авторов ("another") за последний месяц. С помощью шорткода можно управлять количеством вывода статей. По умолчанию, если атрибуты "self" и "another" не установлены, будем выводить, например, 25 собственных постов и 15 постов других авторов.

[author_posts]

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

[author_posts self="10" another="5" ]

Посмотрим, что в коде плагина:

<?php
/**
 * @package author_posts
 */
/*
 * Plugin Name: Author posts
 * Description: Author posts
 * Version: 1.0.0
 * Author: V.Taran
 */
function posts_shortcode( $atts ) {

	$atts = shortcode_atts(
		array(
			'self' => '',
			'another' => '',
		),
		$atts,
		'author_posts'
	);
	if(!empty($atts['self'])){ 
		echo $atts['self']." post(s)"; 
	}else{ 
		echo "25 self posts"; }
	if(!empty($atts['another'])){ 
		echo $atts['another']." post(s)"; 
	}else{ 
		echo "15 posts of another authors";
	}
}
add_shortcode( 'author_posts', 'posts_shortcode' );

Здесь мы просто выводим на печать данные, которые пользователь передал через шорткод. В результате мы увидим 10 собственных постов и 5 - других авторов.

Если мы не хотим видеть, например, посты других авторов, то напишем в шорткод "another=0", и посты других авторов отображаться не будут. Но нет! Функция PHP empty() считает строчный "0" как пустое значение, и вместо "0" постов других авторов, отобразит значение по умолчанию, т.е. 15 записей вместо нуля.

Иными словами, эти две записи аналогичны:

[author_posts another="0" ]
[author_posts ]

Чтобы шорткод работал правильно, изменим условие:

$is_null = (int)(-1);

'another' => $is_null,

if($atts['another'] > -1 ){ 
	echo $atts['another']." post(s)"; 
}else{ 
	echo "15 posts of another authors";
}

Если атрибут "another" не задан, то по умолчанию пусть он будет равен (-1). Тогда в условии будем считать, что атрибут задан, если он больше (-1).

Таким образом, этот шорткод:

[author_posts another="0" ]

теперь все правильно нам отображает, т.е. по умолчанию 25 статей автора (т.к. атрибут "self" не указан) и ни одной статьи других авторов.

Весь код:

<?php
/**
 * @package author_posts
 */
/*
 * Plugin Name: Author posts
 * Description: Author posts
 * Version: 1.0.0
 * Author: V.Taran
 */
function posts_shortcode( $atts ) {
	
	$is_null = (int)(-1);
	
	$atts = shortcode_atts(
		array(
			'self' => '',
			'another' => $is_null,
		),
		$atts,
		'author_posts'
	);
	if(!empty($atts['self'])){ 
		echo $atts['self']." post(s)<br>"; 
	}else{ 
		echo "25 self posts<br>"; }
	if($atts['another'] > -1 ){ 
		echo $atts['another']." post(s)<br>"; 
	}else{ 
		echo "15 posts of another authors<br>";
	}
}
add_shortcode( 'author_posts', 'posts_shortcode' );

Вы можете придумать свое условие, важно, что вы увидели особенности работы функции empty() и в дальнейшем не допустите ошибку при ее использовании. Поместите этот файл в папку Wordpress "/wp_content/plugins/" и активируйте плагин в панели администратора, он будет работать.

Варианты замены empty() на другие функции PHP

Чтобы понять, чем заменить empty(), нужно знать, что мы имеем, например:

  • если это массив (array()), то лучше проверять его функциями count или sizeof;
  • если переменная может быть равна false, то используйте if( false == $var );
  • если переменная не установлена, то с помощью isset или is_null();
  • числам подойдет is_numeric().

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

Мы выяснили, что empty PHP - функция, которая проверяет пуста ли переменная. Нужно помнить, какие значения переменной определяются как нулевые, если используете ее в своем коде, чтобы потом не искать, почему программа работает неправильно.

Комментарии