Интернет-журнал "Домашняя лаборатория", 2007 №9
Шрифт:
В пространстве имен библиотеки FCL:
System.Runtime.Serialization.Formatters.Soap
находится класс SoapFormatter. Он является наследником тех же интерфейсов IFormatter и IRemotingFormatter и реализует их методы Serialize и Deserialize, позволяющие выполнять глубокую сериализацию и десериализацию при сохранении данных в формате xml. Помимо методов класса SoapFormatter, xml-сериализацию можно выполнять средствами другого класса — XmlSerializer.
Из
В качестве примера промоделируем сказку Пушкина "О рыбаке и рыбке". Как вы помните, жадная старуха богатела, богатела, но после очередного желания оказалась у разбитого корыта, вернувшись в начальное состояние. Сериализация позволит нам запомнить начальное состояние, меняющееся по мере выполнения рыбкой первых пожеланий рыбака и его старухи. Десериализация вернет все в начальное состояние. Опишу класс, задающий героев пушкинской сказки:
[Serializable]
public class Personage
{
public Personage(string name, int age)
{
this.name = name; this.age = age;
}
//поля класса static int wishes;
public string name, status, wealth; int age;
public Personage couple;
//методы класса
}
Герои сказки — объекты этого класса обладают свойствами, задающими имя, возраст, статус, имущество и супруга. Имя и возраст задаются в конструкторе класса, а остальные свойства задаются в следующем методе:
public void marry (Personage couple)
{
this.couple = couple;
couple.couple = this;
this.status ="крестьянин";
this.wealth ="рыбацкая сеть";
this.couple.status = "крестьянка";
this.couple.wealth = "корыто";
SaveState ;
}
Предусловие метода предполагает, что метод вызывается один раз главным героем (рыбаком). В методе устанавливаются взаимные ссылки между героями сказки, их начальное состояние. Завершается метод сохранением состояния объектов, выполняемого при вызове метода SaveState:
void SaveState
{
BinaryFormatter bf = new BinaryFormatter ;
FileStream fs = new FileStream
("State.bin",FileMode.Create, FileAccess.Write);
bf.Serialize(fs,this);
fs.Close ;
}
Здесь и выполняется сериализация графа объектов. Как видите, все просто. Вначале создается форматер — объект bf класса BinaryFormatter. Затем определяется файл, в котором будет сохраняться состояние объектов, — объект fs класса FileStream. Заметьте, в конструкторе файла, кроме имени файла, указываются его характеристики: статус, режим доступа. На деталях введения файлов я останавливаться не буду. Теперь, когда основные объекты определены, остается вызвать метод Serialize объекта bf, которому в качестве аргументов передается объект fs и текущий объект, представляющий корневой объект графа объектов, которые подлежат сериализации. Глубокая сериализация, реализуемая в данном случае, не потребовала от нас никаких усилий.
Нам понадобится еще метод,
описывающий жизнь героев сказки:public Personage AskGoldFish
{
Personage fisher = this;
if (fisher.name == "рыбак")
{
wishes++;
switch (wishes)
{
case 1: ChangeStateOne;break;
case 2: ChangeStateTwo;break;
case 3: ChangeStateThree;break;
default: BackState(ref fisher);break;
}
}
return(fisher);
}//AskGoldFish
Метод реализует анализ желаний героини сказки. Первые три желания исполняются, и состояние героев меняется:
void ChangeStateOne
{
this.status = "муж дворянки";
this.couple.status = "дворянка";
this.couple.wealth = "имение";
}
void ChangeStateTwo
{
this.status = "муж боярыни";
this.couple.status = "боярыня";
this.couple.wealth = "много поместий";
}
void ChangeStateThree
{
this.status = "муж государыни";
this.couple.status = "государыня";
this.couple.wealth = "страна";
}
Начиная с четвертого желания, все возвращается в начальное состояние — выполняется десериализация графа объектов:
void BackState(ref Personage fisher)
{
BinaryFormatter bf = new BinaryFormatter ;
FileStream fs = new FileStream
("State.bin",FileMode.Open, FileAccess.Read);
fisher = (Personage)bf.Deserialize(fs);
fs.Close;
}
Обратите внимание, что у метода есть аргумент, передаваемый по ссылке. Этот аргумент получает значение — ссылается на объект, создаваемый методом Deserialize. Без аргумента метода не обойтись, поскольку возвращаемый методом объект нельзя присвоить текущему объекту this. Важно также отметить, что метод Deserialize восстанавливает весь граф объектов, возвращая в качестве результата корень графа. В классе определен еще один метод, сообщающий о текущем состоянии объектов:
public void About
{
Console.WriteLine("имя = {0}, возраст = {1},"+
"статус = {2}, состояние ={3}",name,age,status, wealth);
Console.WriteLine("имя = {0}, возраст = {1}," +
"статус = {2}, состояние ={3}", this.couple.name,
this.couple.age,this.couple.status, this.couple.wealth);
}
Для завершения сказки нам нужно в клиентском классе создать ее героев:
public void TestGoldFish
{
Personage fisher = new Personage("рыбак", 70);
Personage wife = new Personage("старуха", 70);
fisher.marry(wife);
Console.WriteLine("До золотой рыбки"); fisher.About;
fisher = fisher.AskGoldFish;
Console.WriteLine("Первое желание"); fisher.About;
fisher = fisher.AskGoldFish;
Console.WriteLine("Второе желание"); fisher.About;
fisher = fisher.AskGoldFish;
Console.WriteLine("Третье желание"); fisher.About;
fisher = fisher.AskGoldFish;
Console.WriteLine("Еще хочу"); fisher.About;
fisher = fisher.AskGoldFish;
Console.WriteLine("Хочу, но уже поздно"); fisher.About;
}
На рис. 19.6 показаны результаты исполнения сказки.
Рис. 19.6. Сказка о рыбаке и рыбке
Что изменится, если перейти к сохранению данных в xml-формате? немногое. Нужно лишь заменить объявление форматера:
void SaveStateXML
{
SoapFormatter sf = new SoapFormatter;