PHP擴展開發(fā)入門教程
來源:易賢網(wǎng) 閱讀:655 次 日期:2015-03-03 10:02:44
溫馨提示:易賢網(wǎng)小編為您整理了“PHP擴展開發(fā)入門教程”,方便廣大網(wǎng)友查閱!

這篇文章主要介紹了PHP擴展開發(fā)入門教程,本文講解了使用C語言在Linux系統(tǒng)下開發(fā)一個PHP擴展應該具備的最基本知識,需要的朋友可以參考下

PHP擴展開發(fā)

我準備在此系列博文中總結(jié)我有關(guān)PHP擴展開發(fā)的學習和感悟,力圖簡單清晰地描述在Linux系統(tǒng)下開發(fā)一個PHP擴展應該具備的最基本知識。水平較低,難免有錯誤,望指出。

準備工作

首先要獲取一份PHP源碼(可以從Github上簽出,或者到官網(wǎng)上下載最新的穩(wěn)定版),然后編譯之。為了加快編譯速度,我們推薦禁用所有額外的擴展(使用--disable-all選項),但最好打開debug(使用--enable-debug選項)和線程安全(使用--enable-maintainer-zts),但要在發(fā)布擴展的時候關(guān)閉debug,視情況選擇是否需要打開線程安全:

代碼如下:

$ ./buildconf --force

$ ./configure --disable-all --enable-debug --enable-maintainer-zts

$ make

注意,我們沒有指定--prefix選項(同時也沒有make install),因為這不是必須的。注意查看輸出信息,也許你需要安裝一些依賴包才能成功編譯PHP。

編譯后的PHP的可執(zhí)行程序在源碼的sapi目錄下,對應不同的宿主環(huán)境有不同的子目錄,我們以后都主要使用cli(command line interface)環(huán)境,可以建一個別名方便引用:

代碼如下:

$ alias php-dev=/usr/local/src/php-5.6.5/sapi/cli/php

有一些命令行選項是很有用的:

代碼如下:

php-dev -h          # 打印幫助信息

php-dev -v          # 打印版本信息

php-dev --ini        # 打印配置信息

php-dev -m          # 打印加載的模塊信息

php-dev -i          # phpinfo

php-dev -r <code>      # 執(zhí)行code里的代碼

擴展骨架

PHP的所有官方擴展都在源碼的ext目錄下,我們自己寫的擴展也可以放在該目錄下。注意,該目錄下有個名為ext_skel的shell腳本,它是用來生成PHP擴展骨架的,使用該腳本,可以幫我們快速創(chuàng)建PHP擴展:

代碼如下:

$ ./ext_skel --extname=myext

上面的命令幫我們創(chuàng)建了一個名為myext的擴展,源碼在myext目錄下。不帶任何參數(shù)的執(zhí)行該腳本可以打印幫助信息,這樣你可以查看到該腳本提供的更多選項。

接下來讓我們完成我們的擴展。進入myext目錄,編輯config.m4配置文件,找到PHP_ARG_ENABLE宏函數(shù),去掉前面的dnl注釋(共三行)。退回到源碼根目錄,重新執(zhí)行buildconf、configure和make命令:

代碼如下:

$ ./buildconf --force

$ ./configure --help | grep myext

--enable-myext Enable myext support

$ ./configure --disable-all --enable-myext --enable-debug --enable-maintainer-zts

$ make

注意,我們用./configure --help | grep myext打印了我們擴展的加載情況,如果看不到下面的輸出,則說明我們的擴展沒有配置成功,回頭檢查下config.m4文件。

這次編譯應該非???,因為大部分代碼都已經(jīng)編譯過了。PHP還有另外一種編譯擴展的方法(使用動態(tài)連接的方式,將擴展編譯為.so的文件),不過我們推薦在開發(fā)擴展的時候使用靜態(tài)編譯,因為這樣省去了在配置文件中加載擴展的步驟。

一切順利的話,我們的第一個擴展就已經(jīng)可以執(zhí)行了:

代碼如下:

$ php-dev -m | grep myext

myext

