3

我有一个由此创建的文件句柄对象(为清楚起见,稍作编辑):

sub TIEHANDLE
{
    return $_[0] if ref($_[0]);
    my $class = shift;
    my $self = bless Symbol::gensym(), $class;
    return $self;
}

sub new
{
    my ($class, $fh, $chunk, $interval, $cb) = @_;

    my $self = bless Symbol::gensym(), ref($class) || $class;
    tie *$self, $self;  

    my $data = {
        fh       => $fh,
    };

    ${*$self}{'data'} = $data;

    return $self;
}

sub fileno
{
    my $self = $_[0];
    return ${*$self}{'data'}->{'fh'}->fileno();
}

*FILENO = \&fileno;

我想将它与AnyEvent. 它与 . 一起工作正常,AnyEvent::Impl::Perl但与 AnyEvent::Impl::EV. 我认为我已经从以下位置追踪到这种方法EV

static int
s_fileno (SV *fh, int wr)
{
  dTHX;
  SvGETMAGIC (fh);

  if (SvROK (fh))
    {
      fh = SvRV (fh);
      SvGETMAGIC (fh);
    }

  if (SvTYPE (fh) == SVt_PVGV)
    return PerlIO_fileno (wr ? IoOFP (sv_2io (fh)) : IoIFP (sv_2io (fh)));

  if (SvOK (fh) && (SvIV (fh) >= 0) && (SvIV (fh) < 0x7fffffffL))
    return SvIV (fh);

  return -1;
}

我认为它没有通过SvTYPE (fh) == SVt_PVGV测试。使用Devel::Peek's Dump() 方法我得到:

SV = PVMG(0x9c98460) at 0x44313b0
  REFCNT = 1
  FLAGS = (PADMY,ROK)
  IV = 0
  NV = 0
  RV = 0x9c21d00
  SV = PVGV(0x9c35510) at 0x9c21d00
    REFCNT = 1
    FLAGS = (OBJECT,RMG,MULTI)
    MAGIC = 0x9a68ee0
      MG_VIRTUAL = &PL_vtbl_backref
      MG_TYPE = PERL_MAGIC_backref(<)
      MG_OBJ = 0x9c217a8
    STASH = 0x4374440   "MetadataStream"
    NAME = "GEN5"
    NAMELEN = 4
    GvSTASH = 0x25d01c8 "Symbol"
    GP = 0x9a43d50
      SV = 0x0
      REFCNT = 1
      IO = 0x9c214a8
      FORM = 0x0  
      AV = 0x0
      HV = 0x9c21ce8
      CV = 0x0
      CVGEN = 0x0
      LINE = 102
      FILE = "/usr/share/perl5/Symbol.pm"
      FLAGS = 0x2
      EGV = 0x9c21d00   "GEN5"
  PV = 0x9c21d00 ""
  CUR = 0
  LEN = 0

任何有关我如何调整我的句柄的创建以使其可以通过此测试的帮助将不胜感激。

4

1 回答 1

2

绑定文件句柄仅在 perl 中部分实现,不适用于 EV。通常无法使绑定句柄与事件库一起使用:虽然某些特定情况可能有效,但在使用绑定句柄的大多数情况下,它们无法工作,因为底层文件描述符的就绪通知与就绪无关绑定句柄的通知。

如果您的目标只是在真实文件描述符周围有一个事物包装器,那么一种方法是复制 IO::Handle 和/或 FileHandle 所做的事情(非常丑陋,但在 perl 中起作用的唯一方法),或者将它们用作基类。这不允许您做有趣的事情,例如挂钩读取和写入,但挂钩它们可能会使它们与事件库不兼容。

另一种方法是实现一个 perlio 层(PerlIO::via)。以我的经验,这个模块有点脆弱,但给了你所有的选择。同样,如果您引入缓冲或更有趣的东西将文件描述符与实际 I/O 分离,那么它就无法工作。

最后,如果你想添加一种新的句柄类型,那么你可以创建一个新的观察者类型。这可以像拥有一个函数 my_handle_io_watcher 一样简单,该函数获取您的句柄,检查准备情况并创建所需的低级观察程序。

于 2014-06-04T14:59:11.463 回答