<?xml version="1.0" encoding="UTF-8" ?>
<rss version="2.0">
<channel>
<title><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></title> 
<link>http://www.jackxiang.com/index.php</link> 
<description><![CDATA[赢在IT，Playin' with IT,Focus on Killer Application,Marketing Meets Technology.]]></description> 
<language>zh-cn</language> 
<copyright><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></copyright>
<item>
<link>http://www.jackxiang.com/post//</link>
<title><![CDATA[用C++进行简单的文件I/O操作ofstream  fout，fin ]]></title> 
<author>jack &lt;xdy108@126.com&gt;</author>
<category><![CDATA[WEB2.0]]></category>
<pubDate>Fri, 14 Mar 2008 06:55:31 +0000</pubDate> 
<guid>http://www.jackxiang.com/post//</guid> 
<description>
<![CDATA[ 
	”文件”中如果没有指定路径，则在该可执行文件相同的文件夹下，如果指定路径则在该路径下，如：<br/>ofstream ofile(&quot;output.dat&quot;);<br/>ofstream ofile(&quot;c:&#92;&#92;temp&#92;&#92;output.dat&quot;);<br/><br/>如果文件不存在，则一般新建一个文件。打开方式，如按二进制打开<br/>ofstream ofile(&quot;output.dat&quot;, ios_base::binary);<br/>[b]如果追加方式打开，则：[/b]<br/>ofstream ofile(&quot;output.dat&quot;, ios_base::app);<br/>注意：ofstream 流在打开方式中已经默认了ios_base::out，故这个参数一般不必重复使用，如：<br/>ofstream ofile(&quot;output.dat&quot;, ios_base::out);<br/><br/>早期的C++语言使用ios代替ios_base<br/><br/>序论<br/>　　我曾发表过文件输入输出的文章，现在觉得有必要再写一点。文件 I/O 在C++中比烤蛋糕简单多了。 在这篇文章里，我会详细解释ASCII和二进制文件的输入输出的每个细节，值得注意的是，所有这些都是用C++完成的。<br/><br/>一、ASCII 输出<br/>　　为了使用下面的方法, 你必须包含头文件&lt;fstream.h&gt;(译者注：在标准C++中，已经使用&lt;fstream&gt;取代&lt; fstream.h&gt;，所有的C++标准头文件都是无后缀的。)。这是 &lt;iostream.h&gt;的一个扩展集, 提供有缓冲的文件输入输出操作. 事实上, &lt;iostream.h&gt; 已经被&lt;fstream.h&gt;包含了, 所以你不必包含所有这两个文件, 如果你想显式包含他们，那随便你。我们从文件操作类的设计开始, 我会讲解如何进行ASCII I/O操作。如果你猜是&quot;fstream,&quot; 恭喜你答对了！ 但这篇文章介绍的方法,我们分别使用&quot;ifstream&quot;?和 &quot;ofstream&quot; 来作输入输出。<br/>如果你用过标准控制台流&quot;cin&quot;?和 &quot;cout,&quot; 那现在的事情对你来说很简单。 我们现在开始讲输出部分，首先声明一个类对象。<br/><br/>ofstream fout;<br/><br/>这就可以了，不过你要打开一个文件的话, 必须像这样调用ofstream::open()。<br/><br/>fout.open(&quot;output.txt&quot;);<br/><br/>你也可以把文件名作为构造参数来打开一个文件.<br/><br/>ofstream fout(&quot;output.txt&quot;);<br/><br/>　　这是我们使用的方法, 因为这样创建和打开一个文件看起来更简单. 顺便说一句, 如果你要打开的文件不存在，它会为你创建一个, 所以不用担心文件创建的问题. 现在就输出到文件，看起来和&quot;cout&quot;的操作很像。 对不了解控制台输出&quot;cout&quot;的人, 这里有个例子。<br/><br/>int num = 150;char name[] = &quot;John Doe&quot;;fout &lt;&lt; &quot;Here is a number: &quot; &lt;&lt; num &lt;&lt; &quot;&#92;n&quot;;fout &lt;&lt; &quot;Now here is a string: &quot; &lt;&lt; name &lt;&lt; &quot;&#92;n&quot;;<br/><br/>　　现在保存文件，你必须关闭文件，或者回写文件缓冲. 文件关闭之后就不能再操作了, 所以只有在你不再操作这个文件的时候才调用它，它会自动保存文件。 回写缓冲区会在保持文件打开的情况下保存文件, 所以只要有必要就使用它。回写看起来像另一次输出, 然后调用方法关闭。像这样：<br/><br/>fout &lt;&lt; flush; fout.close();<br/><br/>现在你用文本编辑器打开文件，内容看起来是这样：<br/><br/>Here is a number: 150 Now here is a string: John Doe<br/><br/>　　很简单吧! 现在继续文件输入, 需要一点技巧, 所以先确认你已经明白了流操作，对 &quot;&lt;&lt;&quot; 和&quot;&gt;&gt;&quot; 比较熟悉了, 因为你接下来还要用到他们。继续…<br/><br/>二、ASCII 输入<br/>　　输入和&quot;cin&quot; 流很像. 和刚刚讨论的输出流很像, 但你要考虑几件事情。在我们开始复杂的内容之前, 先看一个文本：<br/><br/>12 GameDev 15.45 L This is really awesome!<br/><br/>为了打开这个文件，你必须创建一个in-stream对象,?像这样。<br/><br/>ifstream fin(&quot;input.txt&quot;);<br/><br/>　　现在读入前四行. 你还记得怎么用&quot;&lt;&lt;&quot; 操作符往流里插入变量和符号吧？好,?在 &quot;&lt;&lt;&quot; (插入)?操作符之后，是&quot;&gt;&gt;&quot; (提取) 操作符. 使用方法是一样的. 看这个代码片段.<br/><br/>int number; float real; char letter, word[8]; fin &gt;&gt; number; fin &gt;&gt; word; fin &gt;&gt; real; fin &gt;&gt; letter;<br/><br/>也可以把这四行读取文件的代码写为更简单的一行。<br/><br/>fin &gt;&gt; number &gt;&gt; word &gt;&gt; real &gt;&gt; letter;<br/><br/>　　它是如何运作的呢? 文件的每个空白之后, &quot;&gt;&gt;&quot; 操作符会停止读取内容, 直到遇到另一个&gt;&gt;操作符. 因为我们读取的每一行都被换行符分割开(是空白字符), &quot;&gt;&gt;&quot; 操作符只把这一行的内容读入变量。这就是这个代码也能正常工作的原因。但是，可别忘了文件的最后一行。<br/><br/>This is really awesome!<br/><br/>　　如果你想把整行读入一个char数组, 我们没办法用&quot;&gt;&gt;&quot;?操作符，因为每个单词之间的空格（空白字符）会中止文件的读取。为了验证：<br/><br/>char sentence[101]; fin &gt;&gt; sentence;<br/><br/>　　我们想包含整个句子, &quot;This is really awesome!&quot; 但是因为空白, 现在它只包含了&quot;This&quot;. 很明显, 肯定有读取整行的方法, 它就是getline()。这就是我们要做的。<br/><br/>fin.getline(sentence, 100);<br/><br/>　　这是函数参数. 第一个参数显然是用来接受的char数组. 第二个参数是在遇到换行符之前，数组允许接受的最大元素数量. 现在我们得到了想要的结果：“This is really awesome!”。<br/>你应该已经知道如何读取和写入ASCII文件了。但我们还不能罢休，因为二进制文件还在等着我们。<br/><br/>三、二进制 输入输出<br/>　　二进制文件会复杂一点, 但还是很简单的。 首先你要注意我们不再使用插入和提取操作符(译者注：&lt;&lt; 和 &gt;&gt; 操作符). 你可以这么做，但它不会用二进制方式读写。你必须使用read() 和write() 方法读取和写入二进制文件. 创建一个二进制文件, 看下一行。<br/><br/>ofstream fout(&quot;file.dat&quot;, ios::binary);<br/><br/>　　这会以二进制方式打开文件, 而不是默认的ASCII模式。首先从写入文件开始。函数write() 有两个参数。 第一个是指向对象的char类型的指针, 第二个是对象的大小（译者注：字节数）。 为了说明，看例子。<br/><br/>int number = 30; fout.write((char *)(&amp;number), sizeof(number));<br/><br/>　　第一个参数写做&quot;(char *)(&amp;number)&quot;. 这是把一个整型变量转为char *指针。如果你不理解，可以立刻翻阅C++的书籍，如果有必要的话。第二个参数写作&quot;sizeof(number)&quot;. sizeof() 返回对象大小的字节数. 就是这样!<br/>二进制文件最好的地方是可以在一行把一个结构写入文件。 如果说，你的结构有12个不同的成员。 用ASCII?文件，你不得不每次一条的写入所有成员。 但二进制文件替你做好了。 看这个。<br/><br/>struct OBJECT { int number; char letter; } obj; obj.number = 15;obj.letter = ‘M’; fout.write((char *)(&amp;obj), sizeof(obj));<br/><br/>　　这样就写入了整个结构! 接下来是输入. 输入也很简单，因为read()?函数的参数和 write()是完全一样的, 使用方法也相同。<br/><br/>ifstream fin(&quot;file.dat&quot;, ios::binary); fin.read((char *)(&amp;obj), sizeof(obj));<br/><br/>　　我不多解释用法, 因为它和write()是完全相同的。二进制文件比ASCII文件简单, 但有个缺点是无法用文本编辑器编辑。 接着, 我解释一下ifstream 和ofstream 对象的其他一些方法作为结束.<br/><br/>四、更多方法<br/>　　我已经解释了ASCII文件和二进制文件, 这里是一些没有提及的底层方法。<br/><br/>检查文件<br/>你已经学会了open() 和close() 方法, 不过这里还有其它你可能用到的方法。<br/>方法good() 返回一个布尔值，表示文件打开是否正确。<br/>类似的，bad() 返回一个布尔值表示文件打开是否错误。 如果出错，就不要继续进一步的操作了。<br/>最后一个检查的方法是fail(), 和bad()有点相似, 但没那么严重。<br/><br/>读文件<br/>方法get() 每次返回一个字符。<br/>方法ignore(int,char) 跳过一定数量的某个字符, 但你必须传给它两个参数。第一个是需要跳过的字符数。 第二个是一个字符, 当遇到的时候就会停止。 例子,<br/><br/>fin.ignore(100, ‘&#92;n’);<br/><br/>会跳过100个字符，或者不足100的时候，跳过所有之前的字符，包括 ‘&#92;n’。<br/>方法peek() 返回文件中的下一个字符, 但并不实际读取它。所以如果你用peek() 查看下一个字符, 用get() 在peek()之后读取，会得到同一个字符, 然后移动文件计数器。<br/>方法putback(char) 输入字符, 一次一个, 到流中。我没有见到过它的使用，但这个函数确实存在。<br/><br/>写文件<br/>只有一个你可能会关注的方法.?那就是 put(char), 它每次向输出流中写入一个字符。<br/><br/>打开文件<br/>当我们用这样的语法打开二进制文件:<br/><br/>ofstream fout(&quot;file.dat&quot;, ios::binary);<br/><br/>　　&quot;ios::binary&quot;是你提供的打开选项的额外标志. 默认的, 文件以ASCII方式打开, 不存在则创建, 存在就覆盖. 这里有些额外的标志用来改变选项。<br/>ios::app&nbsp;&nbsp; 添加到文件尾<br/>ios::ate&nbsp;&nbsp; 把文件标志放在末尾而非起始。<br/>ios::trunc&nbsp;&nbsp; 默认. 截断并覆写文件。<br/>ios::nocreate&nbsp;&nbsp; 文件不存在也不创建。<br/>ios::noreplace&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;文件存在则失败。<br/><br/>文件状态<br/>　　我用过的唯一一个状态函数是eof(), 它返回是否标志已经到了文件末尾。 我主要用在循环中。 例如, 这个代码断统计小写‘e’ 在文件中出现的次数。<br/><br/>ifstream fin(&quot;file.txt&quot;); char ch; int counter; while (!fin.eof()) {&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ch = fin.get();&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (ch == ‘e’) counter++; }fin.close();<br/><br/>　　我从未用过这里没有提到的其他方法。 还有很多方法，但是他们很少被使用。参考C++书籍或者文件流的帮助文档来了解其他的方法。<br/><br/>结论<br/>　　你应该已经掌握了如何使用ASCII文件和二进制文件。有很多方法可以帮你实现输入输出，尽管很少有人使用他们。我知道很多人不熟悉文件I/O操作，我希望这篇文章对你有所帮助。 每个人都应该知道. 文件I/O还有很多显而易见的方法,?例如包含文件 &lt;stdio.h&gt;. 我更喜欢用流是因为他们更简单。 祝所有读了这篇文章的人好运, 也许以后我还会为你们写些东西。　<br/><br/><br/>//使用(C++库)ofstream写文件数据<br/>//simple example<br/>#include &lt;iostream&gt;<br/>#include &lt;fstream&gt;<br/>using namespace std;<br/>#ifdef WIN32<br/>#define TEST_FILE &quot;c:&#92;&#92;shi&#92;&#92;aaa.txt&quot;<br/>#else<br/>#define TEST_FILE &quot;/tmp/test.txt&quot;<br/>#endif<br/>void test()<br/>{<br/>//ofstream ofs;<br/>//ofs.open(TEST_FILE);<br/>ofstream ofs(TEST_FILE);<br/>char ch = &#039;#&#039;;<br/>const char buf[] = &quot;1234567890&quot;;<br/>ofs.put(ch);//simple<br/>ofs.write(buf, sizeof(buf));<br/>ofs.put(ch);<br/>ofs.close();<br/>}<br/>int main(int argc, char* argv[])<br/>{<br/>test();<br/>return 0;<br/>}<br/>
]]>
</description>
</item><item>
<link>http://www.jackxiang.com/post//#blogcomment</link>
<title><![CDATA[[评论] 用C++进行简单的文件I/O操作ofstream  fout，fin ]]></title> 
<author> &lt;user@domain.com&gt;</author>
<category><![CDATA[评论]]></category>
<pubDate>Thu, 01 Jan 1970 00:00:00 +0000</pubDate> 
<guid>http://www.jackxiang.com/post//#blogcomment</guid> 
<description>
<![CDATA[ 
	
]]>
</description>
</item>
</channel>
</rss>