国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

PHP之內置web服務器

MorePainMoreGain / 595人閱讀

摘要:首發于我的博客前言從開始,就提供了一個內置的服務器。在中的安裝一節中介紹了一個命令可以使用內置服務器實現外部訪問的命令。

首發于:我的博客
前言

PHP從5.4開始,就提供了一個內置的web服務器。

這個主要是用來做本地的開發用的。不能用于線上環境。現在我就介紹一下這個工具如何使用。

基礎應用

首先我們假定項目目錄是/home/baoguoxiao/www/php/demo,外界可訪問的目錄是/home/baoguoxiao/www/php/demo/public。然后訪問的端口是8000,入口文件是index.phpindex.html。那么我們可以執行如下命令:

cd /home/baoguoxiao/www/php/demo/public
php -S localhost:8000

然后這個時候就可以正常訪問了。

那么現在有個問題,就是難道每次必須要進入public文件夾才能啟動web服務器嗎,其實我們可以指定根目錄的,那么可以使用如下命令:

cd /home/baoguoxiao/www/php/demo
php -S localhost:8000 -t public/

那么現在有一個問題就是說,如果我們使用了單入口,而且還是用了PATHINFO模式。那么上面的可能就有問題了。

對此,我們可以使用如下方案:

cd /home/baoguoxiao/www/php/demo
php -S localhost:8000 router.php

router.php 文件的代碼

/**
 * 對URL進行解析,并獲取請求的文件名
 */
$uri = urldecode(parse_url($_SERVER["REQUEST_URI"], PHP_URL_PATH));

/**
 * 判斷是否存在該文件,如果不存在,則直接繼續加載入口文件
 */
if ($uri !== "/" && file_exists(__DIR__ . "$uri")) {
    return false;
}

/**
 * 加載入口文件
 */
require_once "./index.php";

通過這個路由文件,我們就可以支持目前常用的開發情況了。

框架參考

上面的方式是我們自己的實現,那么我們也可以看看相關知名框架的實現方法。

比如 Laravel 和 Symfony。

Laravel

在Laravel中的安裝一節中介紹了一個命令可以使用PHP內置web服務器實現外部訪問的命令。實現的命令是:

php artisan serve

我們可以看一下相關代碼:

具體的文件路徑為:vendor/laravel/framework/src/Illuminate/Foundation/Console/ServeCommand.php

/**
 * 執行命令.
 *
 * @return int
 *
 * @throws Exception
 */
public function handle()
{
    // 切換路徑到 public 目錄
    chdir(public_path());

    // 在命令臺進行輸出相關內容
    $this->line("Laravel development server started: host()}:{$this->port()}>");

    // 執行外部程序,并且 $status 為系統的返回狀態
    passthru($this->serverCommand(), $status);

    // $status 為0 表示執行正常, 為其他大于0的數字表示出現了錯誤,有可能是端口被搶占了,這個時候就會接著判斷是否進行再次嘗試
    if ($status && $this->canTryAnotherPort()) {
        // 對綁定的端口號加1 默認是8000, 如果失敗則重試端口號為8001,再次失敗重試端口號為8002,以此類推。
        $this->portOffset += 1;
        // 再次調用此程序
        return $this->handle();
    }
    // 返回狀態值
    return $status;
}

/**
 * 獲取完整的 server 命令.
 *
 * @return string
 */
protected function serverCommand()
{
    return sprintf("%s -S %s:%s %s",
        
        // 獲取PHP可執行命令的路徑
        ProcessUtils::escapeArgument((new PhpExecutableFinder)->find(false)),
        
        // 獲取需要綁定的host
        $this->host(),

        // 獲取需要綁定的端口
        $this->port(),

        // 對需要執行的參數進行轉義處理。這里的 server 就是我們之前說的路由文件,它在項目的根路徑下
        ProcessUtils::escapeArgument(base_path("server.php"))
    );
}

對上面的命令進行翻譯一下,實際上就是執行的

cd ./public
php -S 0.0.0.0:8000 ../server.php
note:

這里我們可以看到一個區別就是之前我自己寫的代碼,host 都是 localhost, 但是這里寫的是 0.0.0.0。這兩個有什么區別呢?

其實區別很簡單,比如我之前寫的 localhost 綁定的ip 是 127.0.0.1, 這個相當于一個回環地址,那么我們就只允許本機的IP進行訪問。而 0.0.0.0,則表示我們對ip不進行限制,所有的IP都可以進行訪問。

那我們接著再來看看項目根目錄下面的server.php:

