I/O流的設(shè)計(jì)
C++通用框架的設(shè)計(jì) 作者:naven
1 I/O流介紹
I/O流,即輸入/輸出流(Input/Output Stream),是軟件框架中的核心系統(tǒng)。對(duì)程序設(shè)計(jì)語言設(shè)計(jì)者來說,設(shè)計(jì)一個(gè)令人滿意的I/O系統(tǒng),是件極艱巨的任務(wù),也是不可缺少的基礎(chǔ)設(shè)施。C++的標(biāo)準(zhǔn)模板庫已經(jīng)提供一套很不錯(cuò)的I/O流庫了,但是我還是喜歡Java和.Net框架提供的那樣的使用簡單且擴(kuò)展性強(qiáng)的I/O系統(tǒng),而且還需要自行做一些改進(jìn)已融入到C++通用框架體系中,由于I/O系統(tǒng)是框架的基礎(chǔ)設(shè)施,所以設(shè)計(jì)它是很關(guān)鍵的任務(wù)。
本框架的I/O系統(tǒng)設(shè)計(jì)宗旨與Java的I/O系統(tǒng)設(shè)計(jì)初衷略有不同(Java I/O的設(shè)計(jì)原是為了防止classes膨脹,可是事與愿違),主要目標(biāo)一是要讓使用更簡單,二是讓擴(kuò)展更簡單。本I/O系統(tǒng)參考Java I/O設(shè)計(jì),有InputStream/OutputStream和Reader/Writer兩部分的類,但是含義略有不同,本I/O系統(tǒng)的InputStream/OutputStream定義為流設(shè)備,可將系統(tǒng)中任何設(shè)備或者內(nèi)存或者內(nèi)存中的對(duì)象當(dāng)作輸入/輸出流設(shè)備,它們實(shí)現(xiàn)I/O系統(tǒng)最底層的輸入/輸出的基本操作。而Reader/Writer定義為流設(shè)備的讀寫器,它實(shí)現(xiàn)對(duì)InputStream/OutputStream的流設(shè)備的讀寫功能,可以組裝和擴(kuò)展讀寫功能,提供豐富的讀寫操作符。下面介紹一下I/O流主要的類:
主要有如下一些類
class AbstractFile 表示I/O設(shè)備(如File、Socket等)的Abstract基類
class File 文件設(shè)備,表示系統(tǒng)的文件、目錄等
class InputStream 表示輸入流設(shè)備的Abstract基類
class FileInputStream 表示文件輸入流設(shè)備
class SocketInputStream 表示Socket輸入流設(shè)備
class StringBufferInputStream 表示String對(duì)象輸入流設(shè)備
class OutputStream 表示輸出流設(shè)備的Abstract基類
class FileOutputStream 表示文件輸出流設(shè)備
class SocketOutputStream 表示Socket輸出流設(shè)備
class StringBufferOutputStream 表示String對(duì)象輸出流設(shè)備
class SystemFileOutputStream 表示系統(tǒng)文件(STDOUT等)輸出流設(shè)備
class ConsoleOutputStream 表示系統(tǒng)控制臺(tái)輸出流設(shè)備
class STDERROutputStream 表示系統(tǒng)STDERR輸出流設(shè)備
class STDOUTOutputStream 表示系統(tǒng)STDOUT輸出流設(shè)備
class Reader 表示讀操作器的Abstract基類
class BufferedReader 表示緩沖的讀操作器
class FileReader 表示文件的讀操作器
class InputStreamReader 把輸入流設(shè)備當(dāng)作的讀操作器
class SocketReader 表示Socket的讀操作器
class StringReader 表示String對(duì)象的讀操作器
class Writer 表示寫操作器的Abstract基類
class BufferedWriter 表示緩沖的寫操作器
class FileWriter 表示文件的寫操作器
class OutputStreamWriter 把輸出流設(shè)備當(dāng)作的寫操作器
class SocketWriter 表示Socket的寫操作器
class StringWriter 表示String對(duì)象的寫操作器
2 Hello World!
下面的程序示例如何用上面的類進(jìn)行讀寫操作:
void main()


