cuike519的blog

工作、学习、娱乐

用户操作
[即时聊天] [发私信] [加为好友]
wujianID:cuike519
86770次访问,排名1171(1),好友2人,关注者13人。
cuike519的文章
原创 44 篇
翻译 0 篇
转载 0 篇
评论 133 篇
cuike519的公告
很久没有写blog了,总觉得生活少点什么。我的新邮件Mathew.mike@hotmail.com My MSNspase is http://gunsbypower.spaces.msn.com/
最近评论
nychanglingfeng:真是挺牛的
nychanglingfeng:真是挺牛的
fmmmeck:今天我也遇到这个问题了,多亏了这篇文章和 lf_shao 的修正 !
fmmmeck:今天我也遇到这个问题了,多亏了这篇文章和 lf_shao 的修正 !
fmmmeck:今天我也遇到这个问题了,多亏了这篇文章和 lf_shao 的修正 !
文章分类
收藏
    相册
    程序代码
    存档
    软件项目交易
    订阅我的博客
    XML聚合  FeedSky
    订阅到鲜果
    订阅到Google
    订阅到抓虾
    订阅到BlogLines
    订阅到Yahoo
    订阅到GouGou
    订阅到飞鸽
    订阅到Rojo
    订阅到newsgator
    订阅到netvibes

    原创 如何有效的使用C#读取文件收藏

    新一篇: TreeView父子联动效果保持节点状态一致 | 旧一篇: 如何复制一个目录里面的所有目录和文件

    你平时是怎么读取文件的?使用流读取。是的没错,C#给我们提供了非常强大的类库(又一次吹捧了.NET一番),里面封装了几乎所有我们可以想到的和我们没有想到的类,流是读取文件的一般手段,那么你真的会用它读取文件中的数据了么?真的能读完全么?

    通常我们读取一个文件使用如下的步骤:

    1、声明并使用FileOpenRead实例化一个文件流对象,就像下面这样

           FileStream fs = File.OpenRead(filename);

    或者

           FileStream fs = FileStream(filename, FileMode.Open, FileAccess.Read, FileShare.Read);

    2、准备一个存放文件内容的字节数组,fs.Length将得到文件的实际大小,就像下面这样

           byte[] data = new byte[fs.Length];

    3、哇!开始读了,调用一个文件流的一个方法读取数据到data数组中

           fs.Read (data, 0, data.Length);

    呵呵!我们只写了3句就可以把文件里面的内容原封不动的读出来,真是太简洁了!可以这段代码真的能像你预期的那样工作么?答案是:几乎可以!在大部分情况下上面的代码工作的很好,但是我们应该注意Read方法是有返回值的,既然有返回值那么一定有其道理,如果按照上面的写法完全可以是一个没有返回值的函数。我想返回值的目的是,为了给我们一个机会判断实际读取文件的大小,从而来判断文件是否已经完全读完。所以上面的代码不能保证我们一定读完了文件里面的所有字节(虽然在很多情况下是读完了)。下面的方法提供了一个比上面方法更安全的方法,来保证文件被完全读出

    public static void SafeRead (Stream stream, byte[] data){

        int offset=0;

        int remaining = data.Length;

           // 只要有剩余的字节就不停的读

        while (remaining > 0){

            int read = stream.Read(data, offset, remaining);

            if (read <= 0)

                throw new EndOfStreamException("文件读取到"+read.ToString()+"失败!");

                  // 减少剩余的字节数

            remaining -= read;

                  // 增加偏移量

            offset += read;

        }

    }

     

    有些情况下你不知道流实际的长度比如:网络流。此时可以使用类似的方法读取流直到流里面的数据完全读取出来为止。我们可以先初始化一段缓存,再将流读出来的流信息写到内存流里面,就像下面这样:

    public static byte[] ReadFully (Stream stream){

           // 初始化一个32k的缓存

        byte[] buffer = new byte[32768];

        using (MemoryStream ms = new MemoryStream()){ //返回结果后会自动回收调用该对象的Dispose方法释放内存

                  // 不停的读取

            while (true){

                int read = stream.Read (buffer, 0, buffer.Length);

                         // 直到读取完最后的3M数据就可以返回结果了

                if (read <= 0)

                    return ms.ToArray();

                ms.Write (buffer, 0, read);

            }

        }

    }

     

    虽然上面的例子都比较简单,效果也不是很明显(大部分都是对的),也许你早就会了,没关系这篇文章本来就是写给初学者的。

    下面的方法提供了一种使用指定缓存长度的方式读取流,虽然在很多情况下你可以直接使用Stream.Length得到流的长度,但是不是所有的流都可以得到。

    public static byte[] Read2Buffer (Stream stream, int BufferLen){

           // 如果指定的无效长度的缓冲区,则指定一个默认的长度作为缓存大小

           if (BufferLen < 1){

                  BufferLen = 0x8000;

           }

           // 初始化一个缓存区

           byte[] buffer = new byte[BufferLen];

           int read=0;  

           int block;

           // 每次从流中读取缓存大小的数据,知道读取完所有的流为止

           while ( (block = stream.Read(buffer, read, buffer.Length-read)) > 0){

                  // 重新设定读取位置

                  read += block;

         

                  // 检查是否到达了缓存的边界,检查是否还有可以读取的信息

                  if (read == buffer.Length){

                         // 尝试读取一个字节

                         int nextByte = stream.ReadByte();

             

                         // 读取失败则说明读取完成可以返回结果

                         if (nextByte==-1){

                                return buffer;

                         }

             

                         // 调整数组大小准备继续读取

                         byte[] newBuf = new byte[buffer.Length*2];

                         Array.Copy(buffer, newBuf, buffer.Length);

                         newBuf[read]=(byte)nextByte;

                         buffer = newBuf;// buffer是一个引用(指针),这里意在重新设定buffer指针指向一个更大的内存

                         read++;

                  }

           }

           // 如果缓存太大则使用ret来收缩前面while读取的buffer,然后直接返回

           byte[] ret = new byte[read];

           Array.Copy(buffer, ret, read);

           return ret;

    }

    发表于 @ 2005年01月19日 12:42:00|评论(loading...)|编辑

    新一篇: TreeView父子联动效果保持节点状态一致 | 旧一篇: 如何复制一个目录里面的所有目录和文件

    评论

    #tmfc 发表于2005-01-24 15:16:00  IP: 222.64.206.*
    不知道什么时候会读取不正确呢?
    #csharpfun 发表于2005-02-13 12:19:00  IP: 222.53.43.*
    好,搜藏.
    #suifeng108 发表于2007-08-29 09:37:34  IP: 61.141.242.*
    试一下看看
    #kissyougoodby 发表于2007-09-29 11:13:56  IP: 202.102.140.*
    这么做会不会有效能问题的产生呢?
    #nighting1029 发表于2008-05-10 11:37:14  IP: 220.160.149.*
    楼主!想把你的blog文章贴转到我的网站,不知可否?
    我的QQ:123714162
    发表评论  


    当前用户设置只有注册用户才能发表评论。如果你没有登录,请点击登录
    Csdn Blog version 3.1a
    Copyright © cuike519