• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            公告

            <2025年6月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            293012345

            統(tǒng)計(jì)

            • 隨筆 - 9
            • 文章 - 13
            • 評(píng)論 - 3
            • 引用 - 0

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            Perl===網(wǎng)絡(luò)編程入門聊天室
            原作者:Brian Slesinsky 1997年5月7日
            編譯者:【Perl之旅】Nighthawk 2000年7月15日
            Brian Slesinsky原來是HotWired公司的工程師,后來他離開公司忙于自己事業(yè)。

            前言:

            我對(duì)在線聊天沒有什么興趣,說是實(shí)在的,與電子郵件和網(wǎng)絡(luò)會(huì)議系統(tǒng)相比,聊天室顯得很膚淺.但是寫一個(gè)聊天室服務(wù)程序倒是一件很有意思的事情.我將告訴你如何來寫一個(gè)小型的聊天室服務(wù)程序,可能會(huì)很簡陋,有很多要擴(kuò)展的地方.

            先決條件:

            你必須有很好的Perl編程的知識(shí),一臺(tái)服務(wù)器,安裝Perl 5.002或更高的版本.注意大多數(shù)ISP不會(huì)允許普通用戶運(yùn)行聊天室程序.但是你也許可以通過一個(gè)MODEN連接來與少數(shù)幾個(gè)用戶試試你的聊天室系統(tǒng). (如果你從CPAN獲得了最新版本的IO:Select,這個(gè)聊天室程序可以在Windows環(huán)境下使用).

            你還需要一個(gè)telnet客戶端程序,因?yàn)槲覀円脕碜隽奶焓业目蛻舳?



            Socket簡易編程:

            開始聊天,你需要在internet上建立一個(gè)連接,對(duì)Perl程序員來說,這意味著要和socket打交道.而以前這是很困難的,因?yàn)槟悴坏貌皇褂胮ack()來建立一個(gè)C結(jié)構(gòu)來進(jìn)行底層的系統(tǒng)調(diào)用.但在最新版的Perl中我們可以使用IO::Socket包,很容易地打開一個(gè)socket. 當(dāng)用戶連接聊天服務(wù)器時(shí),telnet程序在指定的端口打開一個(gè)連接,所以服務(wù)器也必須在那個(gè)端口打開一個(gè)socket,監(jiān)聽所有進(jìn)來的連接.下面如何通過IO::Socket來做到這一點(diǎn):

            ????use IO::Socket;

            ????my $listening_socket =
            ????????IO::Socket::INET->new(Proto => 'tcp',
            ??????????????????????????????LocalPort => 2323,
            ??????????????????????????????Listen => 1,
            ??????????????????????????????Reuse => 1) or die $!;

            所有參量的含義:

            Proto: 定義網(wǎng)絡(luò)所用的協(xié)議 - 在這里我們用的是TCP. 在internet上通常有兩種協(xié)議用得比較廣泛 - TCP 和 UDP. TCP適用于穩(wěn)定的連接,可以重新發(fā)送丟失的數(shù)據(jù)包,而UDP用于那些不用重發(fā)數(shù)據(jù)包的場合(如實(shí)時(shí)音頻數(shù)據(jù)流).

            LocalPort: 定義連接的端口號(hào).

            Listen: 我們將監(jiān)聽來自其它計(jì)算機(jī)的連接,而不是自己建立一個(gè)連接.所以用戶要先telnet到端口2323,然后運(yùn)行了聊天服務(wù)程序的計(jì)算機(jī)來建立連接.

            Reuse: 這個(gè)選項(xiàng)意思是如果我們"殺掉"聊天服務(wù)程序然后再重新啟動(dòng),將能夠馬上重新使用原來的端口,而不用等待以前那個(gè)連接完全結(jié)束.



            我們正等待某個(gè)連接的到來.... 一個(gè)連接到來以后,我們需要accept這個(gè)新的連接:

            $socket = $listening_socket->accept;

            一旦我們建立了一個(gè)連接,我們可以發(fā)送一些文字給這個(gè)用戶(還不完全是,請(qǐng)看本文的結(jié)尾部分):

            $socket->send("hello\r\n") or print "connection closed at other end\n";

            我們也可以接收用戶發(fā)來的信息:

            $socket->recv($line, 80);
            if($line eq "") {
            print "connection closed at other end\n";
            }

            最后我們完成了連接,可以關(guān)閉它:

            $socket->close;

            大部分程序只在一個(gè)時(shí)刻處理一個(gè)用戶.如果用戶還沒有準(zhǔn)備好,程序就沒有什么好做的.所以Perl程序沒有從讀到什么東西,它就停下來等待直到用戶準(zhǔn)備好. (這叫blocking I/O.)

            這種方式不能用于聊天服務(wù)程序,用戶不可能排著隊(duì)來.一個(gè)用戶可能離開去喝些咖啡,但其它用戶還在拼命地敲打鍵盤(聊天),服務(wù)程序還得處理他們的信息.

            解決這個(gè)問題的一個(gè)辦法是為每個(gè)用戶創(chuàng)建一個(gè)入口(entity),或者用fork()創(chuàng)建另外一個(gè)進(jìn)程,或者用多線程編程方法(遺憾地是Perl還用不了).這樣系統(tǒng)就可以為多個(gè)用戶服務(wù), 但每個(gè)用戶有他自己的入口(entity)等待他輸入命令. 但是進(jìn)程的系統(tǒng)開銷比較大,如果很多用戶登錄的話,系統(tǒng)資源很快會(huì)變得不足.最好是用一個(gè)進(jìn)程來處理所有人的請(qǐng)求.

            我們真正需要的是要知道誰正在等待服務(wù),必須馬上處理(除非沒有一個(gè)人想聊天).這就是select()函數(shù)所要做的.

            象socket函數(shù)一樣,select()曾經(jīng)也是很難用,所以大多數(shù)程序員都盡量避免使用它. 但Perl給它加了一個(gè)面向?qū)ο缶幊痰陌b,叫做IO::Select,使得使用非常簡單.

            假設(shè)我們要等待兩個(gè)sockets, $thing1 and $thing2. 首先我們創(chuàng)建一個(gè)包含兩個(gè)socket的select()對(duì)象:

            $select = IO::Select->new($thing1,$thing2);

            下一步,當(dāng)我們需要知道誰有數(shù)據(jù)要處理時(shí),我們就查詢select對(duì)象:

            my @ready = $select->can_read;

            這個(gè)調(diào)用將等待直到$thing1或$thing2中任何一個(gè)準(zhǔn)備好, 它將返回一個(gè)包含socket的數(shù)組. (如果它們都準(zhǔn)備好了,@ready將包含兩個(gè)socket.) 一旦有了準(zhǔn)備好的socket, 我們一個(gè)一個(gè)地讀取數(shù)據(jù)找出它們發(fā)送的是是什么:

            ?? for $socket (@ready) {
            ????????$socket->recv($line,80);
            ????????if($line eq "") { die "they hung up on me"; }
            ????????print "someone sent $line.??Sending it back.\n";
            ????????$socket->send($line) or die "hey, where did they go?";
            ?? }

            現(xiàn)在我們有足夠的片段來寫我們的第一個(gè)聊天服務(wù)程序. 這個(gè)聊天室里的交談沒有什么意思,除非你中意和自己聊天 - 服務(wù)程序會(huì)把你說的全部回送. 但它將告訴你如果結(jié)合socket和select()來建立一個(gè)一個(gè)時(shí)刻只能做一件事的服務(wù)器.下面是程序源碼:

            #!/usr/local/bin/perl -wT
            require 5.002;
            use strict;
            use IO::Socket;
            use IO::Select;

            #創(chuàng)建一個(gè)socket然后監(jiān)聽一個(gè)端口
            my $listen = IO::Socket::INET->new(Proto => 'tcp',
            ?? LocalPort => 2323,
            ?? Listen => 1,
            ?? Reuse => 1) or die $!;

            # 開始$select只包含我們監(jiān)聽的socket
            my $select = IO::Select->new($listen);

            my @ready;

            #等待,直到有事情發(fā)生
            while(@ready = $select->can_read) {

            ????my $socket;

            ????# 處理每個(gè)準(zhǔn)備好了的socket
            ????for $socket (@ready) {

            # 如果被監(jiān)聽的socket準(zhǔn)備好了,接收一個(gè)新的連接
            if($socket == $listen) {
            ????my $new = $listen->accept;
            ????$select->add($new);
            ????print $new->fileno . ": connected\n";
            } else {

            ????# 否則讀入一行文字,然后發(fā)送回去
            ????my $line="";
            ????$socket->recv($line,80);
            ????$line ne "" and $socket->send($line) or do {

            # 如果沒有什么可發(fā)送和接收的,中斷連接
            print $socket->fileno . ": disconnected\n";
            $select->remove($socket);
            $socket->close;
            ????};
            }
            ????}
            }



            廣播:

            接下來的工作是把聊天信息發(fā)送給所有的用戶(不光是你自己),也就是所謂"廣播".

            我們可以用$select, 它new()或add()來返回所有給$select的sockets,從而得知"所有用戶"到底是誰.我們來修改下程序:


            ????????????$socket->recv($line,80);
            ????????????if($line eq "") {
            ????????????????print $socket->fileno . ": disconnected\n";
            ????????????????$select->remove($socket);
            ????????????????$socket->close;
            ????????????};
            ????????????my $socket;

            ????????????# 向所有用戶廣播.如果send()失敗了就關(guān)閉連接.
            ????????????
            ????????????for $socket ($select->handles) {
            ????????????????next if($socket==$listen);
            ????????????????$socket->send($line) or do {
            ????????????????????print $socket->fileno . ": disconnected\n";????????
            ????????????????????$select->remove($socket);
            ????????????????????$socket->close;
            ????????????????};
            ????????????}

            下面是這個(gè)聊天程序的所有代碼:


            #!/usr/local/bin/perl -wT
            require 5.002;
            use strict;
            use IO::Socket;
            use IO::Select;

            #創(chuàng)建一個(gè)socket監(jiān)聽端口
            my $listen = IO::Socket::INET->new(Proto => 'tcp',
            ?? LocalPort => 2323,
            ?? Listen => 1,
            ?? Reuse => 1) or die $!;

            #$select只包含我們正在監(jiān)聽的socket
            my $select = IO::Select->new($listen);

            my @ready;

            # 等待
            while(@ready = $select->can_read) {

            ????my $socket;

            ????# 處理每個(gè)準(zhǔn)備好的端口
            ????for $socket (@ready) {

            # 如果被監(jiān)聽的端口準(zhǔn)備好,接收一個(gè)新的連接
            if($socket == $listen) {
            ????my $new = $listen->accept;
            ????$select->add($new);
            ????print $new->fileno . ": connected\n";
            } else {

            ????# 讀入一行文字
            ????# 如果recv()失敗,關(guān)閉連接
            ????my $line="";
            ????$socket->recv($line,80);
            ????if($line eq "") {
            print $socket->fileno . ": disconnected\n";
            $select->remove($socket);
            $socket->close;
            ????};
            ????my $socket;

            ????# 向所有人廣播,如果send()失敗則關(guān)閉連接.
            ????for $socket ($select->handles) {
            next if($socket==$listen);
            $socket->send($line) or do {
            ????print $socket->fileno . ": disconnected\n";
            ????$select->remove($socket);
            ????$socket->close;
            };
            ????}
            }
            ????}
            }

            1;
            ????????????


            我是誰?



            我們的聊天程序還有一個(gè)問題,就是我們不知道是誰在說話.真正的聊天室服務(wù)器能讓你知道誰是誰,在發(fā)言后面把他們的名字顯示出來.

            如果我們只能在一個(gè)時(shí)刻做一件事情,請(qǐng)求一個(gè)handle的較為直接的程序代碼就象這個(gè)樣子:

            ???????? my $new = $listen->accept;
            ????????????$select->add($new);
            ????????????print $new->fileno . ": connected\n";
            ????????????$new->write("choose a handle> ");
            ????????????$handle[$new->fileno] = $new->recv;
            ????????????

            問題是,我們不能要服務(wù)器停下來等待用戶輸入,我們需要把用戶在那里的信息保存下來,當(dāng)一個(gè)用戶在輸入的時(shí)候,可以處理其他用戶,當(dāng)這個(gè)用戶輸入完了以后在回來.完成這些功能的代碼可以分為兩部分:

            sub login {
            ????????????my($new) = @_;
            ????????????$select->add($new);
            ????????????print $new->fileno . ": connected\n";
            ????????????$new->write("choose a handle> ");
            ????????????save_where_we_are();
            ????????}

            ????????sub get_handle {
            ????????????my($socket) = @_;
            ????????????$handle[$socket->fileno] = $socket->recv;
            ????????}
            ????????
            #!/usr/local/bin/perl -wT
            require 5.002;
            use strict;
            use IO::Socket;
            use IO::Select;

            my $port = scalar(@ARGV)>0 ? $ARGV[0] : 2323;

            $| = 1;
            my $listen = IO::Socket::INET->new(Proto => 'tcp',
            ?? LocalPort => $port,
            ?? Listen => 1,
            ?? Reuse => 1) or die $!;
            $ENV{'PATH'} = "/usr/bin";
            my $date = `date`;
            warn "started on $port on $date";
            my $select = IO::Select->new($listen);
            my @chatters;

            # 在win32中,注釋掉下面這句
            $SIG{'PIPE'} = 'IGNORE';

            my @ready;
            while(@ready = $select->can_read) {
            ????print "going: ".join(', ',map {$_->fileno} @ready) . "\n";
            ????my $socket;
            ????for $socket (@ready) {
            if($socket == $listen) {
            ????my $new_socket = $listen->accept;
            ????Chatter->new($new_socket, $select, \@chatters);
            } else {
            ????my $chatter = $chatters[$socket->fileno];
            ????if(defined $chatter) {
            &{$chatter->nextsub}();
            ????} else {
            print "unknown chatter\n";
            ????}
            }
            ????}
            }

            package Chatter;
            use strict;

            sub new {
            ????my($class,$socket,$select,$chatters) = @_;

            ????my $self = {
            'socket' => $socket,
            'select' => $select,
            'chatters' => $chatters
            };
            ????bless $self,$class;

            ????$chatters->[$socket->fileno] = $self;
            ????$self->select->add($socket);

            ????$self->log("connected");
            ????$self->ask_for_handle;

            ????return $self;
            }

            sub socket { $_[0]->{'socket'} }
            sub select { $_[0]->{'select'} }
            sub chatters { $_[0]->{'chatters'} }
            sub handle { $_[0]->{'handle'} }
            sub nextsub { $_[0]->{'nextsub'} }

            sub ask_for_handle {
            ????my($self) = @_;
            ????my $welcome =<< END;

            歡迎你來到我的聊天室.

            使用指南:
            請(qǐng)注意這個(gè)聊天室程序不完全兼容telnet協(xié)議,所以有些telnet客戶端程序可能不工作,抱歉!
            如果你輸入的字符都分行顯示,請(qǐng)退出然后試一試其它的telnet客戶端程序,最好發(fā)一個(gè)電子郵件
            (bslesins-code\@hotwired.com)告訴我你用的是什么程序.

            我們已經(jīng)試過下面的客戶端程序,它們都能很好的工作:
            ??- "telnet" on Solaris
            ??- "telnet" on IRIX
            ??- CRT on Windows 95
            我們已經(jīng)收到報(bào)告,微軟的Telnet不能工作.

            另外,有些人登錄以后可能去干別的事情了,所以他們不會(huì)馬上看到你的信息.所以輸入以后,保持telnet
            窗口開著,等待一會(huì)兒.

            關(guān)閉你的telnet窗口就可以退出.或者假如你是在Unix命令行運(yùn)行telnet的話,按Control-]然后在提示中按"close"鍵.

            __Brian__

            END
            ????$welcome =~ s:\n:\r\n:g;
            ????$self->write($welcome);

            ????$self->write("choose a handle> ");

            ????$self->{'nextsub'} = sub { $self->get_handle };
            }

            sub get_handle {
            ????my($self) = @_;

            ????my $handle = $self->read or return;
            ????$handle =~ tr/ -~//cd;
            ????$self->{'handle'} = $handle;
            ????$self->broadcast("[$handle is here]");
            ????$self->log("handle: $handle");
            ????$self->{'nextsub'} = sub { $self->chat };
            }

            sub chat {
            ????my($self) = @_;

            ????my $line = $self->read;
            ????return if($line eq "");
            ????$line =~ tr/ -~//cd;
            ????my $handle = $self->handle;
            ????$self->broadcast("$handle> $line");
            }

            sub broadcast {
            ????my($self,$msg) = @_;

            ????my $socket;
            ????for $socket ($self->select->handles) {
            my $chatter = $self->chatters->[$socket->fileno];
            $chatter->write("$msg\r\n") if(defined $chatter);
            ????}
            }

            sub read {
            ????my($self) = @_;

            ????my $buf="";
            ????$self->socket->recv($buf,80);
            ????$self->leave if($buf eq "");
            ????return $buf;
            }

            sub write {
            ????my($self,$buf) = @_;
            ????$self->socket->send($buf) or $self->leave;
            }

            sub leave {
            ????my($self) = @_;

            ????print "leave called\n";

            ????$self->chatters->[$self->socket->fileno] = undef;
            ????$self->select->remove($self->socket);
            ????my $handle = $self->handle;
            ????$self->broadcast("[$handle left]") if(defined $handle);
            ????$self->log("disconnected");
            ????$self->socket->close;
            }

            sub log {
            ????my($self,$msg) = @_;
            ????my $fileno = $self->socket->fileno;
            ????print "$fileno: $msg\n";
            }

            __END__

            # and here's a chat server in 4 lines :-)

            #!/usr/local/bin/perl -- minchat: run and telnet to port 5555 - bslesins
            sub p{print@_}$SIG{CHLD}=sub{wait};socket S,2,2,6;bind S,pack(Snx12,2,5555);
            listen S,5;while(accept C,S){if(!fork){open(STDOUT,">&C");p"name:";$n=substr
            ,0,-2;$f=fork||exec"tail -f chatlog";open W,">>chatlog";select(W);$|=1;p
            "[$n here]\r\n";while(){p"$n> $_";}p"[$n gone]\r\n";kill 15,$f;exit}}



            如何保存用戶位置信息呢? 一個(gè)方法是保存一個(gè)子程序的指針,而這個(gè)子例程包含了下一步該做什么:

            $nextsub[$socket->fileno] = &get_handle;

            這樣我們就可以在@nextsub中適當(dāng)?shù)娜肟谡业轿覀兂霭l(fā)的位置. 綜合以上所述,我們把程序整理如下.





            剩下的工作:

            我們的聊天室程序還不是一個(gè)完整的作品,如果你象把它放在你的服務(wù)器上工作,還有許多事情要做.他們是:

            輸入緩沖區(qū): 關(guān)于recv()函數(shù),它并不總是每次接收一行數(shù)據(jù).一個(gè)真正的聊天服務(wù)器需要把recv()的結(jié)果添加到緩沖區(qū)中,并找到折行字符,把它分成幾行.

            輸出緩沖區(qū): 如果有人掛起它的telnet進(jìn)程太長時(shí)間,調(diào)用send()會(huì)中斷它.但可以用select()來發(fā)現(xiàn)一個(gè)socket是否已經(jīng)準(zhǔn)備好.

            更好地支持telnet協(xié)議

            加入常用的命令:幫助,列出在聊天室中的用戶名單,退出等等

            用戶賬號(hào)密碼保護(hù)

            多個(gè)聊天房間

            權(quán)限控制

            私人聊天房間

            等等...

            posted on 2006-09-25 15:28 blues 閱讀(416) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            天堂久久天堂AV色综合| 精品久久久久中文字| 色综合久久久久无码专区| 久久久久亚洲AV无码永不| 久久久综合九色合综国产| 久久se精品一区精品二区国产| 无夜精品久久久久久| 久久婷婷激情综合色综合俺也去| 精品综合久久久久久97超人| 久久婷婷五月综合成人D啪| 色综合久久无码五十路人妻| 久久99精品国产麻豆蜜芽| 亚洲乱码精品久久久久..| 久久er国产精品免费观看8| 色欲av伊人久久大香线蕉影院| 精品久久久久久国产三级| 日本久久久久亚洲中字幕| 国产叼嘿久久精品久久| 色8久久人人97超碰香蕉987| 久久久免费观成人影院| 久久国产精品-国产精品| 久久婷婷五月综合97色直播| 久久国产视屏| 狠狠久久综合伊人不卡| 久久香蕉国产线看观看99| 久久久亚洲欧洲日产国码aⅴ| 香蕉久久AⅤ一区二区三区| 99久久综合狠狠综合久久| 国内精品久久人妻互换| 激情伊人五月天久久综合 | 99久久精品毛片免费播放| 伊人久久综合无码成人网| 午夜精品久久久久久影视riav| 国产激情久久久久影院小草| 日本精品久久久久中文字幕8 | 中文精品久久久久国产网址| 久久99精品国产自在现线小黄鸭| 亚洲va久久久噜噜噜久久天堂| 久久久久久久97| 久久发布国产伦子伦精品 | 亚洲午夜无码AV毛片久久|