fcgi原理和異步模型

1. 介紹

由于CGI解釋器的反復(fù)加載會使CGI性能低下秩彤,F(xiàn)astCGI可以將CGI解釋器保持在內(nèi)存中嘹叫, 提高性能
相關(guān)地址:https://fastcgi-archives.github.io

2. Fastcgi特點:

  • 性能
  • 簡單迷雪,容易移植
  • 語言無關(guān)
  • 進程隔離
  • 通用性碘勉,主流WebServer都支持,nginx奇瘦、apache都支持
  • 支持分布式計算棘催,F(xiàn)astCGI提供遠程運行應(yīng)用程序的功能,這對于分發(fā)負載和管理外部Web站點非常有用(這點待驗證是否好用耳标,我希望的能支持異步處理)

3. 原理

  1. 簡單來說醇坝,利用后面說的spawn-fcgi(fastcgi的管理程序)啟動fastcgi,支持啟動多實例次坡,并通過socket進行數(shù)據(jù)綁定
  2. WebServer(如:nginx)在收到請求后呼猪,根據(jù)配置中配置要轉(zhuǎn)發(fā)的fastcgi地址,發(fā)到固定的地址砸琅,例如:fastcgi綁定的是127.0.0.1:8088
  3. fastcgi Accept并處理數(shù)據(jù)后宋距,將響應(yīng)數(shù)據(jù)通過Socket返回給WebServer,并斷開連接症脂,這時表示一次處理完成

管理程序 -- spawn-fcgi

下載地址:http://redmine.lighttpd.net/projects/spawn-fcgi/wiki
-f 指定調(diào)用 FastCGI的進程的執(zhí)行程序位置
-a 綁定到 地址addr乡革。
-p 綁定到 端口 port。
-s 綁定到 unix domain socket
-C 指定產(chǎn) 生的FastCGI的進程數(shù)摊腋, 默認為 5沸版。 ( 僅用 于PHP)
-P 指定產(chǎn) 生的進程的PID文件路徑。
-F 指定產(chǎn) 生的FastCGI的進程數(shù)( C的CGI用 這個)
例子:./spawn-fcgi -a 127.0.0.1 -p 8088 -F 500 -f cgi(啟動500個cgi程序兴蒸,監(jiān)聽的端口為8088视粮,綁定地址為127.0.0.1)

4. nginx如何配置

  1. fastcgi_pass
    fastcgi_pass address;
    address為后端的fastcgi server的地址
    可用位置:location,if in location
  2. fastcgi_index
    fastcgi_index name;
    fastcgi默認的主頁資源
    示例:fastcgi_index index.php;
  3. fastcgi_param
    fastcgi_param parameter value [if_not_empty];
    設(shè)置傳遞給FastCGI服務(wù)器的參數(shù)值,可以是文本橙凳,變量或組合

5. 阻塞與非阻塞模式

fastcgi示例 -- 阻塞模式

#include "fcgi_stdio.h"
#include <stdlib.h>

void main(void)
{
    int count = 0;
    while(FCGI_Accept() >= 0)
        printf("Content-type: text/html\r\n"
               "\r\n"
               "<title>FastCGI Hello!</title>"
               "<h1>FastCGI Hello!</h1>"
               "Request number %d running on host <i>%s</i>\n",
                ++count, getenv("SERVER_NAME"));
}

fastcgi示例 -- 生產(chǎn)者蕾殴,消費者模式(可實現(xiàn)非阻塞)

非阻塞的實現(xiàn),將會話狀態(tài)與requests進行binding即可

ZEN_Message_Queue_Deque<ZEN_MT_SYNCH, uint32_t> t_que(100000);

// 這步非常關(guān)鍵岛啸,因為FCGX_Accept_r中的數(shù)據(jù)流綁定是直接和request的ptr進行綁定的
// 所以在未處理完數(shù)據(jù)的前提下钓觉,requests的生命周期要和會話的生命周期一致
FCGX_Request requests[1000];
uint32_t use_idx[1000];

void *do_session(void *arg)
{
    int ret = 0;

    while (1)
    {
        uint32_t idx = 0;
        ret = t_que.try_dequeue(idx);

        if (ret != 0)
        {
            // no data sleep 1ms
            usleep(1000);
            continue;
        }

        FCGX_Request &request = requests[idx];
        std::string out = "Content-type:application/json\r\n\r\n";
        Json::Value root;
        root["ret"] = 1000;
        root["t_id"] = (int)gettid();
        out.append(root.toStyledString());
        FCGX_FPrintF(request.out, out.c_str());
        FCGX_Finish_r(&request);
        use_idx[idx] = 0;
    }

    return NULL;
}

