Chinaunix首页 | 论坛 | 博客
  • 博客访问: 7770298
  • 博文数量: 701
  • 博客积分: 2150
  • 博客等级: 上尉
  • 技术积分: 13233
  • 用 户 组: 普通用户
  • 注册时间: 2011-06-29 16:28
个人简介

天行健,君子以自强不息!

文章分类

全部博文(701)

文章存档

2019年(2)

2018年(12)

2017年(76)

2016年(120)

2015年(178)

2014年(129)

2013年(123)

2012年(61)

分类: PERL

2013-03-26 18:01:16

1. Problem
如何管理多个输入流
The next input to your program could be coming from any number of filehandles, 
but you don't know which. 
You've tried using select( ), 
but the need to then do unbuffered I/O is more than you can deal with 
(and it's making your code very difficult to follow).


2. Solution
Use the IO::Multiplex module from CPAN. 
It calls a mux_input( ) function when input is received over 
a socket, and handles input and output buffering for you:


use IO::Multiplex;
$mux = IO::Multiplex->new( );
$mux->add($FH1);
$mux->add($FH2); # ... and so on for all the filehandles to manage
$mux->set_callback_object(_ _PACKAGE_ _);  # or an object
$mux->Loop( );


sub mux_input {
  my ($package, $mux, $fh, $input) = @_;
  # $input is ref to the filehandle's input buffer
  # ...
}


3. Discussion
虽然可以使用select来管理多种输入,但是它有缺点, 如:
不能使用 <>    来读行,因为不知道client是否发送完;
不能使用 print 来写  ,因为buffer可能已满。
Although you can use select to manage input coming at you from multiple directions, 
there are many tricks and traps. 
For example, you can't use <> to read a line of input,
 because you never know whether the client has sent a full line yet 
(or will ever finish sending a line). 
You can't print to a socket without the risk of the output buffer being full and your process blocking. 
You need to use non-blocking I/O and maintain your own buffers, 
and, consequently, life rapidly becomes unmanageably complex.


该模块能管理文件句柄池
Fortunately, we have a way of hiding complexity: modules. 
The IO::Multiplex module from CPAN takes care of non-blocking I/O and select for you. 
You tell it which filehandles to watch, and it tells you when new data arrives. You can even print to the filehandles, 
and it'll buffer and non-blockingly output it. An IO::Multiplex object manages a pool of filehandles.


使用add方法添加句柄
Use the add method to tell IO::Multiplex to manage a filehandle. 
This enables non-blocking I/O and disables the stdio buffering. 
When IO::Multiplex receives data on one of its managed filehandles, 
it calls a mux_input method on an object or class of your choosing. 
Specify where mux_input is by passing a package name (if your callback is a class method) 
or object value (if your callback is an object method) to the IO::Multiplex set_callback_object method. 
In the example in the Solution, 
we pass in the current package name so that IO::Multiplex will call the current package's mux_input method.


可以自定义mux_input函数来处理输入数据
Your mux_input callback is called with four parameters: 
the object or package name that you gave to set_callback_object, 
the IO::Multiplex object that dispatched the callback, the filehandle from which data was received, a
nd a reference to the input buffer. The callback should delete data from the buffer once it has been processed. 
For example, to process line by line:


sub mux_input {
  my ($obj, $mux, $fh, $buffer) = @_;
  my ($line) = $$buffer =~ s{^(.*)\n}{  } or return;
  # ...
}


The IO::Multiplex module also takes care of accepting incoming connections on server sockets. 
Once you have a socket bound and listening (see Recipe 17.2), 
pass it to the listen method of an IO::Multiplex object:


use IO::Socket;
$server = IO::Socket::INET->new(LocalPort => $PORT, Listen => 10)
  or die $@;
$mux->listen($server);


当新的输入连接接受时,方法mux_connection被调用
When new incoming connections are accepted, the mux_connection callback is called. 
There are other callbacks, such as for full and partial closure of a filehandle, timeouts, and so on. 
For a full list of the methods you can use to control an IO::Multiplex object and a full list of the callbacks, 
see the IO::Multiplex documentation.


代码功能:
一个基本的聊天服务器,监听本机的6901端口;
每个客户端有自己的名字,并可以改名;
每个输入文本都会被广播到所有连接的客户端;
Example 1 is a rudimentary chat server that uses IO::Multiplex. 
It listens on port 6901 of the local host address and implements a very rudimentary chat protocol. 
Every client (see Example 17-8) has a "name," which they can change by sending a line that looks like /nick newname. 
Every other incoming line of text is sent out to all connected machines, prefaced with the name of the client that sent it.


测试方法
To test this out, run the server in one window, then start a few clients in other windows. 
Type something into one and see what appears in the others.




Example 1 chatserver
  #!/usr/bin/perl -w
  # chatserver - very simple chat server
  use IO::Multiplex;
  use IO::Socket;
  use strict;
  my %Name;
  my $Server = IO::Socket::INET->new(LocalAddr => "localhost:6901",
                                     Listen   => 10, Reuse => 1,
                                     Proto    => 'tcp') or die $@;
  my $Mux = IO::Multiplex->new( );
  my $Person_Counter = 1;
  $Mux->listen($Server);                                       #添加监听的socket
  $Mux->set_callback_object(_ _PACKAGE_ _);    #设置回调对象为默认值
  $Mux->loop( );                                                   #进入主循环,并开始处理I/O事件
  exit;


############### 重定义回调函数 ###############################################
## 当在监听socket上有新的连接时,调用此函数
  sub mux_connection {
    my ($package, $mux, $fh) = @_;
    $Name{$fh} = [ $fh, "Person " . $Person_Counter++ ];
  }


## 当出现EOF条件时调用
  sub mux_eof {
    my ($package, $mux, $fh) = @_;
    delete $Name{$fh};
  }


## 当输入准备好时调用
  sub mux_input {
    my ($package, $mux, $fh, $input) = @_;
    my $line;
    my $name;
    $$input =~ s{^(.*)\n+}{  } or return;
    $line = $1;
    if ($line =~ m{^/nick\s+(\S+)\s*}) {
      my $oldname = $Name{$fh};
      $Name{$fh} = [ $fh, $1 ];
      $line = "$oldname->[1] is now known as $1";
    } else {
      $line = "<$Name{$fh}[1]> $line";
    }
    foreach my $conn_struct (values %Name) {
      my $conn = $conn_struct->[0];
      $conn->print("$line\n");
    }
  }




Example 2. chatclient
  #!/usr/bin/perl -w
  # chatclient - client for the chat server
  use IO::Multiplex;
  use IO::Socket;
  use strict;
  my $sock = IO::Socket::INET->new(PeerAddr => "localhost:6901",
                                   Proto    => "tcp") or die $@;
  my $Mux = IO::Multiplex->new( );
  $Mux->add($sock);
  $Mux->add(*STDIN);
  $Mux->set_callback_object(_ _PACKAGE_ _);
  $Mux->loop( );
  exit;


############### 重定义回调函数 ###############################################
  sub mux_input {
    my ($package, $mux, $fh, $input) = @_;
    my $line;
    $line = $$input;
    $$input = "";
    if (fileno($fh) =  = fileno(STDIN)) {
      print $sock $line;
    } else {
      print $line;
    }
  }

阅读(3617) | 评论(0) | 转发(1) |
给主人留下些什么吧!~~