目录:html
什么是Stream?c#
什么是字节序列?数组
Stream的重要属性及方法asp.net
Stream异步读写async
Stream 和其子类的类图函数
本章总结工具
MSDN 中的解释太简洁了: 提供字节序列的通常视图
(我可不想这么理解,这一定让我抓狂,我理解的流是向天然界的河流那样清澈而又美丽,c#中的流也是同样,许多技术或者说核心技术都须要流的帮忙)
其实简单的来理解的话字节序列指的是:
字节对象都被存储为连续的字节序列,字节按照必定的顺序进行排序组成了字节序列
那什么关于流的解释能够抽象为下列状况:
打个比方:一条河中有一条鱼游过,这个鱼就是一个字节,这个字节包括鱼的眼睛,嘴巴,等组成8个二进制,显然这条河就是咱们的核心对象:流
立刻进入正题,让咱们来解释下c#的 Stream 是如何使用的
让咱们直接温故或学习下Stream类的结构,属性和相关方法
Stream 类有一个protected 类型的构造函数, 可是它是个抽象类,没法直接以下使用
Stream stream = new Stream();
因此咱们自定义一个流继承自Stream 看看哪些属性必须重写或自定义:
能够看出系统自动帮咱们实现了Stream 的抽象属性和属性方法
1: CanRead: 只读属性,判断该流是否可以读取:
2: CanSeek: 只读属性,判断该流是否支持跟踪查找
3: CanWrite: 只读属性,判断当前流是否可写
当咱们使用流写文件时,数据流会先进入到缓冲区中,而不会马上写入文件,当执行这个方法后,缓冲区的数据流会当即注入基础流
MSDN中的描述:使用此方法将全部信息从基础缓冲区移动到其目标或清除缓冲区,或者同时执行这两种操做。根据对象的状态,可能须要修
改流内的当前位置(例如,在基础流支持查找的状况下即如此)当使用 StreamWriter 或 BinaryWriter 类时,不要刷新 Stream 基对象。
而应使用该类的 Flush 或 Close 方法,此方法确保首先将该数据刷新至基础流,而后再将其写入文件。
(红色部分为关键请你们务必可以理解,从此会在相应的章节中介绍)
5: Length:表示流的长度
*6: Position属性:(很是重要)
虽然从字面中能够看出这个Position属性只是标示了流中的一个位置而已,但是咱们在实际开发中会发现这个想法会很是的幼稚,
不少asp.net项目中文件或图片上传中不少朋友会经历过这样一个痛苦:Stream对象被缓存了,致使了Position属性在流中没法
找到正确的位置,这点会让人抓狂,其实解决这个问题很简单,聪明的你确定想到了,其实咱们每次使用流前必须将Stream.Position
设置成0就好了,可是这还不能根本上解决问题,最好的方法就是用Using语句将流对象包裹起来,用完后关闭回收便可。
*7: abstract int Read(byte[] buffer, int offset, int count)
这个方法包含了3个关键的参数:缓冲字节数组,位移偏量和读取字节个数,每次读取一个字节后会返回一个缓冲区中的总字节数
第一个参数:这个数组至关于一个空盒子,Read()方法每次读取流中的一个字节将其放进这个空盒子中。(所有读完后即可使用buffer字节数组了)
第二个参数:表示位移偏量,告诉咱们从流中哪一个位置(偏移量)开始读取。
最后一个参数:就是读取多少字节数。
返回值即是总共读取了多少字节数.
*8: abstract long Seek(long offset, SeekOrigin origin)
你们还记得Position属性么?其实Seek方法就是从新设定流中的一个位置,在说明offset参数做用以前你们先来了解下SeekOrigin这个枚举:
若是 offset 为负,则要求新位置位于 origin 指定的位置以前,其间隔相差 offset 指定的字节数。若是 offset 为零 (0),则要求新位置位于由 origin 指定的位置处。
若是 offset 为正,则要求新位置位于 origin 指定的位置以后,其间隔相差 offset 指定的字节数.
Stream. Seek(-3,Origin.End); 表示在流末端往前数第3个位置
Stream. Seek(0,Origin.Begin); 表示在流的开头位置
Stream. Seek(3,Orig`in.Current); 表示在流的当前位置日后数第三个位置
查找以后会返回一个流中的一个新位置。其实说道这你们就能理解Seek方法的精妙之处了吧
*9: abstract void Write(byte[] buffer,int offset,int count)
这个方法包含了3个关键的参数:缓冲字节数组,位移偏量和读取字节个数
和read方法不一样的是 write方法中的第一个参数buffer已经有了许多byte类型
的数据,咱们只需经过设置 offset和count来将buffer中的数据写入流中
*10: virtual void Close()
关闭流并释放资源,在实际操做中,若是不用using的话,别忘了使用完流以后将其关闭
这个方法特别重要,使用完当前流千万别忘记关闭!
为了让你们可以快速理解和消化上述属性和方法我会写个示例而且关键部分会详细说明
static void Main(string[] args)
{
byte[] buffer = null;
string testString = "Stream!Hello world";
char[] readCharArray = null;
byte[] readBuffer = null;
string readString = string.Empty;
//关于MemoryStream 我会在后续章节详细阐述
using (MemoryStream stream = new MemoryStream())
{
Console.WriteLine("初始字符串为:{0}", testString);
//若是该流可写
if (stream.CanWrite)
{
//首先咱们尝试将testString写入流中
//关于Encoding我会在另外一篇文章中详细说明,暂且经过它实现string->byte[]的转换
buffer = Encoding.Default.GetBytes(testString);
//咱们从该数组的第一个位置开始写,长度为3,写完以后 stream中便有了数据
//对于新手来讲很难理解的就是数据是何时写入到流中,在冗长的项目代码面前,我遇见过很
//多新手都会有这种经历,我但愿可以用如此简单的代码让新手或者老手们在温故下基础
stream.Write(buffer, 0,3);
Console.WriteLine("如今Stream.Postion在第{0}位置",stream.Position+1);
//从刚才结束的位置(当前位置)日后移3位,到第7位
long newPositionInStream =stream.CanSeek? stream.Seek(3, SeekOrigin.Current):0;
Console.WriteLine("从新定位后Stream.Postion在第{0}位置", newPositionInStream+1);
if (newPositionInStream < buffer.Length)
{
//将重新位置(第7位)一直写到buffer的末尾,注意下stream已经写入了3个数据“Str”
stream.Write(buffer, (int)newPositionInStream, buffer.Length - (int)newPositionInStream);
}
//写完后将stream的Position属性设置成0,开始读流中的数据
stream.Position = 0;
// 设置一个空的盒子来接收流中的数据,长度根据stream的长度来决定
readBuffer = new byte[stream.Length];
//设置stream总的读取数量 ,
//注意!这时候流已经把数据读到了readBuffer中
int count = stream.CanRead?stream.Read(readBuffer, 0, readBuffer.Length):0;
//因为刚开始时咱们使用加密Encoding的方式,因此咱们必须解密将readBuffer转化成Char数组,这样才能从新拼接成string
//首先经过流读出的readBuffer的数据求出从相应Char的数量
int charCount = Encoding.Default.GetCharCount(readBuffer, 0, count);
//经过该Char的数量 设定一个新的readCharArray数组
readCharArray = new char[charCount];
//Encoding 类的强悍之处就是不只包含加密的方法,甚至将解密者都能建立出来(GetDecoder()),
//解密者便会将readCharArray填充(经过GetChars方法,把readBuffer 逐个转化将byte转化成char,而且按一致顺序填充到readCharArray中)
Encoding.Default.GetDecoder().GetChars(readBuffer, 0, count, readCharArray, 0);
for (int i = 0; i < readCharArray.Length; i++)
{
readString += readCharArray[i];
}
Console.WriteLine("读取的字符串为:{0}", readString);
}
stream.Close();
}
Console.ReadLine();
}
显示结果:
你们须要特别注意的是stream.Positon这个很神奇的属性,在复杂的程序中,每每流对象操做也会很复杂,
必定要切记将stream.Positon设置在你所须要的正确位置,还有就是 using语句的使用,它会自动销毁stream对象,
固然Stream.Close()你们都懂的
接着让咱们来讲下关于流中怎么实现异步操做
在Stream基类中还有几个关键方法,它们可以很好实现异步的读写,
异步读取
public virtual IAsyncResult BeginRead(byte[] buffer,int offset,int count,AsyncCallback callback,Object state)
异步写
public virtual IAsyncResult BeginWrite( byte[] buffer, int offset, int count, AsyncCallback callback, Object state )
结束异步读取
public virtual int EndRead( IAsyncResult asyncResult )
结束异步写
public virtual void EndWrite( IAsyncResult asyncResult )
你们很容易的就能发现前两个方法实现了IAsyncResult接口,后2个end方法也顺应带上了一个IAsyncResult参数,
其实并不复杂,(必须说明下 每次调用 Begin方法时都必须调用一次 相对应的end方法)
和通常同步read或write方法一致的是,他们能够当作同步方法使用,可是在复杂的状况下可能也难逃阻塞崩溃等等,可是一旦启用了
异步以后,这些相似于阻塞问题会不复存在,可见微软对于异步的支持正在加大。
最后是有关c#中Stream类和其子类的类图
类图呢?你们确定会这么想把 ^^
为何这个在目录中是灰色的?其实我我的以为这个类图不该该放在这篇博文中,缘由是咱们真正理解并熟练操做了Stream的全部子类?(大牛除外)
(这也是我写后续文章的动力之一,写博能很好的提高知识点的吸取,不只能帮助别人,也能提升本身的对于知识点的理解),因此我想把类图放在这
个系类的总结篇中
本章介绍了流的基本概念和c#中关于流的基类Stream所包含的一些重要的属性和方法,关键是一些方法和属性的细节和咱们操做流对象时必须注意的事项,
文中不少知识点都是自身感悟学习而来,深夜写文不容易,请你们多多关注下,下一章将会介绍操做流类的工具:StreamReader 和StreamWriter
敬请期待!