I/O流的设计
C++通用框架的设计 作者:naven
1 I/O流介绍
I/O流,即输入/输出流(Input/Output Stream),是软件框架中的核心系统。对程序设计语言设计者来说,设计一个令人满意的I/O系统,是件极艰巨的任务,也是不可缺少的基础设施。C++的标准模板库已经提供一套很不错的I/O流库了,但是我还是喜欢Java和.Net框架提供的那样的使用简单且扩展性强的I/O系统,而且还需要自行做一些改进已融入到C++通用框架体系中,由于I/O系统是框架的基础设施,所以设计它是很关键的任务。
本框架的I/O系统设计宗旨与Java的I/O系统设计初衷略有不同(Java I/O的设计原是为了防止classes膨胀,可是事与愿违),主要目标一是要让使用更简单,二是让扩展更简单。本I/O系统参考Java I/O设计,有InputStream/OutputStream和Reader/Writer两部分的类,但是含义略有不同,本I/O系统的InputStream/OutputStream定义为流设备,可将系统中任何设备或者内存或者内存中的对象当作输入/输出流设备,它们实现I/O系统最底层的输入/输出的基本操作。而Reader/Writer定义为流设备的读写器,它实现对InputStream/OutputStream的流设备的读写功能,可以组装和扩展读写功能,提供丰富的读写操作符。下面介绍一下I/O流主要的类:
主要有如下一些类
class AbstractFile 表示I/O设备(如File、Socket等)的Abstract基类
class File 文件设备,表示系统的文件、目录等
class InputStream 表示输入流设备的Abstract基类
class FileInputStream 表示文件输入流设备
class SocketInputStream 表示Socket输入流设备
class StringBufferInputStream 表示String对象输入流设备
class OutputStream 表示输出流设备的Abstract基类
class FileOutputStream 表示文件输出流设备
class SocketOutputStream 表示Socket输出流设备
class StringBufferOutputStream 表示String对象输出流设备
class SystemFileOutputStream 表示系统文件(STDOUT等)输出流设备
class ConsoleOutputStream 表示系统控制台输出流设备
class STDERROutputStream 表示系统STDERR输出流设备
class STDOUTOutputStream 表示系统STDOUT输出流设备
class Reader 表示读操作器的Abstract基类
class BufferedReader 表示缓冲的读操作器
class FileReader 表示文件的读操作器
class InputStreamReader 把输入流设备当作的读操作器
class SocketReader 表示Socket的读操作器
class StringReader 表示String对象的读操作器
class Writer 表示写操作器的Abstract基类
class BufferedWriter 表示缓冲的写操作器
class FileWriter 表示文件的写操作器
class OutputStreamWriter 把输出流设备当作的写操作器
class SocketWriter 表示Socket的写操作器
class StringWriter 表示String对象的写操作器
2 Hello World!
下面的程序示例如何用上面的类进行读写操作:
void main()
{
// 定义一个文件的读操作器,它将自动创建文件输入流设备
FileReader fr("c:\\temp\\test.txt");
// 通过读操作器构造一个缓冲的读操作器
BufferedReader rd(fr);
String s;
// 一行行读取文件内容,并输出
// 注意:每一行自动带“\r\n”
while( rd.readLine(s) > 0 ) {
// 截掉后面的 \r\n 字符
s.rtrimChars(“\r\n”);
printf("%s\r\n", s.c_str());
}
}
各个读写操作器还可以组合使用,跟Java的用法类似。
3 AbstactFile类
它描述了一个可进行输入输出操作的设备的通用公共的Abstract抽象基类,如文件、目录、Socket等。它定义了一些公共的成员和方法,如果错误号等,还定义了一些子类必须实现的接口,即纯虚方法,所以它是一个Abstract类。它的定义看起来如下所示:
class AbstractFile
{
public:
/**//**
* Tests whether the application can read the file denoted by this
* abstract pathname.
*
*/
virtual BOOL canRead() = 0;
/**//**
* Tests whether the application can modify to the file denoted by this
* abstract pathname.
*
*/
virtual BOOL canWrite() = 0;
/**//**
* Tests whether the application can modify and read the file denoted by this
* abstract pathname.
*
*/
virtual BOOL canReadWrite() = 0;
};
4 File类
File类是描述操作系统的文件和目录的类,可读取文件的一些属性信息,如创建时间、大小等,也可以通过此类修改这些信息。备注:目录的一些功能还未完全实现。下面的程序展示了通过它来获取文件的属性信息:
File file("C:\\Temp\\main.txt");
printf("\n-----File--------\n"
"path = %s name = %s parent = %s\n"
"isAbsolute = %d absolutePath = %s\n"
"exists = %d canRead = %d canWrite = %d canReadWrite = %d\n"
"isDirectory = %d isFile = %d\n"
"lastModified = %d length= %d\n"
"lastModified time = %s\n",
file.getPath().c_str(), // 路径
file.getName().c_str(), // 文件名
file.getParent().c_str(), // 父级目录
file.isAbsolute(), // 是否绝对路径
file.getAbsolutePath().c_str(), // 绝对路径
file.exists(), file.canRead(), file.canWrite(), file.canReadWrite(),
file.isDirectory(), file.isFile(), // 是否文件或目录
file.lastModified(), // 最近修改时间
file.length(), // 文件长度
tff.format(file.lastModified()).c_str()); // 最近修改时间
5 InputStream类
它描述了一个输入流设备的通用公共的Abstract抽象基类,每一种数据源都有相应的InputStream子类,如文件输入流、Socket输入流、字符串输入流等。它跟AbstractFile类似,定义公共的方法,也定义了一些子类需要实现的接口。InputStream流设备实现的接口均是最基础的操作,如读取一个byte字节的数据,或者读取指定长度的数据,它的定义看起来如下所示:
class InputStream
{
protected:
/**//**
* Offset for the file handle from origin.
*/
long _position;
/**//**
* Is this stream closed?
*/
BOOL _closed;
public:
/**//**
* Report position in input stream.
*/
virtual long tellp() { return _position; }
/**//**
* Returns the number of bytes that can be read (or skipped over)
* from this input stream without blocking by the next caller of
* a method for this input stream.
*/
virtual int available() = 0;
/**//**
* Reads the next byte of data from the input stream.
*/
virtual int read() = 0;
/**//**
* Reads some number of bytes from the input stream and stores
* them into the buffer array b.
*/
virtual int read(void *b, int len) = 0;
/**//**
* Reads up to len bytes of data from the input stream into
* an array of bytes.
*/
virtual int read(void *b, int off, int len) = 0;
/**//**
* Skips over and discards n bytes of data from this input stream.
*/
virtual long skip(long n) = 0;
/**//**
* Closes this input stream and releases any system resources
* associated with the stream.
*/
virtual void close() = 0;
};
6 OutputStream类
与InputStream输入流相对应,它描述了一个输出流设备的通用公共的Abstract抽象基类,每一种数据源都有相应的OuputStream子类,如文件输出流、Socket输出流、字符串输出流等。它的接口定义看起来如下所示:
class OutputStream
{
protected:
/**//**
* Offset for the file handle from origin.
*/
long _position;
/**//**
* Is this stream closed?
*/
BOOL _closed;
public:
/**//**
* Report position in output stream.
*/
virtual long tellp() { return _position; }
/**//**
* Flushes this output stream and forces any buffered
* output bytes to be written out.
*/
virtual void flush() = 0;
/**//**
* Writes len bytes from the specified byte array to
* this output stream.
*/
virtual OutputStream& write(const void *b, int len) = 0;
/**//**
* Writes len bytes from the specified byte array to
* this output stream.
*/
virtual OutputStream& write(const void *b, int off, int len) = 0;
/**//**
* Writes the specified byte to this output stream.
*/
virtual OutputStream& write(int b) = 0;
/**//**
* Closes this output stream and releases any system
* resources associated with this stream.
*/
virtual void close() = 0;
};
7 Reader类
它描述了一个对流设备的读操作器的通用公共的Abstract抽象基类,每一种输入流设备都有相应的读操作器,如文件读操作器、Socket读操作器、字符串读操作器等。它的定义与Java有所区别,Java里的Reader主要用于“相容于Unicode并基于字符”的数据源的读取操作,而本系统的Reader是用于所有输入流设备进行的读操作。另外某些Reader可以通过另一个Reader来进行创建,以实现比较复杂的读取操作,如带缓冲的或一行行的读取。
它的接口定义看起来如下所示:
class Reader
{
public:
/**//**
* Check this stream if is closed.
*/
BOOL isClosed();
/**//**
* Report position in input stream.
*/
virtual long tellp() = 0;
/**//**
* Read a single character.
*/
virtual int read() = 0;
/**//**
* Read characters into a portion of an array.
*/
virtual int read(char *cbuf, int len) = 0;
/**//**
* Read characters into a portion of an array.
*/
virtual int read(char *cbuf, int off, int len) = 0;
/**//**
* Reads up to <code>len</code> bytes of data from this input stream
* into an array of bytes.
*/
int read(String &b, int len);
/**//**
* Reads up to <code>len</code> bytes of data from this input stream
* into an array of bytes.
*/
int read(String &b, int off, int len);
/**//**
* Skip characters.
*/
virtual long skip(long n) = 0;
/**//**
* Tell whether this stream is ready to be read.
*/
virtual BOOL ready();
/**//**
* Tell whether this stream supports the mark() operation.
*/
virtual BOOL markSupported();
/**//**
* Mark the present position in the stream.
*/
virtual BOOL mark(int readAheadLimit);
/**//**
* Reset the stream.
*/
virtual void reset();
/**//**
* Close the stream.
*/
virtual void close();
};
8 Writer类
与Reader对应,它描述了一个对流设备的写操作器的通用公共的Abstract抽象基类,每一种输出流设备都有相应的写操作器,如文件写操作器、Socket写操作器、字符串写操作器等。同样,它也是用于所有输出流设备进行的写操作。某些Writer也可以通过另一个Writer来进行创建,以实现比较复杂的写操作,如带缓冲的写操作。
考虑到C++的特性,本Writer还实现了所有的Operator<<操作符,所以可以进行与C++标准库类似的格式化输出,如下所示:
FileWriter fw("c:\\temp\\test.txt");
fw << "This is the NOTICE log message"
<< 123456 << " "
<< String("fdsafdas ");
它的接口定义看起来如下所示:
class Writer
{
public:
/**//**
* Check this stream if is closed.
*/
BOOL isClosed();
/**//**
* Report position in output stream.
*/
virtual long tellp() = 0;
/**//**
* Write a single character.
*/
virtual Writer& write(int c) = 0;
/**//**
* Write an array of characters.
*/
virtual Writer& write(const char *cbuf) = 0;
/**//**
* Write a portion of an array of characters.
*/
virtual Writer& write(const char *cbuf, int len) = 0;
/**//**
* Write a portion of an array of characters.
*/
virtual Writer& write(const char *cbuf, int off, int len) = 0;
/**//**
* Write a string.
*/
virtual Writer& write(String &str) = 0;
/**//**
* Write a portion of a string.
*/
virtual Writer& write(String &str, int len) = 0;
/**//**
* Write a portion of a string.
*/
virtual Writer& write(String &str, int off, int len) = 0;
/**//**
* Flush the stream.
*/
virtual void flush();
/**//**
* Close the stream.
*/
virtual void close();
};
9 InputStreamReader类和OuputStreamWriter类
这是两个特殊读写操作器类,它的设计是运用Bridge模式,可以将一个InputStream流设备或OuputStream流设备转换成一个Reader或一个Writer。所以读取文件的操作,通过它们也可以如下实现:
void main()
{
FileInputStream fis("c:\\temp\\test.txt");
InputStreamReader isr(fis);
BufferedReader rd(isr);
String s;
while( rd.readLine(s) > 0 ) {
printf("%s\r\n", s.c_str());
}
}
C++通用框架的设计 作者:naven 日期:2005-11-9