一、什么是Stream

查了一下MSDN,他是这么解释的:提供字节序列的一般视图。

这个解释有点太笼统了,下面,我们来仔细的捋一下

1、什么是字节序列?

字节序列指的是:字节对象被存储为连续的字节序列,字节按照一定的顺序进行排序组成了字节序列。

那么关于流的解释可以抽象为下列情况:

一条河中有一条鱼游过,这条鱼就是一个字节,这个字节包括鱼的眼睛、嘴巴、等组成8个二进制,显然这条河就是我们的核心对象:流

下面我们来认识一下C#中的Stream是如何使用的吧。

二、Stream类的结构,属性和相关方法

1、构造函数:

Stream类有一个protected类型的构造函数,但是他是个抽象类,无法直接使用new来实例化。所以我们自定义一个流继承自Stream,看看哪些属性必须重写或自定义:

 public class MyStreamExample : Stream
{ public override bool CanRead
{
get { throw new NotImplementedException(); }
} public override bool CanSeek
{
get { throw new NotImplementedException(); }
} public override bool CanWrite
{
get { throw new NotImplementedException(); }
} public override void Flush()
{
throw new NotImplementedException();
} public override long Length
{
get { throw new NotImplementedException(); }
} public override long Position
{
get
{
throw new NotImplementedException();
}
set
{
throw new NotImplementedException();
}
} public override int Read(byte[] buffer, int offset, int count)
{
throw new NotImplementedException();
} public override long Seek(long offset, SeekOrigin origin)
{
throw new NotImplementedException();
} public override void SetLength(long value)
{
throw new NotImplementedException();
} public override void Write(byte[] buffer, int offset, int count)
{
throw new NotImplementedException();
}
}

可以看出系统自动帮我们实现了Stream的抽象属性和属性方法

(1)CanRead:只读属性,判断该流是否能够读取;

(2)CanSeek:只读属性,判断该流是否支持跟踪查找;

(3)CanWrite:只读属性,判断当前流是否可写;

(4)void Flush():当我们使用流写文件时,数据流会先进入到缓冲区中,而不会立刻写入文件,当执行这个方法后,缓冲区的数据流会立即写入基础流。

(5)Length:流的长度;

(6)Position:表示流中的一个位置。

(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,Origin.Current):表示在流的当前位置往后数第3个位置。

查找之后会返回一个流中的一个新位置,其实说到这大家就能理解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, ,); Console.WriteLine("现在Stream.Postion在第{0}位置",stream.Position+); //从刚才结束的位置(当前位置)往后移3位,到第7位
long newPositionInStream =stream.CanSeek? stream.Seek(, SeekOrigin.Current):; Console.WriteLine("重新定位后Stream.Postion在第{0}位置", newPositionInStream+);
if (newPositionInStream < buffer.Length)
{
//将从新位置(第7位)一直写到buffer的末尾,注意下stream已经写入了3个数据“Str”
stream.Write(buffer, (int)newPositionInStream, buffer.Length - (int)newPositionInStream);
} //写完后将stream的Position属性设置成0,开始读流中的数据
stream.Position = ; // 设置一个空的盒子来接收流中的数据,长度根据stream的长度来决定
readBuffer = new byte[stream.Length]; //设置stream总的读取数量 ,
//注意!这时候流已经把数据读到了readBuffer中
int count = stream.CanRead?stream.Read(readBuffer, , readBuffer.Length):; //由于刚开始时我们使用加密Encoding的方式,所以我们必须解密将readBuffer转化成Char数组,这样才能重新拼接成string //首先通过流读出的readBuffer的数据求出从相应Char的数量
int charCount = Encoding.Default.GetCharCount(readBuffer, , count);
//通过该Char的数量 设定一个新的readCharArray数组
readCharArray = new char[charCount];
//Encoding 类的强悍之处就是不仅包含加密的方法,甚至将解密者都能创建出来(GetDecoder()),
//解密者便会将readCharArray填充(通过GetChars方法,把readBuffer 逐个转化将byte转化成char,并且按一致顺序填充到readCharArray中)
Encoding.Default.GetDecoder().GetChars(readBuffer, , count, readCharArray, );
for (int i = ; i < readCharArray.Length; i++)
{
readString += readCharArray[i];
}
Console.WriteLine("读取的字符串为:{0}", readString);
} stream.Close();
} Console.ReadLine(); }

结果:

好了,关于流的基本介绍和概念,我们就分享到这里。非常感谢 逆时针の风 的博客对我的帮助