$ php-dev -r 'echo confirm_myext_compiled("myext") . "n";'

Congratulations! You have successfully modified ext/myext/config.m4. Module myext is now compiled into PHP.

第一個命令顯示了我們的擴展已經(jīng)被加載。第二個命令執(zhí)行了ext_skel擴展骨架自動為我們創(chuàng)建的函數(shù)。當然,這個函數(shù)毫無意義,不過我們可以很容易的把這個函數(shù)改編成hello world。

手動創(chuàng)建擴展

大部分教程都是以ext_skel擴展骨架為原型講述擴展開發(fā)的,這種做法當然很方便快捷。但是我個人更喜歡純手工開發(fā)擴展的方式,因為這樣更容易理解其中的每一個細節(jié)。

手動創(chuàng)建擴展,先進入ext目錄,創(chuàng)建我們的擴展目錄myext2。有幾個文件是必須的:config.m4,myext2.c和php_myext2.h。

首先,我們來編寫配置文件config.m4:

代碼如下:

PHP_ARG_ENABLE(myext2, whether to enable myext2 support,

[ --enable-myext2 Enable myext2 support])

if test "PHP_MYEXT2" != "no"; then

PHP_NEW_EXTENSION(myext2, myext2.c, $ext_shared)

fi

config.m4其實是autoconf程序使用的配置文件,autoconf是autotools工具箱里重要的組成。完整介紹autoconf的用法是需要很長時間的,好在我們這里的用法非常簡單。

PHP_ARG_ENABLE是PHP為autoconf定義的宏函數(shù),myext2是它的第一個參數(shù),指出了擴展的名字;后面兩個參數(shù)只是在make和configure執(zhí)行時用來顯示的,所以我們可以隨便寫。[ ]在autoconf語法中的作用類似于雙引號,用來包裹字符串(注意第二個參數(shù)中包含了空格,但是可以不用方括號起來)。還有第四個參數(shù)用來指明擴展默認是開啟還是關(guān)閉(yes或no),默認是no。

下面三行其實就是shell語法,判斷我們是否開啟了PHP_MYEXT2擴展模塊。如果開啟了該擴展模塊(--enable-myext2),則$PHP_MYEXT2變量的值不為no,因此執(zhí)行PHP_NEW_EXTENSION宏。這個宏函數(shù)也是PHP為autoconf定義的擴展語法,第一個參數(shù)同樣是擴展名稱;第二個參數(shù)是擴展要編譯的C文件,如果有多個,依次寫下去就可以了(空格分隔);第三個參數(shù)固定是$ext_shared。

接下來編寫php_myext2.h頭文件,該文件的命名是PHP擴展的規(guī)范 — php_擴展名.h:

代碼如下:

#ifndef PHP_MYEXT2_H

#define PHP_MYEXT2_H

extern zend_module_entry myext2_module_entry;

#define phpext_myext2_ptr &myext2_module_entry

#define PHP_MYEXT2_VERSION "0.1.0"

/* prototypes */

PHP_FUNCTION(hello);

#endif /* PHP_MYEXT2_H */

這里主要的代碼是定義了名為phpext_myext2_ptr的宏,PHP底層通過該宏來引用我們的擴展??梢钥闯觯摵甑拿瑯邮怯幸?guī)范的 — phpext_擴展名_ptr。而myext2_module_entry是我們稍后要在.c文件里定義的結(jié)構(gòu)體,它的命名也是規(guī)范的 — 擴展名_module_entry。

此外我們還定義了一個標識我們擴展版本號的宏和一個函數(shù)原型(通過PHP_FUNCTION宏,PHP_FUNCTION宏函數(shù)的參數(shù)是外部可使用的函數(shù)名),稍后我們會來實現(xiàn)這個函數(shù)。

最后來看下myext2.c文件的實現(xiàn):

代碼如下:

#include "php.h"

#include "php_myext2.h"

/* {{{ myext2_functions[]

*

* Every user visible function must have an entry in myext2_functions[].

*/

static const zend_function_entry myext2_functions[] = {

PHP_FE(hello, NULL)

PHP_FE_END

};

