正確發布PHP程式碼的實例分享
幾乎每一個PHP 程式設計師都發布過程式碼,可能是透過 FTP 或 rsync 同步的,也可能是透過svn 或git 更新的。一個活躍的專案可能每天都要發布若干次程式碼,但是現實卻是很少有人注意其中的細節,實際上這裡面有好多坑,很可能你就在坑中卻渾然不知。
一個正確實現的發布系統至少應該支援原子發布。如果說每一個版本都表示一個獨立的狀態的話,那麼在發布期間,任何一次請求只能在單一狀態下執行。如此稱之為支持原子發布;反之如果在發布期間,一次請求跨越不同的狀態,那麼就不能稱之為原子發布。我們不妨舉個例子來說明一下:假設一次請求需要include
兩個PHP 文件,分別是a.php
和b.php
,當include a.php
完成後,發布程式碼,接著include b.php
,如果處理不當的話,那麼就可能會導致舊版本的a. php
和新版的b.php
同時存在於同一個請求之中,換句話說就是沒有實作原子發布。
開源世界裡有很多不錯的發布程式碼工具,例如 ruby 社群的capistrano,其流程大致就是發佈程式碼到一個全新的目錄,然後再軟連結到真正的發布目錄。
1 2 3 4 5 6 7 8 |
|
不過鑑於 PHP 本身的特殊性,如果只是簡單地套用上面的流程,那麼將很難實現真正的原子發布。要理清個中緣由,還需要了解PHP 中的兩個Cache
的概念:
##opcode cache
正確發布PHP程式碼的實例分享 cache
opcode cache,基本上就是
apc 或
zend opcode,關於它的作用,大家都已經很熟悉,不必多言,需要注意的是
apc 的bug 很多,比如開啟了apc. enable_cli 配置後就會有很多靈異問題,所以說
opcode cache 還是盡可能使用
zend opcache 吧,如果需要快取數據,可以用apcu。此外
apc 和
zend opcode 對快取鍵的選擇有所差異:
apc 選擇的是檔案的
inode,
zend opcode 選擇的是檔案的
path。
正確發布PHP程式碼的實例分享 cache,它的作用是緩衝獲取文件資訊的
IO 操作,大多數時候它對我們而言是透明的,以至於很多人都不知道它的存在,需要注意的是正確發布PHP程式碼的實例分享 cache 是進程層級的,也就是說,每一個
php-fpm 進程都有自己獨立的
正確發布PHP程式碼的實例分享 cache。
opcode cache 或
正確發布PHP程式碼的實例分享 cache 裡的資料出現過期,那麼就會出現一部分快取是舊文件,一部分快取是新文件的非原子發布的情況,為了避免這種情況,我們應該保證緩存過期時間足夠長,最好是除非我們手動刷新,否則永遠不過期,對應到配置上就是:關閉apc.stat、opcache.validate_timestamps配置,設定足夠大的正確發布PHP程式碼的實例分享_cache_size、正確發布PHP程式碼的實例分享_cache_ttl 配置,必要的監控總是有好處的。
1 2 3 4 |
|
php-fpm 自然是能夠解決問題,不過對腳本語言來說重啟太重了!難道除了重啟就沒有別的辦法了?
opcode cache 是透過
正確發布PHP程式碼的實例分享 cache 獲取文件信息,即便軟連結已經指向了新位置,但如果
正確發布PHP程式碼的實例分享 cache 裡還保存著舊資料的話,
opcode cache 依然無法知道新程式碼的存在,預設情況下,正確發布PHP程式碼的實例分享_cache_ttl 快取有效期是兩分鐘,這表示發布代碼後,可能要兩分鐘才能生效。為了讓發布盡快生效,需要以進程為單位清除
正確發布PHP程式碼的實例分享 cache:
1 2 3 4 5 6 7 8 |
|
apc 環境下基本上就能工作了,但是在
zend opcode 環境下還可能有問題。因為在預設情況下opcache.revalidate_path 是關閉的,此時會快取未解析的符號連結的值,這會導致即便軟連結指向修改了,也無法生效,所以在使用
zend opcode的時候,如果使用了軟鏈接,視情況可能需要把
opcache.revalidate_path 啟動。
详细介绍参考:PHP’s OPCache extension review。
BTW:如果需要手动重置 opcode cache
,需要注意的是因为它是基于 SAPI 的概念,所以不能直接在命令行下调用 apc_clear_cache 或者 opcache_reset 方法来重置缓存,当然办法总是有的,那就是使用 CacheTool 在命令行下模拟 fastcgi
请求。
分析到这里,我们不妨反思一下:在 PHP 中原子发布之所以是一个棘手的问题,归根结底是因为软链接和缓存之间的的矛盾。不管是 opcode cache
还是 正確發布PHP程式碼的實例分享 cache
,都是 PHP 固有的缓存特性,基于客观需要无法绕开,如此说来是否有办法绕开软链接,使其成为马奇诺防线呢?答案是 NGINX 的 $正確發布PHP程式碼的實例分享_root:
1 |
|
有了 $正確發布PHP程式碼的實例分享_root
,即便 DOCUMENT_ROOT
目录中含有软链接,NGINX 也会把软链接指向的真正的路径发给 PHP,也就是说,对 PHP 而言,软链接已经不存在了!不过作为代价,每一次请求,NGINX 都要通过相对昂贵的 IO 操作获取 $正確發布PHP程式碼的實例分享_root
的值,通过 strace
命令我们能监控这一过程,下图从 current
到 foo
的过程:
在本例中,压测发现使用 $正確發布PHP程式碼的實例分享_root
后,性能下降了大约 5% 左右,不过明眼人一下就能发现,虽然 $正確發布PHP程式碼的實例分享_root
导致了 lstat
和 readlink
操作,但是 lstat
操作的次数是和目录深度成正比的,也就是说目录越深,执行的 lstat
次数越多,性能下降也就越大。如果能够降低发布目录的深度,那么可以预计还能降低一些性能损耗。
结尾介绍一下 Deployer,它是 PHP 中做得比较好的工具,有很多特色,比如支持并行发布,具体演示如下图,左边是串行,右边是并行,使用「vvv」能得到更详细信息:
不过 Deployer 在原子发布上有一点瑕疵,具体见 release/symlink
代码:
1 2 3 |
|
在 release
的时候,它是先删除再创建,是一个两步的非原子操作,在 symlink
的时候,看上去「ln -sfn」
是单步原子操作,实际上也是错误的:
1 |
|
通过 strace
我们能清晰的看到,虽然表面上使用「ln -sfn」
是一步操作,但是内部依然是按照先删除再创建的逻辑执行的,实际上这里应该搭配使用「ln & mv」
:
1 |
|
先通过 ln
创建一个临时的软链接,再通过 mv
实现原子操作,此时如果使用 strace
监控,会发现 mv
的 「T」
选项实际上仅仅执行了一个 rename
操作,所以是原子的。
BTW:在使用「ln -sfn」
前后,如果使用 stat
查看新旧文件的 inode
的话,可能会发现它们拥有一样的 inode
值,看上去和我们的结论相悖,其实不然,实际上只是复用删除值而已(如果想验证,注意 Linux 会复用,Mac 不会复用)。
据说一千个人的心中就有一千个哈姆雷特,不过我希望所有的 PHP 程序员在发布 PHP 代码的时候都能采用一种方法,那就是本文介绍的方法,正确的方法。
相关推荐:
以上是正確發布PHP程式碼的實例分享的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

PHP和Python各有優勢,選擇依據項目需求。 1.PHP適合web開發,尤其快速開發和維護網站。 2.Python適用於數據科學、機器學習和人工智能,語法簡潔,適合初學者。

PHP在電子商務、內容管理系統和API開發中廣泛應用。 1)電子商務:用於購物車功能和支付處理。 2)內容管理系統:用於動態內容生成和用戶管理。 3)API開發:用於RESTfulAPI開發和API安全性。通過性能優化和最佳實踐,PHP應用的效率和可維護性得以提升。

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP仍然具有活力,其在現代編程領域中依然佔據重要地位。 1)PHP的簡單易學和強大社區支持使其在Web開發中廣泛應用;2)其靈活性和穩定性使其在處理Web表單、數據庫操作和文件處理等方面表現出色;3)PHP不斷進化和優化,適用於初學者和經驗豐富的開發者。