关于Stream的知识分享的更多相关文章

  1. MemoryStream相关知识分享

    一.简单介绍一下MemoryStream MemoryStream是内存流,为系统内存提供读写操作,由于MemoryStream是通过无符号字节数组组成的,可以说MemoryStream的性能可以算比 ...

  2. 分享吉林大学机械科学与工程学院,zhao jun 博士的Halcon学习过程及知识分享

    分享吉林大学机械科学与工程学院,zhao jun 博士的Halcon学习过程及知识分享 全文转载zhao jun 博士的新浪博客,版权为zhaojun博士所有 原文地址:http://blog.sin ...

  3. 使用javascript开发的视差滚动效果的云彩 极客标签 - 做最棒的极客知识分享平台

    www.gbtags.com 使用javascript开发的视差滚动效果的云彩 阅读全文:使用javascript开发的视差滚动效果的云彩 极客标签 - 做最棒的极客知识分享平台

  4. 制作一个简单的部门员工知识分享的python抽取脚本

    需求: 基于公司的文化和公司部门间以及员工之间的工作需求状态,或者想要了解某一些技能.专业方面的知识需求.促进并提高员工们的技能认知和技术水平. 详细代码如下: 先说一下存入csv表格的表头字段: 1 ...

  5. XML的相关基础知识分享(二)

    前面我们讲了一下XML相关的基础知识(一),下面我们在加深一下,看一下XML高级方面. 一.命名空间 1.命名冲突 XML命名空间提供避免元素冲突的方法. 命名冲突:在XML中,元素名称是由开发者定义 ...

  6. Webfunny知识分享:JS错误监控

    现在的前端开发已不再是刀耕火种的年代了,各种框架.编译工具层出不穷,前端监控系统也不甘其后,遍地开花. 前端正承受着越来越重的职责,前端的业务也变得越来越复杂,此时此刻我们就更需要一套完善的监控系统来 ...

  7. GO基础知识分享

    目录 GO基础知识分享 go语言的基本事项 关键字 字符串的拼接和变量的定义方式 空白符 const常量 iota的用法 运算符 Go 没有三目运算符,不能适用?: 语言条件语句 for循环的方式 函 ...

  8. PureStudy:学科知识分享——个人网站开发全解

    PureStudy:学科知识分享--个人网站开发全解 项目描述 PureStudy,学科知识分享网站. 学生可以使用这个网站,来浏览相应学科的知识点.学习总结,获取相关的资料.此外,他们可以选择上传文 ...

  9. 【Stream—6】BufferedStream相关知识分享

    一.简单介绍以下BufferedStream 在前几章的讲述中,我们已经能够掌握流的基本特性和特点,一般进行对流的处理时,系统肩负着IO所带来的开销,调用十分频繁,这时候就应该想个办法减少这种开销,而 ...

随机推荐

  1. Docker下Jedis体验

    jedis是redis的java版本的客户端实现,本文通过一些web请求&响应的实例展示了jedis的基本用法: 开始编码前我们先把环境准备好,总共两个server,对应两个docker容器: ...

  2. apache ignite系列(六): 服务网格

    简介 ​ 服务网格本质上还是远程方法调用(RPC),而在ignite中注册的服务本质体现还是以cache的形式存在,集群中的节点可以相互调用部署在其它节点上的服务,而且ignite集群会负责部署服务的 ...

  3. 【学习笔记】第二章 python安全编程基础---python爬虫基础(urllib)

    一.爬虫基础 1.爬虫概念 网络爬虫(又称为网页蜘蛛),是一种按照一定的规则,自动地抓取万维网信息的程序或脚本.用爬虫最大的好出是批量且自动化得获取和处理信息.对于宏观或微观的情况都可以多一个侧面去了 ...

  4. maven的使用解说

    maven周期及项目中的应用: 周期如下: 1.default生命周期,部署项目(jar包的依赖管理) 2.clear生命周期,项目清理工作 3.site生命周期,处理项目中产生的文档信息 应用: 1 ...

  5. Salesforce学习之路-admin篇

    Salesforce是一款非常强大的CRM(Customer Relationship Management)系统,国外企业使用十分频繁,而国内目前仅有几家在使用(当然,国内外企使用的依旧较多),因此 ...

  6. 从零开始入门 K8s| K8s 的应用编排与管理

    作者 | 张振 阿里巴巴高级技术专家 一.资源元信息 1. Kubernetes 资源对象 我们知道,Kubernetes 的资源对象组成:主要包括了 Spec.Status 两部分.其中 Spec ...

  7. response向客户端写入数据

    1.写入文字: protected void doGet(HttpServletRequest request, HttpServletResponse response) throws Servle ...

  8. 即时聊天APP(四) - 联系人和会话

    联系人和会话界面使用的是RecyclerView进行滑动显示,并将好友列表存储至数据库,以供下次登录时使用,RecyclerView在后面我会详细介绍,这里略过. 联系人初始化时读取数据库并展示: / ...

  9. Java第三次作业第三题

    3. 请补充下面的Socket通信程序内容: (1)Socket通信中的服务端程序:ChatServerSocket.java package naizi; import java.io.*; imp ...

  10. 关于瀑布流的布局原理分析(纯CSS瀑布流与JS瀑布流)

    瀑布流 又称瀑布流式布局,是比较流行的一种网站页面布局方式.即多行等宽元素排列,后面的元素依次添加到其后,等宽不等高,根据图片原比例缩放直至宽度达到我们的要求,依次按照规则放入指定位置. 为什么使用瀑 ...