Ⅰ 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'));
Ⅱ 如何向回調函數中傳入其他參數
成員函數名不能作為參數傳遞,否則會發生錯誤,因為裡面隱藏著一個this 指針
回調函數,就是由你自己寫的。你需要調用另外一個函數,而這個函數的其中一個參數,就
是你的這個回調函數名。這樣,系統在必要的時候,就會調用你寫的回調函數,這樣你就可
以在回調函數里完成你要做的事。
模塊A有一個函數foo,它向模塊B傳遞foo的地址,然後在B裡面發生某種事件(event)時,通過從A裡面傳遞過來的foo的地址調用foo,通知A發生了什麼事情,讓A作出相應反應。 那麼我們就把foo稱為回調函數。
例子:
回調函數是一個很有用,也很重要的概念。當發生某種事件時,系統或其他函數將會自動調用你定義的一段函數。回調函數在windows編程使用的場合很多,比如Hook回調函數:MouseProc,GetMsgProc以及EnumWindows,DrawState的回調函數等等,還有很多系統級的回調過程。本文不準備介紹這些函數和過程,而是談談實現自己的回調函數的一些經驗。
之所以產生使用回調函數這個想法,是因為現在使用VC和Delphi混合編程,用VC寫的一個DLL程序進行一些時間比較長的非同步工作,工作完成之後,需要通知使用DLL的應用程序:某些事件已經完成,請處理事件的後續部分。開始想過使用同步對象,文件影射,消息等實現DLL函數到應用程序的通知,後來突然想到可不可以在應用程序端先寫一個函數,等需要處理後續事宜的時候,在DLL里直接調用這個函數即可。
於是就動手,寫了個回調函數的原形。在VC和 Delphi里都進行了測試
一:聲明回調函數類型。
vc版
typedef int (WINAPI *PFCALLBACK)(int Param1,int Param2) ;
二:聲明回調函數原形
聲明函數原形
vc版
int WINAPI CBFunc(int Param1,int Param2);
三: 回調函數調用調用者
調用回調函數的函數我把它放到了DLL里,這是一個很簡單的VC生成的WIN32 DLL.並使用DEF文件輸出其函數名 TestCallBack。實現如下:
PFCALLBACK gCallBack=0;
void WINAPI TestCallBack(PFCALLBACK Func)
{
if(Func==NULL)return;
gCallBack=Func;
DWORD ThreadID=0;
HANDLE hThread = CreateThread( NULL, NULL, Thread1, LPVOID(0), &ThreadID );
return;
}
此函數的工作把傳入的 PFCALLBACK Func參數保存起來等待使用,並且啟動一個線程。聲明了一個函數指針PFCALLBACK gCallBack保存傳入的函數地址。
四: 回調函數如何被使用:
TestCallBack函數被調用後,啟動了一個線程,作為演示,線程人為的進行了延時處理,並且把線程運行的過程列印在屏幕上.
五:萬事具備
使用vc和Delphi各建立了一個工程,編寫回調函數的實現部分
VC版
int WINAPI CBFunc(int Param1,int Param2)
{
int res= Param1+Param2;
TCHAR Buffer[256]="";
sprintf(Buffer,"callback result = %d",res);
MessageBox(NULL,Buffer,"Testing",MB_OK); //演示回調函數被調用
return res;
}
Delphi版
function CBFunc(Param1,Param2:integer):integer;
begin
result:= Param1+Param2;
TForm1.Edit1.Text:=inttostr(result); / /演示回調函數被調用
end;
使用靜態連接的方法連接DLL里的出口函數 TestCallBack,在工程里添加 Button( 對於Delphi的工程,還需要在Form1上放一個Edit控制項,默認名為Edit1)。
響應ButtonClick事件調用 TestCallBack
TestCallBack(CBFunc) //函數的參數CBFunc為回調函數的地址
函數調用創建線程後立刻返回,應用程序可以同時干別的事情去了。現在可以看到屏幕上不停的顯示字元串,表示dll里創建的線程運行正常。一會之後,線程延時部分結束結束,vc的應用程序彈出MessageBox,表示回調函數被調用並顯示根據Param1,Param2運算的結果,Delphi的程序edit控制項里的文本則被改寫成Param1,Param2 的運算結果。
Ⅲ php回調函數是什麼樣子的靠什麼原理運行
回調函數就是那些自己寫的,但是不是自己來調,而是給別人來掉的函數。
就像下面的odd()和even()函數一樣。
<?php
function odd($var)
{
return($var % 2 == 1);
}
function even($var)
{
return($var % 2 == 0);
}
$array1 = array("a"=>1, "b"=>2, "c"=>3, "d"=>4, "e"=>5);
$array2 = array(6, 7, 8, 9, 10, 11, 12);
echo "Odd :\n";
print_r(array_filter($array1, "odd"));//這里把array1的值依次傳入到odd這個函數裡面,這種方式就稱為回調
echo "Even:\n";
print_r(array_filter($array2, "even"));
?>
下面這個例子實現函數的回調
<?
function fnCallBack($msg1, $msg2)
{
echo 'msg1:'.$msg1;
echo '<br/>';
echo 'msg2:'.$msg2;
}
$fnName = 'fnCallBack';//函數名
$params = array('hello', 'world');//將要傳入到函數裡面的參數
call_user_func_array($fnName, $params);
?>
Ⅳ 回調函數什麼概念,php中如何使用自定義的回調函數
回調函數就是一個通過函數指針調用的函數。如果你把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用為調用它所指向的函數時,我們就說這是回調函數。回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用於對該事件或條件進行響應。
Ⅳ javascript的回調函數里如何傳遞參數
function doAjax(u,param,callback){
$.ajax({
type:'POST',
url:u,
data:param,
success:callback
});
}
function showAlert(data,test1,test2){
alert(data+" "+test1+" "+test2);
}
window.onload = doAjax("server.php","id=12&type=1",function(data){showAlert(data,3,5)});
執行doAjax的時候,回調函數的調用改一下就可以了。
測試過沒問題。
Ⅵ PHP的幾種函數參數類型及一些特殊函數
給你整理如下:
默認參數:在函數聲明時給參數賦值,而且此默認參數只能在最後,調用時可不傳遞任何參數(若只有默認參數的情況下)
可變參數:一個函數可能需要可個可變數目的參數。在php中,提供了三個函數用於檢索函數中所傳遞的參數。func_get_args()返回一個提供提供給函數的所有參數的數組;func_num_args()返回提供給函數的參數數目;func_get_arg()返回一個來自參數的特定參數。具體如下:
$array = func_get_args();
$count = func_num_args();
$value = func_get_arg();
遺漏參數:當調用函數時,可以傳遞任意個參數給函數。當函數必要的參數沒有i被傳遞時,此參數值是空,並且PHP會為每個遺漏的參數發出警告
可變函數:使用可變變數,可以基於變數的值調用函數
匿名函數:又叫閉包函數,允許臨時建立一個沒有指定名稱的函數,最經常用作回調函數參數的值。需要注意的是,在匿名函數內的變數的用法不同於全局變數,在匿名函數內的變數是一個閉包變數,另外,被調用閉包的作用域不必是相同的。
Ⅶ PHP回調函數到底是個啥
當做一個變數去理解比較容易,比如$a=function(){echo "aaa";};$aa();或者function a(){};將函數名作為一個變數去賦值;$b="a"; $b();一個是匿名函數作為變數去調用,另一個是函數名作為變數去調用,這就是回調函數常用的方式,簡單來說,就這把這樣調用函數的方式叫做回調函數,個人經驗分享而已,O(∩_∩)O哈哈~
Ⅷ 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,…)的參數個數根據回調函數的參數來確定的。
Ⅸ 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}是存在的。
Ⅹ 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的事件模型就可以實現了,而且可以將回調函數放在某個對象中。