php回調函數用this報錯
A. PHP 5.3 回調函數裡面怎麼使用this
$a = new myclass();
$a->fun();
在這里,你知道是用 $a-> 來調用方法,可以理解吧。。
class myclass {
function fun() {
echo 'myfun';
}
function fun1() {
//$a->fun(); 理論上是這專樣,但是,你怎麼知屬道是 $a 呢?不知道
$this->fun(); // this 是自己的意思。因為不知道你實例化之後的變數名。
}
}
B. PHP中的回調函數是怎麼實現的
第一種, 常規的php全局函數
function test($echo)
{
echo $echo, "\n";
}
$param = array("www.zeroplace.cn");
call_user_func_array(test, $param);
call_user_func_array("test", $param);
輸出:
www.zeroplace.cn www.zeroplace.cn
兩種調用方式都可以成功調用, 說明call_user_func_array調用常規php局函數時的第一個參數可以為函數本身,也可以為表示函數名的字元串。
第二種, 類的靜態方法
class TestC
{
static function test($echo)
{
echo $echo, "\n";
}
}
$param = array("www.zeroplace.cn");
call_user_func_array(array(TestC, "test"), $param);
call_user_func_array(array("TestC", "test"), $param);
輸出:
www.zeroplace.cn www.zeroplace.cn
這時傳第一個參數的時候傳了一個數據。數組的第一個元素可以為類本身,也可以為類名的一個字元串。第二個元素則是一個表示方法名的字元串。
第三種,對象的方法
class TestC
{
protected $_a = "hello word";
function test($echo)
{
$this->_a = $echo;
}
function show()
{
echo $this->_a, "\n";
}
}
$param = array("www.zeroplace.cn");
$obj = new TestC();
call_user_func_array(array($obj, "test"), $param);
$obj->show();
$obj->test("WEB應用開發");
$obj->show();
輸出:
www.zeroplace.cn WEB應用開發
這里是調用一個對象的方法。從test方法中設置對象的_a成員,然後通過show方法將它顯示出來,以驗證call_user_func_array調用方法時可以正確等到$this指針。這里輸出的結果表明已經上面的方法是奏效的。其實回調對象的方法還有一種更加簡單的方法,直接$obj->{$method}()就可以調用,只要{$method}是存在的。
C. PHP回調函數的幾種用法
具體代碼如下:
<?php
$ch = curl_init();
$timeout = 5;
curl_setopt ($ch, CURLOPT_URL, '');
curl_setopt ($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt ($ch, CURLOPT_CONNECTTIMEOUT, $timeout);
$file_contents = curl_exec($ch);
curl_close($ch);
echo $file_contents;
?>
PHP 獨特的語法混合了C、Java、Perl以及PHP自創的語法。
它可以比CGI或者回Perl更快速地執行動態網頁。答用PHP做出的動態頁面與其他的編程語言相比,PHP是將程序嵌入到HTML(標准通用標記語言下的一個應用)文檔中去執行,
執行效率比完全生成HTML標記的CGI要高許多;
PHP還可以執行編譯後代碼,編譯可以達到加密和優化代碼運行,使代碼運行更快。
D. 關於php回調函數的問題。下面是看到的一個例子。沒看懂回調函數到底有什麼作用。
可以根據你的函數需要去確定你需要調用的函數回的名稱。答
<?php
$a=array('hello','world');
foreach($aas$value){
call_user_func_array($value,$params);
}
?>
E. PHP 5.3 回調函數裡面怎麼使用this
$a = new myclass();
$a->fun();
在這里,你知道是用 $a-> 來調用方法,可以理解吧。。
class myclass {
function fun() {
echo 'myfun';
}
function fun1() {
//$a->fun(); 理論上是這樣,但是,你怎麼專知道是 $a 呢?不知道
$this->fun(); // this 是自屬己的意思。因為不知道你實例化之後的變數名。
}
}
更多問題,歡迎去php中文網問答社區,大神在線幫你解答。希望對你有幫助
F. 關於PHP中函數的回調函數
有影響的。這個函數的回調函數的作用不是比較值的大小,而是在比較值相同之後,比較key是否相等
比如compare如果這樣寫:
function compare($a, $b) {
return 0;
}
表示認為無論什麼鍵都相等,返回的結果是
Array
(
[b] => brown
[0] => yellow
[2] => red
)
-----
function compare($a, $b) {
return 1;
}
認為無論什麼樣的鍵都不相等,返回的結果是空數組array();
---------------------------
這個就涉及他內部的演算法了因為涉及混合數組的演算法太復雜,用一個簡單的例子舉例:
<?php
$arr1 = array('a'=>1, 'b'=>2, 'c'=>3);
$arr2 = array('a'=>1, 'B'=>2, 'C'=>3);
$arr3 = array('a'=>1, 'b'=>2, 'c'=>3);
$arr4 = array('A'=>1, 'b'=>2, 'C'=>3);
function compare($a, $b) {
if($a === $b) {
return 0;
} else
return 1;
}
print_r(array_intersect_uassoc($arr1,$arr2, 'compare'));
print_r(array_intersect_uassoc($arr3,$arr4, 'compare'));
?>
會發現,第一次比較會返回 a=>1 第二次比較會是空
這個特殊的現象跟他內部演算法有關,這個函數會做三件事:
對$arr1調用回調函數從小到大冒泡排序
對$arr2也排序
然後比較$arr1的第一個鍵和$arr2的第一個鍵,如果相等,比較值,如果不等,比較$arr1的第一個鍵和$arr2的第二個鍵,一直到循環結束。
對$arr1和$arr2最開始比較的是 a, a 調用compare(a,a) 返回是0, 所以認為兩個鍵值相等。
對$arr3和$arr4,調用 compare(a,A) 不等, 調用compare(a, b)不等,調用compare(a, C)不等, 就退出循環了, 所以返回空
你問的這個問題也是這個原因
寫成
function compare($a, $b) {
if($a > $b)
return 1;
else if($a < $b)
return -1;
else return 0;
}
就會發現yellow了
G. PHP回調函數到底是個啥
當做一個變數去理解比較容易,比如$a=function(){echo "aaa";};$aa();或者function a(){};將函數名作為一內個變數去賦值;容$b="a"; $b();一個是匿名函數作為變數去調用,另一個是函數名作為變數去調用,這就是回調函數常用的方式,簡單來說,就這把這樣調用函數的方式叫做回調函數,個人經驗分享而已,O(∩_∩)O哈哈~
H. php回調函數如何傳入第二個參數
我幫你查詢了SaeFetchurl的文檔,其內部實現應該無法滿足你的需求。
但可以使用變通的方法:使用類專方法作為回調函數屬,在構造類時傳遞$pid參數
<?php
classCtx{
publicfunction__construct($pid){
$this->pid=$pid;
}
publicfunctiondemo($content){
//...dosomethingwith$this->pidand$content
}
}
$furl=newSaeFetchurl();
$pid=5780859;
$furl->fetch($url,$opt,array(newCtx($pid),'demo'));
I. php 框架 怎麼使用回調函數
前言
最近在開發一個PHP系統,為了提高系統的擴展性,我想在系統中加入類似Javascript的事件處理機制,例如:我想在一篇新聞被添加以後,我想記錄一下日誌,用類似Javascript的代碼,應該是這樣寫的:
function fnCallBack( $news )
{
//將$news的信息記錄到日誌中
writeLog( $news->getTitle().' has been added successfully!');
}
$newsEventManager->addEventListener( 'add' , fnCallBack );
其中,fnCallBack函數是回調函數,addEventListener表示監聽newsEventManager的add事件。當一篇news被add以後,系統就會調用fnCallBack函數,從而完成writeLog的動作。
但是,PHP中的函數傳遞方法和Javascript有很大的不同。在Javascript中,函數也是對象,它可以很方便的當作參數傳遞,但是PHP不行。
$newsEventManager->addEventListener( 'add' , fnCallBack );
上面這行代碼中的fnCallBack,看上去好像是那個函數的句柄,但實質上它是一個字元串,並不是我們所要的函數。
為了實現我們的事件模型,有必要研究一下PHP的回調函數的實現方法。
全局函數的回調
這里的全局函數的意思,是直接使用function定義的函數,它不包含在任何對象或類之中。請看下面的例子
示例代碼
function fnCallBack( $msg1 , $msg2 )
{
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
call_user_func_array( $fnName , $params );
代碼說明:
這里使用了PHP內置的函數call_user_func_array來進行調用。call_user_func_array有兩個參數,第1個參數是一個字元串,表示要調用的函數名,第2個參數是一個數組,表示參數列表,按照順序依次會傳遞給要調用的函數。
效果如下:
類的靜態方法的回調
如果我們要回調的方法,是一個類的靜態方法,那怎麼辦呢?我們依然可以利用PHP內置的call_user_func_array方法來進行調用,請看示例:
示例代碼:
class MyClass
{
public static function fnCallBack( $msg1 , $msg2 )
{
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
}
$className = 'MyClass';
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
call_user_func_array( array( $className , $fnName ) , $params );
代碼說明:
這段代碼和第1種方法的代碼很相似,我們將類名(MyClass)也作為call_user_func_array的第1個參數傳遞進去,就可以實現類的靜態方法的回調了。注意,這時call_user_func_array的第1個參數是一個數組了,數組的第1個元素是類名,第二個元素是要調用的函數名
運行結果:
(其實和第1種方法的結果是一樣的 ^_^ )
繼續研究
如果我用這種方法調用一個類的非靜態方法(也就是把static去掉),會出現什麼結果呢?請看下面代碼
class MyClass
{
public function fnCallBack( $msg1 , $msg2 )
{
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
}
$className = 'MyClass';
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
call_user_func_array( array( $className , $fnName ) , $params );
運行結果
和前面的結果還是一樣的。。。
現在我為這個類添加一點屬性,並在方法中引用
class MyClass
{
private $name = 'abc';
public function fnCallBack( $msg1 , $msg2 )
{
echo 'object name:'.$this->name;
echo "<br />\n";
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
}
$className = 'MyClass';
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
call_user_func_array( array( $className , $fnName ) , $params );
運行結果
出現解析錯誤,提示$this沒有在對象環境下出現,說明這個方法不能用類來調用,而是要用對象來調用。那我們就修改一下代碼,創建一個對象:
class MyClass
{
public function fnCallBack( $msg1 , $msg2 )
{
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
}
$myobj = new MyClass();
$className = 'myobj';
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
call_user_func_array( array( $className , $fnName ) , $params );
運行結果:
提示call_user_func_array的第1個參數非法,也就是說,調用失敗。看來我們不能用call_user_func_array方法來回調一個對象的方法了,那麼如何實現對象方法的回調的?
對象的方法的回調
我先用最原始的字元串形式的調用方法嘗試了一下,如下所示:
class MyClass
{
private $name = 'abc';
public function fnCallBack( $msg1 = 'default msg1' , $msg2 = 'default msg2' )
{
echo 'object name:'.$this->name;
echo "<br />\n";
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
}
$myobj = new MyClass();
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
$myobj->$fnName();
成功了,輸出結果
調用是成功了,不過如何把參數params傳給這個方法呢,如果把params直接傳進去,那麼它會作為1個參數,怎麼把params拆開來傳進去呢?
查了下PHP手冊,找到了create_function函數,這個方法可以用字元串來創建一個匿名函數,好,有思路了,可以創建一個匿名的函數,在這個匿名函數中,調用我們的回調函數,並把參數傳進去。
我先手動創建一個匿名函數anonymous,在這個函數中,用前面試出來的方法調用回調函數,如下所示:
class MyClass
{
private $name = 'abc';
public function fnCallBack( $msg1 = 'default msg1' , $msg2 = 'default msg2' )
{
echo 'object name:'.$this->name;
echo "<br />\n";
echo 'msg1:'.$msg1;
echo "<br />\n";
echo 'msg2:'.$msg2;
}
}
$myobj = new MyClass();
$fnName = "fnCallBack";
$params = array( 'hello' , 'world' );
function anonymous()
{
global $myobj;
global $fnName;
global $params;
$myobj->$fnName( $params[0] , $params[1] );
}
anonymous();
成功了,可以看到,對象的屬性name也輸出來了
然後,我用create_function來創建這個匿名函數,同時,代碼中的params[0],params[1]應該是動態生成的,代碼如下:
$strParams = '';
$strCode = 'global $myobj;global $fnName;global $params;$myobj->$fnName(';
for ( $i = 0 ; $i < count( $params ) ; $i ++ )
{
$strParams .= ( '$params['.$i.']' );
if ( $i != count( $params )-1 )
{
$strParams .= ',';
}
}
$strCode = $strCode.$strParams.");";
$anonymous = create_function( '' , $strCode);
$anonymous();
這段代碼可以定義一個匿名函數,並保存在$anonymous變數中,最後調用這個$anonymous,實現了方法的回調,如圖
PHP事件模型(觀察者模式)的實現思路
至此,PHP中的3種常見的函數類型(全局函數,類靜態函數,對象的方法)都可以回調了,可以實現文章一開始說的事件模型了 :)
事件模型模仿Firefox的Javascript實現,有3個方法,分別是
addEventListener:注冊一個事件上的響應回調函數
removeEventListener:刪除一個事件上的響應回調函數
fire:觸發一個事件,也就是循環調用所有響應這個事件的回調函數
不過,由於第2、第3種方法需要傳遞上下文(也就是類名和對象名),所以addEventListener和removeEventListener應該有3個參數,我是這樣設計的:
function addEventListener( $evtName , $handler , $scope = null )
第1個參數表示事件名,字元串類型
第2個參數表示回調函數名,字元串類型
第3個參數$scope是上下文環境,一共有3種類型,null表示傳入的handler函數是一個全局函數,字元串類型表示傳入的handler函數是scope類的靜態函數,對象類型表示傳入的scope是一個對象,handler函數是對象的一個方法。
function fire( $evtName , $params = null )
這個方法內,會讀取出所有響應evtName的handler,然後判斷它對應的scope,如果是null,則用本文第1種方法回調,如果是字元串,則用本文第2種方法回調,如果是對象,則用本文第3種方法回調。這樣,一個PHP的事件模型就可以實現了,而且可以將回調函數放在某個對象中。
J. php中的回調函數
回調函數就是一個通過函數指針調用的函數。如果你把函數的指針(地址)作內為參數傳遞給另一個函數容,當這個指針被用來調用其所指向的函數時,我們就說這是回調函數。回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用於對該事件或條件進行響應。
php提供了兩個內置函數call_user_func()和call_user_func_array()提供對回調函數的支持。這兩個函數的區別是call_user_func_array是以數組的形式接收回調函數的參數的,看它的原型就知道了:mixed call_user_func_array ( callable $callback,array$param_arr ),它只有兩個參數。而call_user_func($callback,參數1,參數2,…)的參數個數根據回調函數的參數來確定的。