首頁 後端開發 php教程 正確發布PHP程式碼的實例分享

正確發布PHP程式碼的實例分享

Mar 12, 2018 pm 02:44 PM
php 分享 實例


幾乎每一個PHP 程式設計師都發布過程式碼,可能是透過 FTP 或 rsync 同步的,也可能是透過svngit 更新的。一個活躍的專案可能每天都要發布若干次程式碼,但是現實卻是很少有人注意其中的細節,實際上這裡面有好多坑,很可能你就在坑中卻渾然不知。

一個正確實現的發布系統至少應該支援原子發布。如果說每一個版本都表示一個獨立的狀態的話,那麼在發布期間,任何一次請求只能在單一狀態下執行。如此稱之為支持原子發布;反之如果在發布期間,一次請求跨越不同的狀態,那麼就不能稱之為原子發布。我們不妨舉個例子來說明一下:假設一次請求需要include 兩個PHP 文件,分別是a.phpb.php,當include a.php 完成後,發布程式碼,接著include b.php,如果處理不當的話,那麼就可能會導致舊版本的a. php 和新版的b.php 同時存在於同一個請求之中,換句話說就是沒有實作原子發布。

開源世界裡有很多不錯的發布程式碼工具,例如 ruby 社群的capistrano,其流程大致就是發佈程式碼到一個全新的目錄,然後再軟連結到真正的發布目錄。

├── current -> releases/v1
└── releases
    ├── v1
    │   ├── foo.php
    │   └── bar.php
    └── v2
        ├── foo.php
        └── bar.php
登入後複製

不過鑑於 PHP 本身的特殊性,如果只是簡單地套用上面的流程,那麼將很難實現真正的原子發布。要理清個中緣由,還需要了解PHP 中的兩個Cache 的概念:

  • ##opcode cache

  • 正確發布PHP程式碼的實例分享 cache

#先聊聊

opcode cache,基本上就是apczend opcode,關於它的作用,大家都已經很熟悉,不必多言,需要注意的是apc 的bug 很多,比如開啟了apc. enable_cli 配置後就會有很多靈異問題,所以說opcode cache 還是盡可能使用zend opcache 吧,如果需要快取數據,可以用apcu。此外apczend opcode 對快取鍵的選擇有所差異:apc 選擇的是檔案的inodezend 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 配置,必要的監控總是有好處的。

相關的技術細節特別瑣碎,建議大家仔細閱讀如下資料:

    正確發布PHP程式碼的實例分享_cache
    PHP’s OPCache extension review
    Atomic 正確發布PHP程式碼的實例分享s at Etsy
    Cache invalidation for scripts in symlinked folders
    登入後複製
在採用軟連結發布程式碼的時候,通常遇到的第一個問題多半是新程式碼不生效!即使呼叫了 apc_clear_cache 或 opcache_reset 方法也無效,重啟

php-fpm 自然是能夠解決問題,不過對腳本語言來說重啟太重了!難道除了重啟就沒有別的辦法了?

事實上之所以會出現這樣的問題,主要是因為

opcode cache 是透過正確發布PHP程式碼的實例分享 cache 獲取文件信息,即便軟連結已經指向了新位置,但如果正確發布PHP程式碼的實例分享 cache 裡還保存著舊資料的話,opcode cache 依然無法知道新程式碼的存在,預設情況下,正確發布PHP程式碼的實例分享_cache_ttl 快取有效期是兩分鐘,這表示發布代碼後,可能要兩分鐘才能生效。為了讓發布盡快生效,需要以進程為單位清除正確發布PHP程式碼的實例分享 cache

<?php

    $key = &#39;php.pid_&#39; . getmypid();    if (($rev = apc_fetch($key)) != DEPLOY_VERSION) {        if($rev < DEPLOY_VERSION) {
            apc_store($key, DEPLOY_VERSION);
        }

        clearstatcache(true);
    }
登入後複製

如此在

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:

    fastcgi_param SCRIPT_FILENAME $正確發布PHP程式碼的實例分享_root$fastcgi_script_name;    fastcgi_param DOCUMENT_ROOT $正確發布PHP程式碼的實例分享_root;
登入後複製

有了 $正確發布PHP程式碼的實例分享_root,即便 DOCUMENT_ROOT 目录中含有软链接,NGINX 也会把软链接指向的真正的路径发给 PHP,也就是说,对 PHP 而言,软链接已经不存在了!不过作为代价,每一次请求,NGINX 都要通过相对昂贵的 IO 操作获取 $正確發布PHP程式碼的實例分享_root 的值,通过 strace 命令我们能监控这一过程,下图从 currentfoo 的过程:

正確發布PHP程式碼的實例分享

在本例中,压测发现使用 $正確發布PHP程式碼的實例分享_root 后,性能下降了大约 5% 左右,不过明眼人一下就能发现,虽然 $正確發布PHP程式碼的實例分享_root 导致了 lstatreadlink 操作,但是 lstat 操作的次数是和目录深度成正比的,也就是说目录越深,执行的 lstat 次数越多,性能下降也就越大。如果能够降低发布目录的深度,那么可以预计还能降低一些性能损耗。

