PHP7擴展開發之Hello World


一、下載PHP源代碼
要開發PHP擴展,需要先下載PHP的源代碼,一方面是因為我們的擴展一般會用到PHP自身定義的函數和宏,另一方面我們可以利用官方提供的工具減少工作量。
我下載了PHP-7.0.2,地址是: http://cn2.php.net/get/php-7.0.2.tar.gz。
解壓源碼壓縮包, tar xzf php-7.0.2.tar.gz,我們現在只需要關注Zend和ext這兩個目錄。
Zend目錄里面包含了PHP的Zend Engine源代碼,有些函數和宏的定義我們需要在這里面簡單地看一下。
ext目錄里面包含了PHP原生的擴展,以及我們開發自己的擴展時可以利用的工具,Linux下使用ext_skel,Windows下使用ext_skel_win32.php

二、使用 ext_skel工具
我們可以在ext目錄下看到所有的PHP原生擴展,其中包括了熟悉的curl,json,mbstring,simplexml,sockets等擴展,還有很多沒有用過甚至沒有聽說過的擴展,不用在意這些,我們先打開我們最熟悉的curl來看看,有config.m4配置文件,有php_curl.h,curl_file.c等源代碼,還有一些中間文件,最后還有一個tests目錄,里面放的curl擴展的單元測試。重點關注 config.m4,php_curl.h,curl_file.c即可,最簡單的場景下這三個文件就是一個擴展的全部組成部分了。

打開隨便看一下,不算太復雜,但是自己寫一個類似的還是挺頭疼的,這時就需要用到我前面提到的ext_skel工具了。這個工具也在ext目錄下,我們執行一下,./ext_skel --help,可以看到若干參數,我們用到的只有--extname=module,這里填上自己開發的擴展名稱。想深入了解各個參數的作用可以看這里: http://php.net/manual/en/internals2.buildsys.skeleton.php

   
   
   
           
  1. ./ext_skel --extname=hello
ext目錄下多了一個hello目錄,我們后續的工作都在這個目錄下面,工具已經為我們自動生成了一些文件。

config.m4配置文件
開發PHP擴展,在寫C代碼之前,要先配置一下這里。我們打開可以看到詳細的注釋說明,dnl是注釋語法。
如果你的擴展用到了外部依賴,就配置--with-hello選項,否則配置--enable-hello選項,刪除這下面3行的del注釋
   
   
   
           
  1. PHP_ARG_ENABLE(hello, whether to enable hello support,
  2. Make sure that the comment is aligned:
  3. [ --enable-hello Enable hello support])
PHP_ARG_WITH和PHP_ARG_ENABLE這兩個宏用來配置configure選項,一個配置需要外部依賴的,另一個配置不需要外部依賴的
配置好的內容,在后面執行configure --help時可以看到。

php_hello.h頭文件
類似於C語音的頭文件,包含了一些自定義的結構和函數聲明,在這個demo中暫時不需要改動

hello.c代碼文件
真正的邏輯代碼都在這個文件中,后面會詳細介紹。

三、編寫代碼
好了,到這一步我們終於要開始寫代碼了,打開hello.c文件。
整個擴展的入口是zend_module_entry這個結構,具體的定義可以在Zend目錄下的zend_modules.h文件中看到,一共有十幾個屬性,快速跳過,我們暫時只需要"hello world"。
   
   
   
           
  1. zend_module_entry hello_module_entry = {
  2. STANDARD_MODULE_HEADER,
  3. "hello",
  4. hello_functions,
  5. PHP_MINIT(hello),
  6. PHP_MSHUTDOWN(hello),
  7. PHP_RINIT(hello), /* Replace with NULL if there's nothing to do at request start */
  8. PHP_RSHUTDOWN(hello), /* Replace with NULL if there's nothing to do at request end */
  9. PHP_MINFO(hello),
  10. PHP_HELLO_VERSION,
  11. STANDARD_MODULE_PROPERTIES
  12. };
  • STANDARD_MODULE_HEADER幫我們實現了前面6個屬性
  • "hello"是擴展的名字
  • hello_functions是擴展包含的全部方法的集合
  • 后面5個宏分別代表5個擴展特定方法
  • PHP_HELLO_VERSION是擴展的版本號,定義在頭文件中
  • STANDARD_MODULE_PROPERTIES幫我們實現了剩下的屬性
暫時都不需要修改,知道這是一個入口就行。順着這個入口,我們繼續看怎么給擴展添加方法,在hello_functions[]方法數組中已經有了一個示例方法confirm_hello_compiled,我們參考它寫我們的方法hello_world
   
   
   
           
  1. const zend_function_entry hello_functions[] = {
  2. PHP_FE(confirm_hello_compiled, NULL) /* For testing, remove later. */
  3. PHP_FE(hello_world, NULL)
  4. PHP_FE_END /* Must be the last line in hello_functions[] */
  5. };
先在擴展的方法數組中添加上 hello_world ,然后再定義 hello_world 。找到 confirm_hello_compiled方法定義的地方,在它下面依葫蘆畫瓢, php_printf是Zend Engine中的printf方法。
   
   
   
           
  1. PHP_FUNCTION(hello_world)
  2. {
  3. php_printf("Hello World!\n");
  4. RETURN_TRUE;
  5. }
四、編譯安裝
最后就是編譯安裝我們的擴展了,安裝過PHP擴展的同學不用看,沒有經驗的可以參考一下。
   
   
   
           
  1. phpize
  2. ./configure
  3. make
  4. make install
現在PHP的擴展目錄中已經有了hello.so這個文件,在php.ini中添加上擴展的配置
    
    
    
            
  1. extension = hello.so

五、測試
寫一個test.php方法,執行腳本就可以看到"Hello World!"
   
   
   
           
  1. <?php
  2. hello_world();






免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM