System.IO 二
接着上篇的来 System.IO
FileSystemWatcher
指向这个签名的方法
可以监听目录发生了什么事件
例如:
static void Main(string[] args)
{
Console.WriteLine("请开始你的表演:");
FileSystemWatcher watcher = new FileSystemWatcher();
watcher.Path = @"E:\Test"; //此目录一定需要存在,不然会引发 ArgumentException 异常
//设置需要 留意 的事情
watcher.NotifyFilter = NotifyFilters.LastAccess | NotifyFilters.LastWrite | NotifyFilters.FileName | NotifyFilters.DirectoryName;
//只观察文本文件
watcher.Filter = "*.txt";
//添加事件处理程序
watcher.Changed += new FileSystemEventHandler(Onchange);
watcher.Created += new FileSystemEventHandler(Onchange);
watcher.Deleted += new FileSystemEventHandler(Onchange);
watcher.Renamed += new RenamedEventHandler(OnRenamed);
//开始观察目录
watcher.EnableRaisingEvents = true;
//等待用户退出程序
Console.WriteLine("请按 `q` 退出程序");
while (Console.Read() !='q');
}
static void Onchange(object source, FileSystemEventArgs e)
{
//指定当文件改变、创建或者删除的时候需要做的事情
Console.WriteLine("File:{0} {1}",e.FullPath,e.ChangeType);
}
static void OnRenamed(object source, RenamedEventArgs e)
{
//指定当文件重命名的时候需要做的事情
Console.WriteLine("File: {0} renamed to {1}",e.OldFullPath,e.FullPath);
}
对象序列化
序列化描述了持久化一个对象的状态到流(如文件流和内存流)的过程。序列化服务保存应用程序数据相对于IO命名空间的读读取器/编写器,减少了很多的麻烦。
在看列子之前,你先要理解一个类。
[Serializable] //序列化的对象需要标记为可以序列化
public class UserProp
{
public string WindowColor;
public int FontSize;
} Console.WriteLine("请开始你的表演:");
UserProp userData = new UserProp();
userData.WindowColor = "red";
userData.FontSize = ; //将一个对象保存到一个本地
BinaryFormatter binFormat = new BinaryFormatter();
using (Stream fStrem=new FileStream("Haha.txt", FileMode.Create, FileAccess.Write, FileShare.None))
{
binFormat.Serialize(fStrem,userData); //将对象序列化为流
}
Console.ReadLine();
虽然对象序列化保存对象非常简单,单幕后的调用过程非常复杂。除了保存为二进制格式外还可以保存为简单对象访问协议(SOAP)或XML格式。
最后要知道对象图可以持久化为任意的System.IO.Stream派生类型。
一组关联对象(包括基类和派生类)被总称为一个对象图。对象图提供一种很简明的方式来记载一组对象如何相互引用对方。
为序列化配置对象
为每个一个关联 的类(或结构)加上 [serializable] 特性,不参与到序列化配置中,可以在这么域前面加上 [NoSerialized]特性。
定义可序列化对象
[Serializable]
public class Radio
{
public bool hasTweeters;
public bool hasSubWoofers; [NonSerialized] //把此变量设置为不可序列化的
public string radioID = "X-Y110";
} [Serializable]
public class Car
{
public Radio theRadio = new Radio();
public bool isHatchBack;
} [Serializable]
public class JamesBondCar : Car
{
public bool canFly;
public bool canSubmerge;
}
注:[Serializable]特性不能被继承。因此,如果从被标记为[Serializable]的类派生一个类,子类也必须被标记为[Serializable],否则它不能持久化。
有一句话叫做公共的属性,私有的字段
一般的情况,是在一个类中,我们把字段定义为私有的,属性则定义为公共的。这句的来源则是我们在一个类中声明一个字段,然后把这个字段进行重构,封装字段。
[Serializable]
public class Person
{
//公共字段
public bool isAlive = true; //私有字段
private int personAge = ; //公共属性,私有字段
private string fName;
public string FName
{
get { return fName; }
set { fName = value; }
}
我们使用 BinaryFormatter或 SoapFormatter进行处理,都可以保存到所选的流中。然而,XmLSerializer不会保存personAge的值。
序列化格式方法
一旦将类型配置为参与.NET序列化,接下来就是选择当持久化对象图时使用哪种格式(二进制、SOAP或XML),有以下3中选择:
BinaryFormatter:
SoapFormatter:
XmlSerializer:
在引用不出命名空间的时候请去引用里面添加引用。
第一个上面说过了,是二进制格式。
这里我们可以了解下它的属性和方法,主要是序列化和反序列化的方法。SOAP是一个标准的XML格式。
有很的重载构造函数。
下面会介绍如何使用这三种格式序列化对象。
这三个都市直接派生于System.Object.
IFormatter和IremotingFormatting接口
我们来看看IFormatter接口
它们两个类继承同一个接口,而此接口中这两个主要的方法刚好符合我们的要求 。所有我们建立一个方法使用这两个类中的一个来序列化一个对象图。
/// <summary>
/// 两个类的通用方法
/// </summary>
/// <param name="itfFormat">BinaryFormatter或SoapFormatter的实例</param>
/// <param name="destStrem">准备的Stream流</param>
/// <param name="graph">需要写的序列化对象</param>
static void SerializeObjectGraph(IFormatter itfFormat, Stream destStrem, object graph)
{
itfFormat.Serialize(destStrem,graph);
}
在格式化程序中的类型保真
在3种格式化程序中,最明显不同是对象图被持久化为不同的流(二进制、SOAP或XML)的方式。BinaryFormatter在希望用值(例如以一个完整的副本)跨越.NET应用程序机器边界传递对象时成为理想的选择。当希望尽可能延伸持久化对象图的使用范围时,SoapFormatter和XmlSerializer是理想选择。
①使用BinaryFormatter序列化对象
Ⅰ Serialize():将一个对象图按字节顺序转化为一个对象图。
Ⅱ Deserialize():讲一个持久化的字节顺序转化为一个对象图。
static void Main(string[] args)
{
Console.WriteLine("请开始你的表演:"); JamesBondCar jbc = new JamesBondCar();
jbc.canFly = true;
jbc.canSubmerge = false;
jbc.theRadio.stationPresets = new double[] { 89.3, 105.1, 97.1 };
jbc.theRadio.hasTweeters = true; //以二进制格式保存car
SaveAsBinaryFormat(jbc,"CarDara.dat"); Console.ReadLine(); }
static void SaveAsBinaryFormat(object objGraph, string fileName)
{
BinaryFormatter binFormat = new BinaryFormatter();
using (Stream fStream=new FileStream(fileName,FileMode.Create,FileAccess.Write,FileShare.None))
{
binFormat.Serialize(fStream, objGraph);
}
Console.WriteLine("保存成功了");
}
使用BinaryFormatter反序列化对象,SOAP和XML反序列化类似
/// <summary>
/// 反序列化
/// </summary>
/// <param name="fileName">文件名</param>
static void LoadFormBinaryFile(string fileName)
{
BinaryFormatter binFormat = new BinaryFormatter();
using (Stream fStream=File.OpenRead(fileName))
{
JamesBondCar carFormDisk =(JamesBondCar)binFormat.Deserialize(fStream); //Deserialize返回的是object类型,转下型然后接收就可以了
Console.WriteLine("fly属性:{0}",carFormDisk.canFly);
}
}
②使用SoapFprmatter序列化对象
static void SaveAsSoapFormat(object objGraph,string fileName)
{
SoapFormatter soapFomat = new SoapFormatter();
using (Stream fStream=new FileStream(fileName,FileMode.Create,FileAccess.Write,FileShare.None))
{
soapFomat.Serialize(fStream,objGraph);
}
Console.WriteLine("保存Soap格式成功");
}
③使用XmlSerializer序列化对象
static void SaveAsXmlFormat(object objGraph, string fileName)
{
//此类的构成函数必须给参数,需要确定你对谁进行序列化
XmlSerializer xmlFormat = new XmlSerializer(typeof(JamesBondCar));
using (Stream fStream=new FileStream(fileName,FileMode.Create,FileAccess.Write,FileShare.None))
{
xmlFormat.Serialize(fStream,objGraph);
}
Console.WriteLine("序列化XML完成");
}
XmlSerializer要求对象图中的所有序列化类型支持默认的构造函数。没有的话需要加上
上面三种可以看出,XML格式是最清楚和简单的。
控制生成的XML数据
System.Xml.Serialization命名空间中的部分特性
例如:
如果想更深一点的了解可以去官网查看
序列化对象集合
前面已经演示了如何将一个对象持久化为一个流,下面演示如何保存一组对象。
一组对象就需要用到集合了。(数组、ArrayList或List<T>)。用数组代替我们之前用的单个对象就可以了
List<JamesBondCar> myCars = new List<JamesBondCar>()
{
new JamesBondCar(){ canFly=true, canSubmerge=false, isHatchBack=false},
new JamesBondCar(){ canFly=true, canSubmerge=true, isHatchBack=false},
new JamesBondCar(){ canFly=true, canSubmerge=false, isHatchBack=true},
}; //以二进制格式保存car
// SaveAsBinaryFormat(jbc, "CarDara.dat");
// LoadFormBinaryFile("CarDara.dat");
//SaveAsSoapFormat(jbc,"CarData.soap");
SaveAsXmlFormat(myCars,"CarData.xml");
Console.ReadLine();
static void SaveAsXmlFormat(List<JamesBondCar> objGraph, string fileName)
{
//此类的构成函数必须给参数,需要确定你对谁进行序列化
XmlSerializer xmlFormat = new XmlSerializer(typeof(List<JamesBondCar>));
using (Stream fStream=new FileStream(fileName,FileMode.Create,FileAccess.Write,FileShare.None))
{
xmlFormat.Serialize(fStream,objGraph);
}
Console.WriteLine("序列化XML完成");
}
另外两个类似。
自定义Soap/Binary序列化过程
System.Runtime.Serizlization命名空间核心类型
深入了解对象序列化
当BinaryFormatter序列化一个对象图时,它负责传送下面的信息到指定的流。
①在对象图中对象的完全限定名(如 MyApp.Car)
②定义对象图的程序名称(如 MyApp.exe)
③SerizlizationInfo 类的一个实例,包含了所有由对象图成员保存的所有描述性数据
SoapFormatter也类似。
使用ISerizlizable自定义序列化
被标记了[Serializable]的对象拥有了实现ISerializable接口的选项。
这个接口就一个方法:
AddValue这个重载很多。
实现了ISerizlizable接口的类型也必须定义一个带有下面签名的特殊构造函数:
例如:
[Serializable]
public class StringData : ISerializable
{
private string dataItemOne = "First data block";
private string dataItemTwo = "More data";
public StringData()
{ }
protected StringData(SerializationInfo si,StreamingContext ctx)
{
//从流中得到合并的成员变量
dataItemOne = si.GetString("First_Item").ToLower();
dataItemTwo = si.GetString("dataItemTwo").ToLower();
}
public void GetObjectData(SerializationInfo info, StreamingContext context)
{
//用格式化数据填充SerizlizationInfo对象
info.AddValue("First_Item", dataItemOne.ToUpper());
info.AddValue("dataItemTwo", dataItemTwo.ToUpper());
}
}
StringData myData = new StringData();
SoapFormatter soapFprmat = new SoapFormatter();
using (Stream fStream=new FileStream("MyData.soap",FileMode.Create,FileAccess.Write,FileShare.None))
{
soapFprmat.Serialize(fStream,myData);
}
可以简单的完成
[OnSerializing]
private void OnSerializing(StreamingContext context) //需要带上此参数
{
//在序列化的过程就调用
dataItemOne = dataItemOne.ToUpper();
dataItemTwo = dataItemTwo.ToUpper();
} [OnDeserialized]
private void OnDeserialized(StreamingContext context)
{
//在反序列化的过程就调用
dataItemOne = dataItemOne.ToLower();
dataItemTwo = dataItemTwo.ToLower();
}
这个命名空间介绍完了,涉及的类有点多。。。
System.IO 二的更多相关文章
- C#、.Net代码精简优化(空操作符(??)、as、string.IsNullOrEmpty() 、 string.IsNullOrWhiteSpace()、string.Equals()、System.IO.Path 的用法)
一.空操作符(??)在程序中经常会遇到对字符串或是对象判断null的操作,如果为null则给空值或是一个指定的值.通常我们会这样来处理: .string name = value; if (name ...
- System.IO中的File、FileInfo、Directory与DirectoryInfo类(实例讲解)
一.建立的文件夹(对这些文件进行以上四个类的操作): 父目录: 父目录的子目录以及父目录下的文件: 子目录下的文件: 二.效果图 三.代码实现 using System; using System.I ...
- 多种下载文件方式 Response.BinaryWrite(byte[] DocContent);Response.WriteFile(System.IO.FileInfo DownloadFile .FullName);Response.Write(string html2Excel);
通过html给xls赋值,并下载xls文件 一.this.Response.Write(sw.ToString());System.IO.StringWriter sw = new System.IO ...
- C# System.IO.Path
Path的常用方法 函数列表 对一个路径做相应操作,包括文件路径,目录路径,通常会用到Path这个类, 本文列举一些常用的操作. 获取指定路径字符串的目录信息 public static string ...
- C# System.IO 文件流输入输出
一.读写文本文件 可以用fileStream来读写文本文件,但是FileStream是通过字节形式来读写数据的,要把字节数据转换为文本,要自己处理编码转换. 对于文本文件的读写,通常用 StreamR ...
- Pipe——高性能IO(二)
Pipelines - .NET中的新IO API指引(一) Pipelines - .NET中的新IO API指引(二) 关于System.IO.Pipelines的一篇说明 System.IO.P ...
- 【等待事件】等待事件系列(3+4)--System IO(控制文件)+日志类等待
[等待事件]等待事件系列(3+4)--System IO(控制文件)+日志类等待 1 BLOG文档结构图 2 前言部分 2.1 导读和注意事项 各位技术爱好者,看完本文后,你可 ...
- Fiddler的一些坑: !SecureClientPipeDirect failed: System.IO.IOException
手机的请求Fiddler可以捕捉,但是手机一直无法上网,在logs中看到的日志如下: !SecureClientPipeDirect failed: System.IO.IOException 由于远 ...
- 服务 在初始化安装时发生异常:System.IO.FileNotFoundException: "file:///D:\testService"未能加载文件或程序集。系统找不到指定文件。
@echo.@if exist "%windir%\Microsoft.NET\Framework\v4.0.30319\InstallUtil.exe" goto INSTALL ...
随机推荐
- Robot Framework自动化测试(一)
=============所需要环境========== Python: https://www.python.org/ RF框架是基于python 的,所以一定要有python环境. Robot f ...
- 在windows上安装common lisp开发环境
(2014.1写于CSDN的文章) 最近对lisp非常感兴趣,因此在google中搜索了“common lisp install windows”, 想装一个开发环境玩玩. 第一条结果就是 “Gett ...
- zabbix--高级篇-监控docker服务(一)
一,配置zabbix 客户端环境 rpm -ivh https://mirrors.aliyun.com/zabbix/zabbix//x86_64/zabbix-release-.el7.noarc ...
- 【LDAP】LDAP介绍
原文:http://ldapman.org/articles/intro_to_ldap.html原文作者:Michael Donnelly 什么是LDAP? LDAP的英文全称是Lightweigh ...
- maven实战迷你版记录
1. ~/.m2 文件 默认情况下,该文件夹下放置了 Maven 本地 仓库.m2/repository.所有的 Maven 构件(artifact)都被存储到该仓库中,以方便重用. 默认情况下,~ ...
- EditPlus编写PHP使用技巧
1,建立php模板 方法:在EditPlus的文件目录下,新建template.php文件,写入<?php ?>内容保存,再在editplus的模板中 载入应用即可. 2,建立函数自动补齐 ...
- ubuntu 下安装配置LAMP
详情见: http://www.linuxeden.com/html/softuse/20130731/141934.html
- If you want the rainbow, you have to deal with the rain.
If you want the rainbow, you have to deal with the rain.想要彩虹,就先忍受雨水.
- php cur错误:SSL错误 unable to get local issuer certificatebool
采集https链接时出现的问题 办法:跳过SSL证书检查 curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); curl_setopt($ch, CURLO ...
- ansible使用7-Loops
Standard Loops with_items - name: add several users user: name={{ item }} state=present groups=wheel ...