<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    一江春水向東流

    做一個有思想的人,期待與每一位熱愛思考的人交流,您的關(guān)注是對我最大的支持。

      BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
      44 隨筆 :: 139 文章 :: 81 評論 :: 0 Trackbacks
    ?在網(wǎng)絡(luò)程序里面,一般的來說都是許多客戶機對應(yīng)一個服務(wù)器.為了處理客戶機的請求, 對服務(wù)端的程序就提出了特殊的要求.我們學習一下目前最常用的服務(wù)器模型.

    循環(huán)服務(wù)器:循環(huán)服務(wù)器在同一個時刻只可以響應(yīng)一個客戶端的請求

    并發(fā)服務(wù)器:并發(fā)服務(wù)器在同一個時刻可以響應(yīng)多個客戶端的請求


    9.1 循環(huán)服務(wù)器:

    UDP服務(wù)器
    UDP循環(huán)服務(wù)器的實現(xiàn)非常簡單:
    UDP服務(wù)器每次從套接字上讀取一個客戶端的請求,處理, 然后將結(jié)果返回給客戶機.

    可以用下面的算法來實現(xiàn).

    ?? ?socket(...);
    ?? ?bind(...);
    ?? ?while(1)
    ?? ?{
    ?? ??? ?recvfrom(...);
    ?? ??? ?process(...);
    ?? ??? ?sendto(...);
    ?? ?}

    因為UDP是非面向連接的,沒有一個客戶端可以老是占住服務(wù)端. 只要處理過程不是死循環(huán), 服務(wù)器對于每一個客戶機的請求總是能夠滿足.

    9.2 循環(huán)服務(wù)器:

    TCP服務(wù)器
    TCP循環(huán)服務(wù)器的實現(xiàn)也不難:TCP服務(wù)器接受一個客戶端的連接,然后處理,完成了這個客戶的所有請求后,斷開連接.

    算法如下:

    socket(...);
    bind(...);
    listen(...);

    while(1)
    {
    ?? ?accept(...);
    ?? ?while(1)
    ?? ?{
    ?? ??? ?read(...);
    ?? ??? ?process(...);
    ?? ??? ?write(...);
    ?? ?}
    ?? ?close(...);
    }

    TCP循環(huán)服務(wù)器一次只能處理一個客戶端的請求.只有在這個客戶的所有請求都滿足后, 服務(wù)器才可以繼續(xù)后面的請求.這樣如果有一個客戶端占住服務(wù)器不放時,其它的客戶機都不能工作了.因此,TCP服務(wù)器一般很少用循環(huán)服務(wù)器模型的.

    9.3 并發(fā)服務(wù)器:???? TCP服務(wù)器
    為了彌補循環(huán)TCP服務(wù)器的缺陷,人們又想出了并發(fā)服務(wù)器的模型. 并發(fā)服務(wù)器的思想是每一個客戶機的請求并不由服務(wù)器直接處理,而是服務(wù)器創(chuàng)建一個 子進程來處理.

    算法如下:

    ?? ?socket(...);
    ?? ?bind(...);
    ?? ?listen(...);
    ?? ?while(1)
    ?? ?{
    ?? ??? ?accept(...);
    ?? ??? ?if(fork(..)==0)
    ?? ??? ?{
    ?? ??? ??? ??? ?while(1)
    ?? ??? ??? ??? ?{
    ?? ??? ??? ??? ??? ?read(...);
    ?? ??? ??? ??? ??? ?process(...);
    ?? ??? ??? ??? ??? ?write(...);
    ?? ??? ??? ??? ?}
    ?? ??? ??? ??? ?close(...);
    ?? ??? ??? ??? ?exit(...);
    ?? ??? ?}
    ?? ?close(...);
    ?? ?}

    TCP并發(fā)服務(wù)器可以解決TCP循環(huán)服務(wù)器客戶機獨占服務(wù)器的情況. 不過也同時帶來了一個不小的問題.
    為了響應(yīng)客戶機的請求,服務(wù)器要創(chuàng)建子進程來處理. 而創(chuàng)建子進程是一種非常消耗資源的操作.

    9.4 并發(fā)服務(wù)器:

    多路復(fù)用I/O
    為了解決創(chuàng)建子進程帶來的系統(tǒng)資源消耗,人們又想出了多路復(fù)用I/O模型.

    首先介紹一個函數(shù)select

    int select(int nfds,
    ?? ??? ?fd_set *readfds,fd_set *writefds,
    ?? ?? fd_set *except fds,
    ?? ?? struct timeval *timeout
    ?? ?? )

    void FD_SET(int fd,fd_set *fdset)

    void FD_CLR(int fd,fd_set *fdset)

    void FD_ZERO(fd_set *fdset)

    int FD_ISSET(int fd,fd_set *fdset)

    一般的來說當我們在向文件讀寫時,進程有可能在讀寫出阻塞,直到一定的條件滿足.

    ?比如我們從一個套接字讀數(shù)據(jù)時,可能緩沖區(qū)里面沒有數(shù)據(jù)可讀(通信的對方還沒有發(fā)送數(shù)據(jù)過來),
    ?這個時候我們的讀調(diào)用就會等待(阻塞)直到有數(shù)據(jù)可讀.如果我們不 希望阻塞,
    ?我們的一個選擇是用select系統(tǒng)調(diào)用.
    ?
    ?只要我們設(shè)置好select的各個參數(shù),那么當文件可以讀寫的時候select回"通知"我們 說可以讀寫了.
    ?readfds所有要讀的文件? 文件描述符的集合
    writefds所有要寫的文件?? 文件描述符的集合

    exceptfds其他的服要向我們通知的文件描述符

    timeout超時設(shè)置.

    nfds所有我們監(jiān)控的文件描述符中最大的那一個加1

    在我們調(diào)用select時進程會一直阻塞直到以下的一種情況發(fā)生.
    1)有文件可以讀.
    2)有文件可以寫.
    3)超時所設(shè)置的時間到.

    為了設(shè)置文件描述符我們要使用幾個宏. FD_SET將fd加入到fdset

    FD_CLR將fd從fdset里面清除

    FD_ZERO從fdset中清除所有的文件描述符

    FD_ISSET判斷fd是否在fdset集合中

    使用select的一個例子

    int use_select(int *readfd,int n)
    {
    ??? ??? fd_set my_readfd;
    ??? ??? int maxfd;
    ??? ??? int i;
    ??? ???
    ??? ??? maxfd=readfd[0];
    ??? ???
    ??? ??? for(i=1;i<n;i++)
    ??? ??? {
    ??? ??? ??? if(readfd[i]>maxfd) maxfd=readfd[i];
    ??? ??? }
    ??? ???
    ??? ???
    ??? ???
    ??? ??? while(1)
    ??? ??? {
    ??? ??? ??? /* 將所有的文件描述符加入 */
    ??? ??? ??? FD_ZERO(&my_readfd);
    ??? ??? ???
    ??? ??? ??? for(i=0;i<n;i++)
    ??? ??? ??? FD_SET(readfd[i],*my_readfd);
    ??? ??? ???
    ??? ??? ??? /* 進程阻塞 */
    ??? ??? ??? select(maxfd+1,& my_readfd,NULL,NULL,NULL);
    ??? ??? ???
    ??? ??? ??? /* 有東西可以讀了 */
    ??? ??? ??? for(i=0;i<n;i++)
    ??? ??? ??? {
    ??? ??? ??? ??? if(FD_ISSET(readfd[i],&my_readfd))
    ??? ??? ??? ??? {
    ??? ??? ??? ??? ??? ??? /* 原來是我可以讀了 */
    ??? ??? ??? ??? ??? ??? we_read(readfd[i]);
    ??? ??? ??? ??? }
    ??? ??? ??? }
    ??? ??? }
    }

    使用select后我們的服務(wù)器程序就變成了.


    初始話(socket,bind,listen);

    while(1)
    {
    ??? ??? 設(shè)置監(jiān)聽讀寫文件描述符(FD_*);
    ??? ???
    ??? ??? 調(diào)用select;
    ??? ???
    ??? ??? 如果是傾聽套接字就緒,說明一個新的連接請求建立
    ??? ??? {
    ??? ??? ??? 建立連接(accept);
    ??? ??? ??? 加入到監(jiān)聽文件描述符中去;
    ??? ??? }
    ??? ??? 否則說明是一個已經(jīng)連接過的描述符
    ??? ??? {
    ??? ??? ??? 進行操作(read或者write);
    ??? ??? }

    }

    多路復(fù)用I/O可以解決資源限制的問題.著模型實際上是將UDP循環(huán)模型用在了TCP上面. 這也就帶來了一些問題.如由于服務(wù)器依次處理客戶的請求,所以可能會導(dǎo)致有的客戶 會等待很久.

    ?并發(fā)服務(wù)器:??????? UDP服務(wù)器
    人們把并發(fā)的概念用于UDP就得到了并發(fā)UDP服務(wù)器模型.
    ?并發(fā)UDP服務(wù)器模型其實是簡單的.和并發(fā)的TCP服務(wù)器模型一樣是
    ?創(chuàng)建一個子進程來處理的 算法和并發(fā)的TCP模型一樣.

    除非服務(wù)器在處理客戶端的請求所用的時間比較長以外,人們實際上很少用這種模型.
    posted on 2008-03-29 20:14 allic 閱讀(922) 評論(0)  編輯  收藏 所屬分類: TCP/IP 開發(fā)
    主站蜘蛛池模板: GOGOGO免费观看国语| 亚洲狠狠色丁香婷婷综合| 国产精品无码永久免费888| 日本免费网站在线观看| 亚洲国产精品18久久久久久| 好爽又高潮了毛片免费下载| 亚洲精品天堂成人片AV在线播放 | 亚洲av成人无码网站…| 午夜视频在线在免费| 日本亚洲欧美色视频在线播放| 免费人成视网站在线观看不卡| 一区二区三区AV高清免费波多| 亚洲午夜av影院| 日韩免费电影网址| 亚洲人成激情在线播放| 午夜爱爱免费视频| 一级做a爰黑人又硬又粗免费看51社区国产精品视| 夜色阁亚洲一区二区三区| 中文字幕不卡免费视频| 亚洲一级二级三级不卡| 最近高清国语中文在线观看免费| 亚洲国产精品ⅴa在线观看| 国产精品亚洲精品日韩已方 | 91免费在线视频| 亚洲综合精品香蕉久久网97| 在线看片无码永久免费视频| 亚洲AV无码专区在线观看成人| 亚洲麻豆精品国偷自产在线91| 日韩在线不卡免费视频一区| 中文字幕乱码亚洲精品一区| 亚洲精品人成无码中文毛片 | 曰韩亚洲av人人夜夜澡人人爽| 97在线视频免费播放| 亚洲av无码专区青青草原| 最新精品亚洲成a人在线观看| 18禁止观看免费私人影院| 黄色一级视频免费| 久久亚洲精品无码aⅴ大香| 国产成人免费a在线资源| 午夜理伦剧场免费| 看Aⅴ免费毛片手机播放|