/**
 * Laravel - A PHP Framework For Web Artisans
 *
 * @package  Laravel
 * @author   Taylor Otwell 
 */

$uri = urldecode(
    parse_url($_SERVER["REQUEST_URI"], PHP_URL_PATH)
);

// 這個文件允許我們從內置 PHP web 服務器中模擬 Apache 的 "mod_rewrite" 功能.
// 這提供了一種測試 Laravel 應用程序的便捷方法,
// 而無需在此安裝"真正的" web 服務器軟件。
if ($uri !== "/" && file_exists(__DIR__."/public".$uri)) {
    return false;
}

require_once __DIR__."/public/index.php";

發現跟我之前寫的路由文件相同。沒錯,我就是從這里抄過來的。

基本上 Larvel 的實現方法就是這樣了。

Symfony

如果你在使用 Symfony 框架話,發現Symfony有一個組件叫做web-server-bundle,這個組件的作用跟Laravel相同,也是不借助web服務器,實現通過瀏覽器訪問應用程序。

基本的操作可以參考該頁面

我在這里主要說一下Symfony是如何實現的.

在Symfony中有一段代碼是這樣的:

public function start(WebServerConfig $config, $pidFile = null)
{
    // 獲取默認的PID文件位置
    $pidFile = $pidFile ?: $this->getDefaultPidFile();

    // 判斷是否在運行,如果運行則提示已經在監聽了
    if ($this->isRunning($pidFile)) {
        throw new RuntimeException(sprintf("A process is already listening on http://%s.", $config->getAddress()));
    }

    // fork了一個子進程,如果成功,會有兩個進程進行同時執行下面的文件,父進程,也就是當前執行的進程會返回子進程的PID,而子進程則返回的PID為0,
    // 如果失敗,則子進程不會創建,并且父進程會返回的pid為-1。更多內容可查看 https://www.php.net/manual/zh/function.pcntl-fork.php
    $pid = pcntl_fork();

    // 表示fork進程失敗
    if ($pid < 0) {
        throw new RuntimeException("Unable to start the server process.");
    }

    // 進入這個判斷,表示執行的是父進程,表示不用繼續向下執行
    if ($pid > 0) {
        return self::STARTED;
    }

    // 從此往后是子進程運行,首先通過 posix_setsid 變為守護進程,意思是使其脫離終端的管理,自立門戶,誰也沒辦法管理這個進程,除了PID。
    if (posix_setsid() < 0) {
        throw new RuntimeException("Unable to set the child process as session leader.");
    }

    // 創建命令,命令類似Laravel,不過這里的路由文件跟Laravel類似。也是處理加載規則,并加載入口文件。具體的router.php 路徑為:
    // vendorsymfonyweb-server-bundle/Resources/router.php
    // 下面是禁用輸出并且開始運行
    $process = $this->createServerProcess($config);
    $process->disableOutput();
    $process->start();

    // 判斷是否運行成功
    if (!$process->isRunning()) {
        throw new RuntimeException("Unable to start the server process.");
    }

    // 寫入PID文件
    file_put_contents($pidFile, $config->getAddress());

    // 檢測PID文件,如果PID文件刪除了,那么進程就立即退出。
    while ($process->isRunning()) {
        if (!file_exists($pidFile)) {
            $process->stop();
        }

        sleep(1);
    }

    // 返回停止的狀態
    return self::STOPPED;
}

/**
 * 啟動PHP內置web服務器
 * @return Process The process
 */
private function createServerProcess(WebServerConfig $config)
{
    // 查找PHP的可執行程序
    $finder = new PhpExecutableFinder();
    if (false === $binary = $finder->find(false)) {
        throw new RuntimeException("Unable to find the PHP binary.");
    }

    $xdebugArgs = ini_get("xdebug.profiler_enable_trigger") ? ["-dxdebug.profiler_enable_trigger=1"] : [];

    // 實例化PHP要執行的命令 php_path -dvariables_order=EGPCS -S 127.0.0.1:8000 vendorsymfonyweb-server-bundle/Resources/router.php
    $process = new Process(array_merge([$binary], $finder->findArguments(), $xdebugArgs, ["-dvariables_order=EGPCS", "-S", $config->getAddress(), $config->getRouter()]));
    // 設置工作目錄
    $process->setWorkingDirectory($config->getDocumentRoot());
    // 設置超時時間
    $process->setTimeout(null);

    // 設置環境變量
    if (in_array("APP_ENV", explode(",", getenv("SYMFONY_DOTENV_VARS")))) {
        $process->setEnv(["APP_ENV" => false]);
        $process->inheritEnvironmentVariables();
    }

    // 返回相關變量
    return $process;
}

