深刻理解is_callable和method_exists

1、函數解析php

is_callable()數組

定義:函數

 

(PHP 4 >= 4.0.6, PHP 5, PHP 7)測試

 

is_callable — 檢測參數是否爲合法的可調用結構編碼

bool is_callable ( callable $name [, bool $syntax_only = false [, string &$callable_name ]] )

驗證變量的內容可否做爲函數調用。 這能夠檢查包含有效函數名的變量,或者一個數組,包含了正確編碼的對象以及函數名。spa

參數:.net

namecode

要檢查的回調函數。對象

syntax_onlyblog

若是設置爲 TRUE,這個函數僅僅驗證 name 多是函數或方法。 它僅僅拒絕非字符,或者未包含能用於回調函數的有效結構。有效的應該包含兩個元素,第一個是一個對象或者字符,第二個元素是個字符。

callable_name

接受「可調用的名稱」。下面的例子是「someClass::someMethod」。 注意,儘管 someClass::SomeMethod() 的含義是可調用的靜態方法,但例子的狀況並非這樣的。

返回值:

若是 name 可調用則返回 TRUE,不然返回 FALSE

參考文獻: http://php.net/manual/zh/function.is-callable.php

 

2、函數測試

測試一:

echo '<pre >';
$func = function ($a)
{
    echo $a;
};
$re = is_callable($func, true, $callable_name1);
echo '<hr />';
$re1 = is_callable($func, false, $callable_name2);

//結果
bool(true)
string(17) "Closure::__invoke"
-------------------------------------
bool(true)
string(17) "Closure::__invoke"

測試結果:

對於匿名函數,傳入函數變量後,參數二syntax_only true 和 false,打印結果是同樣的。

測試二:

 

function c_b($d)
{
    echo $d;
}
$re = is_callable('c_b', false, $callable_name1);
$re1 = is_callable('c_b', true, $callable_name2);
var_dump($re);
echo '<hr />';
var_dump($re1);
echo '<hr />';
var_dump($callable_name1);
echo '<hr />';
var_dump($callable_name2);
//結果
bool(true)
----------------
bool(true)
----------------
string(3) "c_b"
----------------
string(3) "c_b"

 

測試結果:

對於通常函數,傳入函數名稱後,參數二syntax_only true 和 false,打印結果是同樣的。

測試三:

class Person
{
    public static function get($a)
    {
        echo $a;
    }

    protected function _set()
    {
        echo 1;
    }
}
$p = new Person();

$re = is_callable([$p, 'get'], false, $callable_name1);
$re1 = is_callable([$p, 'get'], true, $callable_name2);
var_dump($re);
echo '<hr />';
var_dump($re1);
echo '<hr />';
var_dump($callable_name1);
echo '<hr />';
var_dump($callable_name2);
//結果
bool(true)
-----------------
bool(true)
-----------------
string(11) "Person::get"
---------------------------
string(11) "Person::get"

測試結果:

對於類的方法,參數以數組結構(類對象或類名稱 + 方法名稱),參數二syntax_only true 和 false,打印結果也是同樣的。

測試四:

$a = 'i am string';
$re = is_callable($a, false, $callable_name1);
$re1 = is_callable($a, true, $callable_name2);
var_dump($re);
echo '<hr />';
var_dump($re1);
echo '<hr />';
var_dump($callable_name1);
echo '<hr />';
var_dump($callable_name2);
//結果
bool(false)
----------------
bool(true)
----------------
string(11) "i am string"
-------------------------
string(11) "i am string"

測試結果:

對於傳入的驗證name,若是syntax_only 設置爲true,它驗證傳入name是不是字符串,是否含有非法字符,若是不含有,則返回true,它並不會驗證name是否爲合法調用結構。

測試五:

 

$re = is_callable(['Class', 'Method'], false, $callable_name1);
$re1 = is_callable(['Class', 'Method'], true, $callable_name2);
var_dump($re);
echo '<hr />';
var_dump($re1);
echo '<hr />';
var_dump($callable_name1);
echo '<hr />';
var_dump($callable_name2);
//結果
bool(false)
--------------
bool(true)
--------------
string(13) "Class::Method"
-----------------------------
string(13) "Class::Method"
    

 

測試結果:

對於傳入的驗證name,若是syntax_only 設置爲true,它只驗證傳入name是不是字符串,是否含有非法字符或是否爲數組參數字符串1 + 字符串二,若是符合條件,則返回true,它並不會驗證name是否爲合法調用結構。否者返回false;

測試六:

class Person
{
    public static function get($a)
    {
        echo $a;
    }

    protected function _set()
    {
        echo 1;
    }
}
$p = new Person();
$re = is_callable([$p, '_set'], false);
var_dump($re);
echo '<hr />';
$re1 = method_exists($p, '_set');
var_dump($re1);
//結果
bool(false)
------------
bool(true)

測試結果:

對於函數is_callable() 來講,若是驗證的類方法,訪問修飾符爲protected或private 則返回false

對於method_exists() 來講,則不受訪問修飾符的影響,只要類方法存在,則返回true。

 

 3、總結、

一、is_callable() 函數,可傳入的name類型有:函數字符串,匿名函數變量,類或類對象和方法名稱組成的數組。其函數第二參數,若是是true,則只驗證name是不是字符串或則是類或字符串1(類對象)和字符串二(方法名稱)組成的數組。而不驗證name是否爲合法調用結構。若是是false,則驗證name是否爲合法調用結構。

二、method_exists() 函數,不受訪問修飾符的影響,只要類方法存在,則返回true。函數is_callable()來講,若是驗證的類方法,訪問修飾符爲protected或private 則返回false。

相關文章
相關標籤/搜索