/* }}} */

/* {{{ myext2_module_entry

*/

zend_module_entry myext2_module_entry = {

STANDARD_MODULE_HEADER,

"myext2", /* module name */

myext2_functions, /* module functions */

NULL, /* module initialize */

NULL, /* module shutdown */

NULL, /* request initialize */

NULL, /* request shutdown */

NULL, /* phpinfo */

PHP_MYEXT2_VERSION, /* module version */

STANDARD_MODULE_PROPERTIES

};

/* }}} */

#ifdef COMPILE_DL_MYEXT2

ZEND_GET_MODULE(myext2)

#endif

/* {{{ proto void hello()

Print "hello world!" */

PHP_FUNCTION(hello)

{

php_printf("hello world!n");

}

/* }}} */

對比下擴展骨架創(chuàng)建的.c文件就會發(fā)現(xiàn),我們的.c文件非常的簡單,其實這些對一個最基本的擴展來說就已經(jīng)足夠了。

上面的代碼是簡單而清晰的,大部分注釋已經(jīng)很具說明性了。我們再簡要概括下:

1.開頭包含我們要用到的頭文件。php.h是必須的,它已經(jīng)幫我們包含了我們會用到的絕大多數(shù)的標準庫文件,比如stdio.h,stdlib.h等等。

2.myext2_functions定義了由我們要暴露出去的函數(shù)構(gòu)成的結(jié)構(gòu)體數(shù)組,每一個元素通過PHP_FE宏來指定。PHP_FE宏有兩個參數(shù),第一個是外部可使用的函數(shù)名,第二個是參數(shù)信息(這里我們簡單使用了NULL),最后一個元素必須是PHP_FE_END。注意它的注釋,再次強調(diào),每一個要暴露給外部使用的函數(shù),都必須在該結(jié)構(gòu)體數(shù)組中有定義。

3.myext2_module_entry定義了我們的模塊信息,它是一個結(jié)構(gòu)體,大部分屬性都已經(jīng)通過注釋給出了說明。注意中間的五個函數(shù)指針,我們都簡單的置為了NULL,在后續(xù)的博文中會講述它們的用法。

4.ZEND_GET_MODULE(myext2)宏函數(shù)是被ifdef宏包含的,所以說它是否調(diào)用是視情況而定的。至于什么情況下會被調(diào)用,什么情況下不會被調(diào)用,在后續(xù)的博文中會講述。

5.最后幾行代碼我們實現(xiàn)了hello函數(shù),它很簡單,調(diào)用php_printf輸出hello world!跟一個換行符,php_printf的用法和printf完全一樣。

6.注釋里的 {{{ 和 }}} 是為了方便vim等編輯器折疊而使用的,我們推薦你也這樣來寫注釋。

這里面涉及了一些宏,比如PHP_FE,PHP_FE_END,PHP_FUNCTION等等,完整介紹這些宏要到后續(xù)的博文中才可以,眼下最簡單的辦法就是記住這些宏。

注意到我們每一個文件的命名,變量的命名,空格和縮進,以及注釋等都是非常規(guī)范的,遵循這些規(guī)范,可以使我們編寫的代碼和PHP本身的代碼更加契合,我們也推薦你使用這樣的規(guī)范來開發(fā)PHP擴展。

最后,編譯運行我們的擴展:

代碼如下:

$ ./buildconf --force

$ ./configure --help | grep myext2

--enable-myext2 Enable myext2 support

$ ./configure --disable-all --enable-myext2 --enable-debug --enable-maintainer-zts

$ make

$ php-dev -m | grep myext2

myext2

$ php-dev -r 'hello();'

hello world!

更多信息請查看IT技術(shù)專欄

更多信息請查看網(wǎng)絡(luò)編程
易賢網(wǎng)手機網(wǎng)站地址:PHP擴展開發(fā)入門教程

2025國考·省考課程試聽報名

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網(wǎng)安備53010202001879號 人力資源服務許可證:(云)人服證字(2023)第0102001523號
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關(guān)注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權(quán)所有:易賢網(wǎng)