我在上面的代碼中進行了注釋, 描述了Symfony是如何啟動的.

里面有一個問題就是使用pcntl_fork, 該擴展在Windows中是不受支持的. 所以 Symfony框架會提示使用php bin/console server:run命令運行程序.

未來展望

其實還有一個方式, 就是 Workman 是通過自身的實現的web服務器,它并沒有借助php -S命令。這一塊的代碼我還沒有吃透,并且我覺得這個也可以多帶帶拎幾章出來講。希望以后有這個機會。

總結

通過我們學習 PHP 命令實現web服務器訪問以及對 Laravel 和 Symfony 框架的分析, 讓我了解到在Windows的開發過程中,我們完全可以借助該方式來擺脫對web服務器的依賴.既能方便我們在Windows環境進行開發并且學習了PHP一個技巧.感覺挺好的.

大家如果對此有什么疑問可以評論進行交流.

參考

PHP: 內置Web Server - Manual

Laravel

How to Use PHP"s built-in Web Server

PHP: pcntl_fork - Manual

PHP: posix_setsid - Manual

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/31529.html

相關文章

  • 【modernPHP專題(14)】內置的http務器

    摘要:起就在中內置了服務器但只是提供開發測試使用,不推薦使用中生產環境中。因為這個服務器接受處理請求時順序執行的,不能并發處理。這個內置的服務器使用起來非常的方便,你只需要執行下面的命令啟動服務器然后就可以訪問了。 PHP 5.4起就在CLI SAPI中內置了web服務器,但只是提供開發測試使用,不推薦使用中生產環境中。因為這個服務器接受處理請求時順序執行的,不能并發處理。 這個內置的web...

    RancherLabs 評論0 收藏0
  • PHP新特性字節碼緩存和內置務器

    摘要:從開始,內置了字節碼緩存功能,名為。因為是解釋性語言,解釋器執行腳本時會解析腳本代碼,生成一系列的操作碼,然后執行字節碼,每次的請求都是這樣,會消耗很多資源,使用字節碼緩存可以緩存預先編譯的字節碼,減少響應時間,降低系統資源的壓力。 Zend OPcache 1).從PHP5.0開始,內置了字節碼緩存功能,名為Zend OPcache。因為PHP是解釋性語言,PHP解釋器執行PHP腳本...

    lolomaco 評論0 收藏0
  • Swoft 系列教程:(1)使用 Docker 安裝部署 Swoft

    摘要:即異步非阻塞,,事件驅動。優雅的注解聲明,容器,嚴格遵循規范。鏡像的主要用途官方提供了基于的鏡像。鏡像中已安裝配置好運行的所需組件及依賴。修改鏡像的使得容器啟動時不同時啟動服務,這就不需要要求我們掛載的本地項目必須完全安裝好依賴了。 之前有寫過一篇 Docker 安裝部署 Swoft 的文章,但有些冗余混亂,故重寫作為教程的開篇。要不讀讀看? Swoft項目:https://gith...

    宋華 評論0 收藏0
  • Swoft 系列教程:(1)使用 Docker 安裝部署 Swoft

    摘要:即異步非阻塞,,事件驅動。優雅的注解聲明,容器,嚴格遵循規范。鏡像的主要用途官方提供了基于的鏡像。鏡像中已安裝配置好運行的所需組件及依賴。修改鏡像的使得容器啟動時不同時啟動服務,這就不需要要求我們掛載的本地項目必須完全安裝好依賴了。 之前有寫過一篇 Docker 安裝部署 Swoft 的文章,但有些冗余混亂,故重寫作為教程的開篇。要不讀讀看? Swoft項目:https://gith...

    psychola 評論0 收藏0
  • PHP 自 5.2 到 5.6 中新增的功能詳解

    摘要:本文將會介紹自起,直至中增加的新特征。棄用的功能以下幾個功能被棄用,若在配置文件中啟用,則會在運行時發出警告。該類的完整限定名是其中第一個反斜杠表示全局命名空間。目的在于模仿的不對,目的是為了讓發布應用程序更加方便。 截至目前(2014.2), PHP 的最新穩定版本是 PHP5.5, 但有差不多一半的用戶仍在使用已經不在維護 [注] 的 PHP5.2, 其余的一半用戶在使用 PHP5...

    番茄西紅柿 評論0 收藏0

發表評論

0條評論

最新活動
閱讀需要支付1元查看
<