# 20.4 覆寫INI\_SYSTEM和INI\_PERDIR選項
# 覆寫INI\_SYSTEM和INI\_PERDIR選項
在上一章中, 你曾經使用zend\_alter\_ini\_setting()修改過?些php的ini選項. 由于samp/embed直接將你的腳本推入了運行時模式, 因此許多重要的INI選項在控制返回到你的應用 時并沒有被修改. 為了修改這些值, 就需要在主引擎啟動之后而請求啟動之前執行代碼.
有一種方式是拷貝php\_embed\_init()的內容到你的應用中, 在你的本地拷貝中做必要 的修改, 接著使用你修改后的版本替代它. 當然這種方式可能會有問題.
首先也是最重要的, 你實際已經對別人的部分代碼做了分支, 然而可能別人還會向其 中添加新的代碼. 現在, 你就不再是只維護自己的應用了, 還需要保持分支出來的代碼和主 分支保持一致. 幸運的是, 還有幾種更簡單的方法:
#### 覆寫默認的php.ini文件
因為嵌入式和其他的php sapi實現一樣都是sapi, 它通過?個sapi\_module\_struct掛入 到引擎中. 嵌入式SAPI定義并設置了這個結構體的一個實例, 你的應用可以在調用 php\_embed\_init()之前訪問它.
在這個結構體中, 有一個名為php\_ini\_path\_override的char \*類型字段. 為了讓嵌入的 請求使用你的可選文件擴展php和Zend, 只需要在調用php\_embed\_init()之前將這個字段 設置為NULL終止的字符串. 下面是embed4.c中修改版的startup\_php()函數:
```
static void startup_php(void)
{
/* Create "dummy" argc/argv to hide the arguments
* meant for our actual application */
int argc = 1;
char *argv[2] = { "embed4", NULL };
php_embed_module.php_ini_path_override = "/etc/php_embed4.ini";
php_embed_init(argc, argv PTSRMLS_CC);
}
```
這就使得每個使用嵌入庫的應用可以保持自定義, 而不用將自己的配置暴露給別人. 相反, 如果你想要你的應用不使用php.ini, 只需要設置php\_embed\_module的 php\_ini\_ignore字段, 這樣所有的設置都將使用內建的默認值, 除非由你的應用手動進行修改.
#### 覆寫嵌入啟動
sapi\_module\_struct結構還包含?些回調函數, 下面是其中4個在PHP啟動和終止階段 比較有用的回調:
```
/* From main/SAPI.h */
typedef struct _sapi_module_struct {
...
int (*startup)(struct _sapi_module_struct *sapi_module);
int (*shutdown)(struct _sapi_module_struct *sapi_module);
int (*activate)(TSRMLS_D);
int (*deactivate)(TSRMLS_D);
...
} sapi_module_struct;
```
這些方法的名字熟悉嗎? 它們對應于擴展的MINIT, MSHUTDOWN, RINIT, RSHUTDOWN, 并且和對應在擴展生命周期中的階段?致. 要利用這些鉤子, 可以如下修 改embed4中的startup\_php()函數:
```
static int (*original_embed_startup)(struct _sapi_module_struct *sapi_module);
static int embed4_startup_callback(struct _sapi_module_struct *sapi_module)
{
/* 首先調用原來的啟動回調, 否則環境未就緒 */
if (original_embed_startup(sapi_module) == FAILURE) {
/* 這里可以做應用的失敗處理 */
return FAILURE;
}
/* 調用原來的embed_startup實際上讓我們進入到ACTIVATE階段而不是STARTUP階段, * 但是我們仍然可以修改多數INI_SYSTEM和INI_PERDIR選項.
*/
zend_alter_ini_entry("max_execution_time", sizeof("max_execution_time"),
"15", sizeof("15") - 1, PHP_INI_SYSTEM, PHP_INI_STAGE_ACTIVATE);
zend_alter_ini_entry("safe_mode", sizeof("safe_mode"),
"1", sizeof("1") - 1, PHP_INI_SYSTEM, PHP_INI_STAGE_ACTIVATE);
return SUCCESS;
}
static void startup_php(void)
{
/* 創建假的argc/argv, 隱藏應用實際的參數 */ int argc = 1;
char *argv[2] = { "embed4", NULL };
/* 使用我們自己的啟動函數覆寫標準的啟動方法, 但是保留了原來的指針, 因此它仍然能夠被調用到 */ original_embed_startup = php_embed_module.startup;
php_embed_module.startup = embed4_startup_callback;
php_embed_init(argc, argv PTSRMLS_CC);
}
```
使用safe\_mode, open\_basedir這樣的選項, 以及其他用以限制獨立腳本行為的選項, 可以讓你的應用更加安全可靠.
## links
- [目錄](preface.md)
- 20.3 [初始化php](20.3.html)
- 20.5 [捕獲輸出](20.5.html)
- 介紹
- 1 PHP的生命周期
- 1.1 讓我們從SAPI開始
- 1.2 PHP的啟動與終止
- 1.3 PHP的生命周期
- 1.4 線程安全
- 1.5 PHP的生命周期
- 2 PHP變量在內核中的實現
- 2.1 變量的類型
- 2.2 變量的值
- 2.3 創建PHP變量
- 2.4 變量的存儲方式
- 2.5 變量的檢索
- 2.6 類型轉換
- 2.7 小結
- 3 內存管理
- 3.1 內存管理
- 3.2 引用計數
- 3.3 內存管理
- 4 動手編譯PHP
- 4.1 動手編譯PHP
- 4.2 動手編譯PHP
- 4.3 Unix/Linux平臺下的編譯
- 4.4 在Win32平臺上編譯PHP
- 4.5 動手編譯PHP
- 5 Your First Extension
- 5.1 Your First Extension
- 5.2 編譯我們的擴展
- 5.3 靜態編譯
- 5.4 編寫函數
- 5.5 Your First Extension
- 6 函數返回值
- 6.1 函數返回值
- 6.2 引用與函數的執行結果
- 6.3 函數返回值
- 7 函數的參數
- 7.1 函數的參數
- 7.2 函數的參數
- 7.3 函數的參數
- 8 使用HashTable與{數組}
- 8.1 使用HashTable與{數組}
- 8.2 使用HashTable與{數組}
- 8.3 使用HashTable與{數組}
- 8.4 使用HashTable與{數組}
- 9 PHP中的資源類型
- 9.1 PHP中的資源類型
- 9.2 PHP中的資源類型
- 9.3 PHP中的資源類型
- 9.4 PHP中的資源類型
- 10 PHP中的面向對象(一)
- 10.1 PHP中的面向對象(一)
- 10.2 PHP中的面向對象(一)
- 10.3 PHP中的面向對象(一)
- 10.4 PHP中的面向對象(一)
- 10.5 PHP中的面向對象(一)
- 11 PHP中的面向對象(二)
- 11.1 PHP中的面向對象(二)
- 11.2 PHP中的面向對象(二)
- 11.3 PHP中的面向對象(二)
- 12 啟動與終止的那點事
- 12.1 關于生命周期
- 12.2 MINFO與phpinfo
- 12.3 常量
- 12.4 PHP擴展中的全局變量
- 12.5 PHP語言中的超級全局變量(Superglobals)
- 12.6 小結
- 13 INI設置
- 13.1 聲明和訪問INI設置
- 13.2 小結
- 14 流式訪問
- 14.1 流的概覽
- 14.2 訪問流
- 14.3 靜態資源操作
- 14.4 links
- 15 流的實現
- 15.1 php流的表象之下
- 15.2 包裝器操作
- 15.3 實現一個包裝器
- 15.4 操縱
- 15.5 檢查
- 15.6 小結
- 16 有趣的流
- 16.1 上下文
- 16.2 過濾器
- 16.3 小結
- 17 配置和鏈接
- 17.1 autoconf
- 17.2 庫的查找
- 17.3 強制模塊依賴
- 17.4 Windows方言
- 17.5 小結
- 18 擴展生成
- 18.1 ext_skel
- 18.2 PECL_Gen
- 18.3 小結
- 19 設置宿主環境
- 19.1 嵌入式SAPI
- 19.2 構建并編譯一個宿主應用
- 19.3 通過嵌入包裝重新創建cli
- 19.4 老技術新用
- 19.5 小結
- 20 高級嵌入式
- 20.1 回調到php中
- 20.2 錯誤處理
- 20.3 初始化php
- 20.4 覆寫INI_SYSTEM和INI_PERDIR選項
- 20.5 捕獲輸出
- 20.6 同時擴展和嵌入
- 20.7 小結