结尾介绍一下 Deployer,它是 PHP 中做得比较好的工具,有很多特色,比如支持并行发布,具体演示如下图,左边是串行,右边是并行,使用「vvv」能得到更详细信息:

正確發布PHP程式碼的實例分享

不过 Deployer 在原子发布上有一点瑕疵,具体见 release/symlink 代码:

<?php// 正確發布PHP程式碼的實例分享:releaserun("cd {{正確發布PHP程式碼的實例分享_path}} && if [ -h release ]; then rm release; fi");
run("ln -s $releasePath {{正確發布PHP程式碼的實例分享_path}}/release");// 正確發布PHP程式碼的實例分享:symlinkrun("cd {{正確發布PHP程式碼的實例分享_path}} && ln -sfn {{release_path}} current");
run("cd {{正確發布PHP程式碼的實例分享_path}} && rm release");?>
登入後複製

release 的时候,它是先删除再创建,是一个两步的非原子操作,在 symlink 的时候,看上去「ln -sfn」是单步原子操作,实际上也是错误的:

shell> strace ln -sfn releases/foo currentsymlink("releases/foo", "current")      = -1 EEXIST (File exists)unlink("current")                       = 0symlink("releases/foo", "current")      = 0
登入後複製

通过 strace 我们能清晰的看到,虽然表面上使用「ln -sfn」是一步操作,但是内部依然是按照先删除再创建的逻辑执行的,实际上这里应该搭配使用「ln & mv」

shell> ln -sfn releases/foo current.tmpshell> mv -fT current.tmp current
登入後複製

先通过 ln 创建一个临时的软链接,再通过 mv 实现原子操作,此时如果使用 strace 监控,会发现 mv「T」 选项实际上仅仅执行了一个 rename 操作,所以是原子的。

BTW:在使用「ln -sfn」前后,如果使用 stat 查看新旧文件的 inode 的话,可能会发现它们拥有一样的 inode 值,看上去和我们的结论相悖,其实不然,实际上只是复用删除值而已(如果想验证,注意 Linux 会复用,Mac 不会复用)。

据说一千个人的心中就有一千个哈姆雷特,不过我希望所有的 PHP 程序员在发布 PHP 代码的时候都能采用一种方法,那就是本文介绍的方法,正确的方法。

相关推荐:

php代码标志基础讲解

提高PHP代码质量的方法

JS和PHP代码实现用户输入数字后显示最大的值

以上是正確發布PHP程式碼的實例分享的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

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

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

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

熱門話題

Java教學
1665
14
CakePHP 教程
1423
52
Laravel 教程
1321
25
PHP教程
1269
29
C# 教程
1249
24
PHP和Python:比較兩種流行的編程語言 PHP和Python:比較兩種流行的編程語言 Apr 14, 2025 am 12:13 AM

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

PHP行動:現實世界中的示例和應用程序 PHP行動:現實世界中的示例和應用程序 Apr 14, 2025 am 12:19 AM

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

PHP:網絡開發的關鍵語言 PHP:網絡開發的關鍵語言 Apr 13, 2025 am 12:08 AM

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

PHP的持久相關性:它還活著嗎? PHP的持久相關性:它還活著嗎? Apr 14, 2025 am 12:12 AM

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

PHP與Python:了解差異 PHP與Python:了解差異 Apr 11, 2025 am 12:15 AM

PHP和Python各有優勢,選擇應基於項目需求。 1.PHP適合web開發,語法簡單,執行效率高。 2.Python適用於數據科學和機器學習,語法簡潔,庫豐富。

PHP和Python:代碼示例和比較 PHP和Python:代碼示例和比較 Apr 15, 2025 am 12:07 AM

PHP和Python各有優劣,選擇取決於項目需求和個人偏好。 1.PHP適合快速開發和維護大型Web應用。 2.Python在數據科學和機器學習領域佔據主導地位。

PHP與其他語言:比較 PHP與其他語言:比較 Apr 13, 2025 am 12:19 AM

PHP適合web開發,特別是在快速開發和處理動態內容方面表現出色,但不擅長數據科學和企業級應用。與Python相比,PHP在web開發中更具優勢,但在數據科學領域不如Python;與Java相比,PHP在企業級應用中表現較差,但在web開發中更靈活;與JavaScript相比,PHP在後端開發中更簡潔,但在前端開發中不如JavaScript。

PHP和Python:解釋了不同的範例 PHP和Python:解釋了不同的範例 Apr 18, 2025 am 12:26 AM

PHP主要是過程式編程,但也支持面向對象編程(OOP);Python支持多種範式,包括OOP、函數式和過程式編程。 PHP適合web開發,Python適用於多種應用,如數據分析和機器學習。

See all articles