首頁 後端開發 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,其流程大致就是發佈程式碼到一個全新的目錄,然後再軟連結到真正的發布目錄。

1

2

3

4

5

6

7

8

├── 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 配置,必要的監控總是有好處的。

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

    1

    2

    3

    4

    正確發布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

1

2

3

4

5

6

7

8

<?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:

1

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 代码:

1

2

3

<?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」是单步原子操作,实际上也是错误的:

1

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」

1

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

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

熱工具

記事本++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教學
1664
14
CakePHP 教程
1423
52
Laravel 教程
1317
25
PHP教程
1268
29
C# 教程
1246
24
在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

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

php程序在字符串中計數元音 php程序在字符串中計數元音 Feb 07, 2025 pm 12:12 PM

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

解釋PHP中的晚期靜態綁定(靜態::)。 解釋PHP中的晚期靜態綁定(靜態::)。 Apr 03, 2025 am 12:04 AM

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

什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? 什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? Apr 03, 2025 am 12:03 AM

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

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不斷進化和優化,適用於初學者和經驗豐富的開發者。

See all articles