{
// 定義一個(gè)文件的讀操作器,它將自動(dòng)創(chuàng)建文件輸入流設(shè)備
FileReader fr("c:\\temp\\test.txt");
// 通過讀操作器構(gòu)造一個(gè)緩沖的讀操作器
BufferedReader rd(fr);
String s;

// 一行行讀取文件內(nèi)容,并輸出
// 注意:每一行自動(dòng)帶“\r\n”

while( rd.readLine(s) > 0 )
{
// 截掉后面的 \r\n 字符
s.rtrimChars(“\r\n”);
printf("%s\r\n", s.c_str());
}

}

各個(gè)讀寫操作器還可以組合使用,跟Java的用法類似。
3 AbstactFile類
它描述了一個(gè)可進(jìn)行輸入輸出操作的設(shè)備的通用公共的Abstract抽象基類,如文件、目錄、Socket等。它定義了一些公共的成員和方法,如果錯(cuò)誤號(hào)等,還定義了一些子類必須實(shí)現(xiàn)的接口,即純虛方法,所以它是一個(gè)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類是描述操作系統(tǒng)的文件和目錄的類,可讀取文件的一些屬性信息,如創(chuàng)建時(shí)間、大小等,也可以通過此類修改這些信息。備注:目錄的一些功能還未完全實(shí)現(xiàn)。下面的程序展示了通過它來獲取文件的屬性信息:
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(), // 父級(jí)目錄
file.isAbsolute(), // 是否絕對(duì)路徑
file.getAbsolutePath().c_str(), // 絕對(duì)路徑
file.exists(), file.canRead(), file.canWrite(), file.canReadWrite(),
file.isDirectory(), file.isFile(), // 是否文件或目錄
file.lastModified(), // 最近修改時(shí)間
file.length(), // 文件長度
tff.format(file.lastModified()).c_str()); // 最近修改時(shí)間

5 InputStream類
它描述了一個(gè)輸入流設(shè)備的通用公共的Abstract抽象基類,每一種數(shù)據(jù)源都有相應(yīng)的InputStream子類,如文件輸入流、Socket輸入流、字符串輸入流等。它跟AbstractFile類似,定義公共的方法,也定義了一些子類需要實(shí)現(xiàn)的接口。InputStream流設(shè)備實(shí)現(xiàn)的接口均是最基礎(chǔ)的操作,如讀取一個(gè)byte字節(jié)的數(shù)據(jù),或者讀取指定長度的數(shù)據(jù),它的定義看起來如下所示:
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輸入流相對(duì)應(yīng),它描述了一個(gè)輸出流設(shè)備的通用公共的Abstract抽象基類,每一種數(shù)據(jù)源都有相應(yīng)的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類
它描述了一個(gè)對(duì)流設(shè)備的讀操作器的通用公共的Abstract抽象基類,每一種輸入流設(shè)備都有相應(yīng)的讀操作器,如文件讀操作器、Socket讀操作器、字符串讀操作器等。它的定義與Java有所區(qū)別,Java里的Reader主要用于“相容于Unicode并基于字符”的數(shù)據(jù)源的讀取操作,而本系統(tǒng)的Reader是用于所有輸入流設(shè)備進(jìn)行的讀操作。另外某些Reader可以通過另一個(gè)Reader來進(jìn)行創(chuàng)建,以實(shí)現(xiàn)比較復(fù)雜的讀取操作,如帶緩沖的或一行行的讀取。
它的接口定義看起來如下所示:
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對(duì)應(yīng),它描述了一個(gè)對(duì)流設(shè)備的寫操作器的通用公共的Abstract抽象基類,每一種輸出流設(shè)備都有相應(yīng)的寫操作器,如文件寫操作器、Socket寫操作器、字符串寫操作器等。同樣,它也是用于所有輸出流設(shè)備進(jìn)行的寫操作。某些Writer也可以通過另一個(gè)Writer來進(jìn)行創(chuàng)建,以實(shí)現(xiàn)比較復(fù)雜的寫操作,如帶緩沖的寫操作。
考慮到C++的特性,本Writer還實(shí)現(xiàn)了所有的Operator<<操作符,所以可以進(jìn)行與C++標(biāo)準(zhǔn)庫類似的格式化輸出,如下所示:
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類
這是兩個(gè)特殊讀寫操作器類,它的設(shè)計(jì)是運(yùn)用Bridge模式,可以將一個(gè)InputStream流設(shè)備或OuputStream流設(shè)備轉(zhuǎn)換成一個(gè)Reader或一個(gè)Writer。所以讀取文件的操作,通過它們也可以如下實(shí)現(xiàn):
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++通用框架的設(shè)計(jì) 作者:naven 日期:2005-11-9