int get_free()
{
    static uint32_t idx = 0;

    for (; idx < 1000; idx ++)
    {
        if (use_idx[idx] == 0)
        {
            use_idx[idx] = 1;
            return idx;
        }
    }

    return -1;
}

int main(int argc, char **argv)
{
    pthread_t pthread_id;
    int iThreadNum = 10;

    for (int index = 0; index != iThreadNum; ++ index)
    {
       pthread_create(&pthread_id, NULL, do_session, NULL);
    }

    memset(use_idx, 0, sizeof(use_idx));
    int ret = FCGX_Init();

    if (ret != 0)
    {
        printf("init fail\n");
        return -1;
    }

    while (1)
    {
        int idx = get_free();

        if (idx < 0)
        {
            continue;
        }

        FCGX_Request &request = requests[idx];
        ret = FCGX_InitRequest(&request, 0, 0);

        if (ret != 0)
        {
            printf("init fail\n");
            return -1;
        }

        ret = FCGX_Accept_r(&request);

        if (ret < 0)
        {
            continue;
        }

        t_que.enqueue(idx);
    }

    return 0;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市坚踩,隨后出現(xiàn)的幾起案子荡灾,更是在濱河造成了極大的恐慌,老刑警劉巖瞬铸,帶你破解...
    沈念sama閱讀 212,718評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件批幌,死亡現(xiàn)場離奇詭異,居然都是意外死亡嗓节,警方通過查閱死者的電腦和手機荧缘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評論 3 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來拦宣,“玉大人截粗,你說我怎么就攤上這事信姓。” “怎么了绸罗?”我有些...
    開封第一講書人閱讀 158,207評論 0 348
  • 文/不壞的土叔 我叫張陵意推,是天一觀的道長。 經(jīng)常有香客問我从诲,道長左痢,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,755評論 1 284
  • 正文 為了忘掉前任系洛,我火速辦了婚禮俊性,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘描扯。我一直安慰自己定页,他們只是感情好,可當我...
    茶點故事閱讀 65,862評論 6 386
  • 文/花漫 我一把揭開白布绽诚。 她就那樣靜靜地躺著典徊,像睡著了一般。 火紅的嫁衣襯著肌膚如雪恩够。 梳的紋絲不亂的頭發(fā)上卒落,一...
    開封第一講書人閱讀 50,050評論 1 291
  • 那天,我揣著相機與錄音蜂桶,去河邊找鬼儡毕。 笑死,一個胖子當著我的面吹牛扑媚,可吹牛的內(nèi)容都是我干的腰湾。 我是一名探鬼主播,決...
    沈念sama閱讀 39,136評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼疆股,長吁一口氣:“原來是場噩夢啊……” “哼费坊!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起旬痹,我...
    開封第一講書人閱讀 37,882評論 0 268
  • 序言:老撾萬榮一對情侶失蹤附井,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后唱凯,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體羡忘,經(jīng)...
    沈念sama閱讀 44,330評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,651評論 2 327
  • 正文 我和宋清朗相戀三年磕昼,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片节猿。...
    茶點故事閱讀 38,789評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡票从,死狀恐怖漫雕,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情峰鄙,我是刑警寧澤浸间,帶...
    沈念sama閱讀 34,477評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站吟榴,受9級特大地震影響魁蒜,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜吩翻,卻給世界環(huán)境...
    茶點故事閱讀 40,135評論 3 317
  • 文/蒙蒙 一兜看、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧狭瞎,春花似錦细移、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,864評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至碗殷,卻和暖如春精绎,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背锌妻。 一陣腳步聲響...
    開封第一講書人閱讀 32,099評論 1 267
  • 我被黑心中介騙來泰國打工代乃, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人从祝。 一個月前我還...
    沈念sama閱讀 46,598評論 2 362
  • 正文 我出身青樓襟己,卻偏偏與公主長得像,于是被迫代替她去往敵國和親牍陌。 傳聞我的和親對象是個殘疾皇子擎浴,可洞房花燭夜當晚...
    茶點故事閱讀 43,697評論 2 351

推薦閱讀更多精彩內(nèi)容