WCF入门 (13)

前言

  公司略无聊,周三前同事推荐跳槽,于是会去更新了一下简历,突然发现,快一年了,我竟然想不出我可以往简历上添加点什么值得自豪的东西。下午和小伙伴聊了一会天,他告诉我,可以往简历上写上“英语口语水平有提高”,惭愧。。。

  希望自己跳槽成功。

  言归正传,几天第13集。

第13集 ExtensionDataObject in WCF WCF里的ExtensionDataObject

  这集讲的是WCF里面的ExtenstionDataObject。 简单来说, ExtenstionDataObject用来保存DataContacts在序列化和反序列化时候的未知的元素。在服务端,当服务端接收到来自客户端的未知元素内容时,这些内容会被存在ExtenstionDataObject里面。在往客户端发送数据的时候,服务必须把数据序列化为XML,这时候服务的序列化进程就会取出ExtensionDataObject里面的内容,然后把他们序列化到XML里面一起发送给客户端。

  下面我们通过一个例子来解释说明一下。

1。 我们来看一下修改过的Employee DataContract类。

    [DataContract]
public class Employee : IExtensibleDataObject
{
[DataMember(IsRequired = true)]
public int Id { get; set; }
//[DataMember(IsRequired = false)]
//public String Name { get; set; }
[DataMember]
public Boolean Gender { get; set; }
[DataMember]
public DateTime DateOfBirth { get; set; }
[DataMember]
public short EmployeeType { get; set; }
[DataMember(IsRequired = true)]
public string City { get; set; } public ExtensionDataObject ExtensionData { get; set; }
}

这里有两点改变,①注释了一个属性Name,②DataContract类实现了IExtensibleDataObject接口,这个接口里定义了一个属性,ExtensionData属性。

因为拿掉了Gender属性,GetEmployee 和 SaveEmployee方法,以及存储过程要稍作修改。这里不说明了。

2。 然后我们把服务端Run起来。

WCF入门 (13)

3。 再把客户端Run起来。

先看看我们数据库里有多少数据:

WCF入门 (13)

总共一条。

然后运行一下客户端的查找方法。

WCF入门 (13)

因为Name属性被注释了,所以从查找出来的结果Name的textbox为空。

3。 下面我们新保存一个Employee。

WCF入门 (13)

如图显示保存成功,但是很明显,数据库里面的Name一定是Null。

WCF入门 (13)

这个Name的Value消失了,如果我们想再获取这个Name的Value,我们应该怎么做? 下面今天的主角登场。 我们来对代码做一些修改。

4。 首先我们对EmployeeService增加一个lastSavedEmployee内部变量。

private static Employee lastSavedEmployee;

这个用来保存最近一次保存的Employee实力。

5。然后是SaveEmployee方法,只需加一句话。

        public bool SaveEmployee(Employee emp)
{
lastSavedEmployee = emp;
var connStr = ConfigurationManager.ConnectionStrings["DefaultConnectionString"].ConnectionString;
using(var conn = new SqlConnection(connStr)) {
conn.Open();
var cmd = conn.CreateCommand();
cmd.CommandType = System.Data.CommandType.StoredProcedure;
cmd.CommandText = "spSaveEmployee";
cmd.Parameters.Add(new SqlParameter("id", emp.Id));
//cmd.Parameters.Add(new SqlParameter("name", emp.Name));
cmd.Parameters.Add(new SqlParameter("gender", emp.Gender));
cmd.Parameters.Add(new SqlParameter("dateOfBirth", emp.DateOfBirth));
cmd.Parameters.Add(new SqlParameter("employeeType", emp.EmployeeType));
return cmd.ExecuteNonQuery() == 1;
}
}

就第一句,每次在save的时候都把lastSavedEmployee更新为传入的emp(这里默认每次都是保存成功的 o(∩_∩)o )

6。 然后是GetEmployee方法,

        public Employee GetEmployeeById(int id)
{
var connStr = ConfigurationManager.ConnectionStrings["DefaultConnectionString"].ConnectionString;
using(var conn = new SqlConnection(connStr)) {
conn.Open();
var cmd = conn.CreateCommand();
cmd.CommandType = System.Data.CommandType.StoredProcedure;
cmd.CommandText = "spGetEmployeeById";
cmd.Parameters.Add(new SqlParameter("id", id)); Employee emp = null;
var reader = cmd.ExecuteReader();
emp = reader.HasRows ? new Employee() : null;
while(reader.Read()) {
emp.Id = Convert.ToInt32(reader["Id"]);
//emp.Name = Convert.ToString(reader["Name"]);
emp.Gender = Convert.ToBoolean(reader["Gender"]);
emp.DateOfBirth = Convert.ToDateTime(reader["DateOfBirth"]);
emp.EmployeeType = Convert.ToInt16(reader["EmployeeType"]);
}
if(lastSavedEmployee != null && lastSavedEmployee.Id == id) {
emp.ExtensionData = lastSavedEmployee.ExtensionData;
}
return emp;
}
}

我们来看一下return前面的那个if语句块, 如果lastSavedEmployee不为空,并且lastSaveEmployee的id和传入的id是同一个,那么就复制return的emp的ExtenstionData属性为lastSavedEmployee的ExtentionData属性。

我们来做个测试。按下图保存一个Employee。

WCF入门 (13)

然后在SaveEmployee里面的第一句给lastSavedEmployee句子打个断点。添加一个监视,来看看我们可以拿到什么。

WCF入门 (13)

展开ExtentionData属性,里面有个Members属性,是个集合。Name为“Name”,Value为“TestName”,也就是我们输入的值。

点击保存,提示保存成功,check一下数据库里面的值。

WCF入门 (13)

Name为Null。

7。然后我们调用一下查找方法。

输入Id=3.

WCF入门 (13)

可以看到,TestName这个值虽然没有存入数据库,但还是被Preserve下来了。

这集介绍了IExtensibleDataObject接口,因为本身也没有太多WCF项目的经验,所以暂时也不晓得这个东西在什么业务场景下比较给力。希望有经验的同学给举个好点的例子。

Thank you。

上一篇:java json-lib配置


下一篇:Window环境下Python和Django的安装