【.net 深呼吸】聊聊WCF服务返回XML或JSON格式数据
有时候,为了让数据可以“跨国经营”,尤其是HTTP Web有关的东东,会将数据内容以 XML 或 JSON 的格式返回,这样一来,不管客户端平台是四大文明古国,还是处于蒙昧时代的原始部落,都可以使用这些数据。
在WCF中实现将数据以XML或JSON格式返回有Y多种方法,不管你用什么方法,只要得到预期结果就好,米芾说了,笔可以八面出锋,当然了,人家指的是绘画。
这里,老周就挑两种方法来演示,仅供参考,没有考古价值,建议司马子长不要把本文收入《史记》。
第一种方法是用到 WebServiceHost 类,它可以自动完成一些与HTTP通信相关的配置,不过,使用该类,要以管理身份运行,不然,会无权限监听。
首先定义一个 Book 类,稍后咱们会把一个Book实例以XML或JSON数据返回。
public sealed class Book
{
public string BookName { get; set; }
public decimal Price { get; set; }
public string BarCode { get; set; }
}
然后,很重要一步,就是声明服务协定,它是个接口,可以对客户端公开,当然客户端也可以重新定义。
[ServiceContract]
interface IService
{
[OperationContract]
[WebGet(UriTemplate = "getdata?f={format}")]
Message GetXml(string format);
}
加上ServiceContract特性表明它是服务协定,如果没有明确指定Name,则它的名字与接口的名字相同;协定接口中,希望向客户端公开的方法要加上OperationContract特性,否则不会被认为是服务操作,无法被客户端使用。
服务协定接口允许在服务器和客户端使用不同定义,只要协定的名称相同,并且方法的参数和返回值类型和数目相同即可。
WebGet特性指定URI的使用方法,地址为相对路径,假如基址是http://dog.net/,那么访问GetXml方法的路径为 http://dog.net/getdata?f=xml。
本来我只想返回XML数据的,所以叫GetXml,后来一想,单返回XML格式的内容也太小气了,索性弄一个参数,来指定格式,可以传入xml或json。
?f后面的{format}会自动把值传给方法的format参数,所以,UriTemplate的参数名字不要写错,如果写成 ?f={firmat},那就识别不了参数了。
然后要实现服务,实现协定接口的类型不必向客户端公开,因为它是在服务器上执行的。
public class MyService : IService{
public Message GetXml(string format)
{
WebOperationContext context = WebOperationContext.Current;
Book b = new Book
{
BookName = "卖女孩的小火柴",
Price = 25.2M,
BarCode = "2811365801"
};
Message msgreturn = null;
// 判断格式
if (format.ToLower() == "xml")
{
msgreturn = context.CreateXmlResponse<Book>(b);
}
else
{
msgreturn = context.CreateJsonResponse<Book>(b);
}
return msgreturn;
}
}
这里通过一个很好玩的方法来完成,所以方法返回类型为Message。静态属性WebOperationContext.Current可以得到与当前调用的操作协定关联的上下文对象,即WebOperationContext实例。
它公开了一堆方法,名字都是 CreateXXXResponse,其中XXX是啥取决于返回内容,
要返回JSON,就调用CreateJsonResponse方法,
返回XML就调用CreateXmlResponse方法。
实例化Book对象后,可以传给带泛型参数的CreateJsonResponse或CreateXmlResponse方法,
把类型参数T指定为Book,就会自动把Book对象序列化,然后返回给客户端。
最后,在配置文件中给服务设定一个基址,可以在代码中写,也可以在配置文件中写,此处老周选用配置文件
好处是可以动态修改而不必重新编译应用程序。
<system.serviceModel>
<services>
<service name="getXmlSample.MyService">
<host>
<baseAddresses>
<add baseAddress="http://localhost:1888/"/>
</baseAddresses>
</host>
</service>
</services>
</system.serviceModel>
可能有初学的朋友说WCF的配置文件很难写,其实啊,是有规律的,你不妨细心研究一下,掌握规律后你会发现配置文件并不难写。
service的 name 属性的值就是服务类的Type的名字(类型名,带命名空间名称)。
在Main中实例化WebServiceHost
static void Main(string[] args){
WebServiceHost host = new WebServiceHost(typeof(MyService));
host.Open();
Console.WriteLine("服务已打开。");
Console.Read();
host.Close();
}
注意,传给构造函数的Type是服务类的类型,与配置文件中service/name的值相同。
以管理员身份运行这个例子,然后打开浏览器,输入http://localhost:1888/getdata?f=xml,回车后,你会看到这样的内容:
把xml改为json,再看看。
怎么样,好玩吧。下面老周再演示另一种方法。
这种方法没使用WebServiceHost,而是使用普通的ServiceHost类来承载服务
可通过WebHttpBinding来得到HTTP交互的支持,不过,不要忘了给终结点配置WebHttpBehavior行为。
同样,先定义一个类,随后用来做测试。
[DataContract(Namespace = "http://sample",Name = "student")]
public sealed class Student
{
[DataMember(Name = "stu_id")]
public int StuID { get; set; }
[DataMember(Name = "stu_name")]
public string StuName { get; set; }
}
这一次,咱们通过将对象进行XML或JSON序列化的方式生成数据,并转为字符串返回。服务协定如下:
[ServiceContract]
public interface IData
{
[OperationContract]
[WebGet(UriTemplate = "getdata?f={format}")]
string GetData(string format);
}
和前面差不多,只是返回类型改为string。
下面代码实现协定接口:
public class MyService : IData
{
public string GetData(string format)
{
string res = null;
Student stu = new Student
{
StuID = 3, StuName = "小白"
};
using (MemoryStream ms=new MemoryStream())
{
XmlObjectSerializer sz = null;
if (format != null && format.ToLower() == "xml")
{
sz = new DataContractSerializer(stu.GetType());
}
else
{
sz = new DataContractJsonSerializer(stu.GetType());
}
sz.WriteObject(ms, stu);
res = Encoding.UTF8.GetString( ms.ToArray());
}
return res;
}
}
接着,在配置文件中配置一下。
<system.serviceModel>
<behaviors>
<endpointBehaviors>
<behavior name="hb">
<webHttp automaticFormatSelectionEnabled="true"/>
</behavior>
</endpointBehaviors>
</behaviors>
<services>
<service name="getXmlSample2.MyService">
<endpoint address="http://localhost:2008" binding="webHttpBinding" contract="getXmlSample2.IData" behaviorConfiguration="hb"/>
</service>
</services>
</system.serviceModel>
behaviors节点下可以配置两种行为——服务行为和终结点行为。此处我们只需配置终结点的行为,需要一个webHttp元素,它映射到 WebHttpBehavior 类。记得要为behavior节点分配名字,随后在/services/service/endpoint节点下,才能通过behaviorConfiguration属性来引用前面的behavior。
实现HTTP交互,应使用webHttpBinding。
在配置webHttp行为时,应该把automaticFormatSelectionEnabled的值设置为true,这样一来,返回给调用方的内容会自动识别格式,其实主要目的是让返回的字符串中能够去掉最外层的双引号。
回到代码,实例化ServiceHost,然后打开服务。
static void Main(string[] args)
{
using (ServiceHost host=new ServiceHost(typeof(MyService)))
{
host.Open();
Console.WriteLine("服务已打开。");
Console.Read();
}
}
运行应用程序,在浏览中输入http://localhost:2008/getdata?f=xml,得到结果如下。
然后再输入http://localhost:2008/getdata?f=json看看。
好了,就演示这两种方法吧,你愿意探索的话,方法是有很多种的。