home *** CD-ROM | disk | FTP | other *** search
-
- <HTML>
- <HEAD>
- <TITLE>IO::WrapTie - wrap tieable objects in IO::Handle interface</TITLE>
- <LINK REL="stylesheet" HREF="../../../Active.css" TYPE="text/css">
- <LINK REV="made" HREF="mailto:">
- </HEAD>
-
- <BODY>
- <TABLE BORDER=0 CELLPADDING=0 CELLSPACING=0 WIDTH=100%>
- <TR><TD CLASS=block VALIGN=MIDDLE WIDTH=100% BGCOLOR="#cccccc">
- <STRONG><P CLASS=block> IO::WrapTie - wrap tieable objects in IO::Handle interface</P></STRONG>
- </TD></TR>
- </TABLE>
-
- <A NAME="__index__"></A>
- <!-- INDEX BEGIN -->
-
- <UL>
-
- <LI><A HREF="#name">NAME</A></LI><LI><A HREF="#supportedplatforms">SUPPORTED PLATFORMS</A></LI>
-
- <LI><A HREF="#synopsis">SYNOPSIS</A></LI>
- <LI><A HREF="#description">DESCRIPTION</A></LI>
- <LI><A HREF="#how it all works">HOW IT ALL WORKS</A></LI>
- <UL>
-
- <LI><A HREF="#the data structures">The data structures</A></LI>
- <LI><A HREF="#how wraptie() works">How <CODE>wraptie()</CODE> works</A></LI>
- <LI><A HREF="#how i/o operators work (on the master)">How I/O operators work (on the master)</A></LI>
- <LI><A HREF="#how methods work (on the master)">How methods work (on the master)</A></LI>
- </UL>
-
- <LI><A HREF="#notes">NOTES</A></LI>
- <LI><A HREF="#warnings">WARNINGS</A></LI>
- <LI><A HREF="#author">AUTHOR</A></LI>
- </UL>
- <!-- INDEX END -->
-
- <HR>
- <P>
- <H1><A NAME="name">NAME</A></H1>
- <P>IO::WrapTie - wrap tieable objects in IO::Handle interface</P>
- <P><EM>This is currently Alpha code, released for comments.
- Please give me your feedback!</EM></P>
- <P>
- <HR>
- <H1><A NAME="supportedplatforms">SUPPORTED PLATFORMS</A></H1>
- <UL>
- <LI>Linux</LI>
- <LI>Solaris</LI>
- <LI>Windows</LI>
- </UL>
- <HR>
- <H1><A NAME="synopsis">SYNOPSIS</A></H1>
- <P>First of all, you'll need tie(), so:</P>
- <PRE>
- require 5.004;</PRE>
- <P><EM>Function interface (experimental).</EM>
- Use this with any existing class...</P>
- <PRE>
- use IO::WrapTie;
- use FooHandle; # implements TIEHANDLE interface
- </PRE>
- <PRE>
-
- # Suppose we want a "FooHandle->new(&FOO_RDWR, 2)".
- # We can instead say...</PRE>
- <PRE>
-
- $FH = wraptie('FooHandle', &FOO_RDWR, 2);</PRE>
- <PRE>
-
- # Now we can use...
- print $FH "Hello, "; # traditional operator syntax...
- $FH->print("world!\n"); # ...and OO syntax as well!</PRE>
- <P><EM>OO interface (preferred).</EM>
- You can inherit from the IO::WrapTie::Slave mixin to get a
- nifty <CODE>new_tie()</CODE> constructor...</P>
- <PRE>
- #------------------------------
- package FooHandle; # a class which can TIEHANDLE
- </PRE>
- <PRE>
-
- use IO::WrapTie;
- @ISA = qw(IO::WrapTie::Slave); # inherit new_tie()
- ...</PRE>
- <PRE>
-
- #------------------------------
- package main;</PRE>
- <PRE>
-
- $FH = FooHandle->new_tie(&FOO_RDWR, 2); # $FH is an IO::WrapTie::Master
- print $FH "Hello, "; # traditional operator syntax
- $FH->print("world!\n"); # OO syntax</PRE>
- <P>See IO::Scalar as an example. It also shows you how to create classes
- which work both with and without 5.004.</P>
- <P>
- <HR>
- <H1><A NAME="description">DESCRIPTION</A></H1>
- <P>Suppose you have a class <CODE>FooHandle</CODE>, where...</P>
- <UL>
- <LI>
- <STRONG>FooHandle does not inherit from IO::Handle;</STRONG> that is, it performs
- filehandle-like I/O, but to something other than an underlying
- file descriptor. Good examples are IO::Scalar (for printing to a
- string) and IO::Lines (for printing to an array of lines).
- <P></P>
- <LI>
- <STRONG>FooHandle implements the TIEHANDLE interface</STRONG> (see <A HREF="../../../lib/Pod/perltie.html">the perltie manpage</A>);
- that is, it provides methods TIEHANDLE, GETC, PRINT, PRINTF,
- READ, and READLINE.
- <P></P>
- <LI>
- <STRONG>FooHandle implements the traditional OO interface</STRONG> of
- FileHandle and IO::Handle; i.e., it contains methods like getline(),
- read(), print(), seek(), tell(), eof(), etc.
- <P></P></UL>
- <P>Normally, users of your class would have two options:</P>
- <UL>
- <LI>
- <STRONG>Use only OO syntax,</STRONG> and forsake named I/O operators like 'print'.
- <P></P>
- <LI>
- <STRONG>Use with tie,</STRONG> and forsake treating it as a first-class object
- (i.e., class-specific methods can only be invoked through the underlying
- object via tied()... giving the object a ``split personality'').
- <P></P></UL>
- <P>But now with IO::WrapTie, you can say:</P>
- <PRE>
- $WT = wraptie('FooHandle', &FOO_RDWR, 2);
- $WT->print("Hello, world\n"); # OO syntax
- print $WT "Yes!\n"; # Named operator syntax too!
- $WT->weird_stuff; # Other methods!</PRE>
- <P>And if you're authoring a class like FooHandle, just have it inherit
- from <CODE>IO::WrapTie::Slave</CODE> and that first line becomes even prettier:</P>
- <PRE>
- $WT = FooHandle->new_tie(&FOO_RDWR, 2);</PRE>
- <P><STRONG>The bottom line:</STRONG> now, almost any class can look and work exactly like
- an IO::Handle... and be used both with OO and non-OO filehandle syntax.</P>
- <P>
- <HR>
- <H1><A NAME="how it all works">HOW IT ALL WORKS</A></H1>
- <P>
- <H2><A NAME="the data structures">The data structures</A></H2>
- <P>Consider this example code, using classes in this distribution:</P>
- <PRE>
- use IO::Scalar;
- use IO::WrapTie;
- </PRE>
- <PRE>
-
- $WT = wraptie('IO::Scalar',\$s);
- print $WT "Hello, ";
- $WT->print("world!\n");</PRE>
- <P>In it, the <CODE>wraptie()</CODE> function creates a data structure as follows:</P>
- <PRE>
- * $WT is a blessed reference to a tied filehandle
- $WT glob; that glob is tied to the "Slave" object.
- | * You would do all your i/o with $WT directly.
- |
- |
- | ,---isa--> IO::WrapTie::Master >--isa--> IO::Handle
- V /
- .-------------.
- | |
- | | * Perl i/o operators work on the tied object,
- | "Master" | invoking the TIEHANDLE methods.
- | | * Method invocations are delegated to the tied
- | | slave.
- `-------------'
- |
- tied(*$WT) | .---isa--> IO::WrapTie::Slave
- V /
- .-------------.
- | |
- | "Slave" | * Instance of FileHandle-like class which doesn't
- | | actually use file descriptors, like IO::Scalar.
- | IO::Scalar | * The slave can be any kind of object.
- | | * Must implement the TIEHANDLE interface.
- `-------------'</PRE>
- <P><EM>NOTE:</EM> just as an IO::Handle is really just a blessed reference to a
- <EM>traditional</EM> filehandle glob... so also, an IO::WrapTie::Master
- is really just a blessed reference to a filehandle
- glob <EM>which has been tied to some ``slave'' class.</EM></P>
- <P>
- <H2><A NAME="how wraptie() works">How <CODE>wraptie()</CODE> works</A></H2>
- <OL>
- <LI>
- The call to function <CODE>wraptie(SLAVECLASS, TIEARGS...)</CODE> is
- passed onto <CODE>IO::WrapTie::Master::new()</CODE>.
- Note that class IO::WrapTie::Master is a subclass of IO::Handle.
- <P></P>
- <LI>
- The <CODE>IO::WrapTie::Master::new</CODE> method creates a new IO::Handle object,
- reblessed into class IO::WrapTie::Master. This object is the <EM>master</EM>,
- which will be returned from the constructor. At the same time...
- <P></P>
- <LI>
- The <CODE>new</CODE> method also creates the <EM>slave</EM>: this is an instance
- of SLAVECLASS which is created by tying the master's IO::Handle
- to SLAVECLASS via <A HREF="../../../lib/Pod/perlfunc.html#item_tie"><CODE>tie(HANDLE, SLAVECLASS, TIEARGS...)</CODE></A>.
- This call to <A HREF="../../../lib/Pod/perlfunc.html#item_tie"><CODE>tie()</CODE></A> creates the slave in the following manner:
- <P></P>
- <LI>
- Class SLAVECLASS is sent the message <CODE>TIEHANDLE(TIEARGS...)</CODE>; it
- will usually delegate this to <CODE>SLAVECLASS::new(TIEARGS...)</CODE>, resulting
- in a new instance of SLAVECLASS being created and returned.
- <P></P>
- <LI>
- Once both master and slave have been created, the master is returned
- to the caller.
- <P></P></OL>
- <P>
- <H2><A NAME="how i/o operators work (on the master)">How I/O operators work (on the master)</A></H2>
- <P>Consider using an i/o operator on the master:</P>
- <PRE>
- print $WT "Hello, world!\n";</PRE>
- <P>Since the master ($WT) is really a [blessed] reference to a glob,
- the normal Perl i/o operators like <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print</CODE></A> may be used on it.
- They will just operate on the symbol part of the glob.</P>
- <P>Since the glob is tied to the slave, the slave's PRINT method
- (part of the TIEHANDLE interface) will be automatically invoked.</P>
- <P>If the slave is an IO::Scalar, that means IO::Scalar::PRINT will be
- invoked, and that method happens to delegate to the <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print()</CODE></A> method
- of the same class. So the <EM>real</EM> work is ultimately done by
- IO::Scalar::print().</P>
- <P>
- <H2><A NAME="how methods work (on the master)">How methods work (on the master)</A></H2>
- <P>Consider using a method on the master:</P>
- <PRE>
- $WT->print("Hello, world!\n");</PRE>
- <P>Since the master ($WT) is blessed into the class IO::WrapTie::Master,
- Perl first attempts to find a <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print()</CODE></A> method there. Failing that,
- Perl next attempts to find a <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print()</CODE></A> method in the superclass,
- IO::Handle. It just so happens that there <EM>is</EM> such a method;
- that method merely invokes the <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print</CODE></A> i/o operator on the self object...
- and for that, see above!</P>
- <P>But let's suppose we're dealing with a method which <EM>isn't</EM> part
- of IO::Handle... for example:</P>
- <PRE>
- my $sref = $WT->sref;</PRE>
- <P>In this case, the intuitive behavior is to have the master delegate the
- method invocation to the slave (now do you see where the designations
- come from?). This is indeed what happens: IO::WrapTie::Master contains
- an AUTOLOAD method which performs the delegation.</P>
- <P>So: when <CODE>sref()</CODE> can't be found in IO::Handle, the AUTOLOAD method
- of IO::WrapTie::Master is invoked, and the standard behavior of
- delegating the method to the underlying slave (here, an IO::Scalar)
- is done.</P>
- <P>Sometimes, to get this to work properly, you may need to create
- a subclass of IO::WrapTie::Master which is an effective master for
- <EM>your</EM> class, and do the delegation there.</P>
- <P>
- <HR>
- <H1><A NAME="notes">NOTES</A></H1>
- <P><STRONG>Why not simply use the object's OO interface?</STRONG>
- Because that means forsaking the use of named operators
- like print(), and you may need to pass the object to a subroutine
- which will attempt to use those operators:</P>
- <PRE>
- $O = FooHandle->new(&FOO_RDWR, 2);
- $O->print("Hello, world\n"); # OO syntax is okay, BUT....
- </PRE>
- <PRE>
-
- sub nope { print $_[0] "Nope!\n" }
- X nope($O); # ERROR!!! (not a glob ref)</PRE>
- <P><STRONG>Why not simply use tie()?</STRONG>
- Because (1) you have to use <A HREF="../../../lib/Pod/perlfunc.html#item_tied"><CODE>tied()</CODE></A> to invoke methods in the
- object's public interface (yuck), and (2) you may need to pass
- the tied symbol to another subroutine which will attempt to treat
- it in an OO-way... and that will break it:</P>
- <PRE>
- tie *T, 'FooHandle', &FOO_RDWR, 2;
- print T "Hello, world\n"; # Operator is okay, BUT...
- </PRE>
- <PRE>
-
- tied(*T)->other_stuff; # yuck! AND...</PRE>
- <PRE>
-
- sub nope { shift->print("Nope!\n") }
- X nope(\*T); # ERROR!!! (method "print" on unblessed ref)</PRE>
- <P><STRONG>Why a master and slave?
- Why not simply write FooHandle to inherit from IO::Handle?</STRONG>
- I tried this, with an implementation similar to that of IO::Socket.
- The problem is that <EM>the whole point is to use this with objects
- that don't have an underlying file/socket descriptor.</EM>.
- Subclassing IO::Handle will work fine for the OO stuff, and fine with
- named operators <EM>if</EM> you tie()... but if you just attempt to say:</P>
- <PRE>
- $IO = FooHandle->new(&FOO_RDWR, 2);
- print $IO "Hello!\n";</PRE>
- <P>you get a warning from Perl like:</P>
- <PRE>
- Filehandle GEN001 never opened</PRE>
- <P>because it's trying to do system-level i/o on an (unopened) file
- descriptor. To avoid this, you apparently have to <A HREF="../../../lib/Pod/perlfunc.html#item_tie"><CODE>tie()</CODE></A> the handle...
- which brings us right back to where we started! At least the
- IO::WrapTie mixin lets us say:</P>
- <PRE>
- $IO = FooHandle->new_tie(&FOO_RDWR, 2);
- print $IO "Hello!\n";</PRE>
- <P>and so is not <EM>too</EM> bad. <CODE>:-)</CODE></P>
- <P>
- <HR>
- <H1><A NAME="warnings">WARNINGS</A></H1>
- <P>Remember: this stuff is for doing FileHandle-like i/o on things
- <EM>without underlying file descriptors</EM>. If you have an underlying
- file descriptor, you're better off just inheriting from IO::Handle.</P>
- <P><STRONG>Be aware that <CODE>new_tie()</CODE> always returns an instance of a
- kind of IO::WrapTie::Master...</STRONG> it does <STRONG>not</STRONG> return an instance
- of the i/o class you're tying to!</P>
- <P>Invoking some methods on the master object causes AUTOLOAD to delegate
- them to the slave object... so it <EM>looks</EM> like you're manipulating a
- ``FooHandle'' object directly, but you're not.</P>
- <P>I have not explored all the ramifications of this use of tie().
- <EM>Here there be dragons</EM>.</P>
- <P>
- <HR>
- <H1><A NAME="author">AUTHOR</A></H1>
- <P>Eryq (<EM><A HREF="mailto:eryq@zeegee.com">eryq@zeegee.com</A></EM>).
- President, ZeeGee Software Inc (<EM><A HREF="http://www.zeegee.com">http://www.zeegee.com</A></EM>).</P>
- <TABLE BORDER=0 CELLPADDING=0 CELLSPACING=0 WIDTH=100%>
- <TR><TD CLASS=block VALIGN=MIDDLE WIDTH=100% BGCOLOR="#cccccc">
- <STRONG><P CLASS=block> IO::WrapTie - wrap tieable objects in IO::Handle interface</P></STRONG>
- </TD></TR>
- </TABLE>
-
- </BODY>
-
- </HTML>
-