• 目的

使用Wcf(C#)搭建一个Restful Service

  • 背景

最近接到一个项目,客户要求使用Restful 方式接收到数据,并对数据提供对数据的统计显示功能,简单是简单,但必须要使用Restful方式,客户端传递数据就必须使用Rest ful的格式的url,提交方式为Post。

其实在之前的公司里边就开发过一个restful服务,但只记得自己使用的wcf,而今已经忘记的差不多了,还以为只是简单的搭建一个wcf就可以,今天起初就创建了一个wcf项目,发现wcf项目http://localhost:6001/Test.svc好像很让人纠结就是这个url就是多出一个。svc,wsd什么之类的,根本就不是restful,后来上网上搜到原来我们之前不是用的简单的wcf工程,而是通过Wcf rest service template工程创建的。

  • 创建Wcf rest service template工程:

在vs 新建工程-》online templates-》在搜索框中输入Wcf rest service template-》

创建好工程的工程:

  • 这个工程值得注意的有以下地方:

1, Gobal.asax文件中的代码:

  public class Global : HttpApplication
{
void Application_Start(object sender, EventArgs e)
{
RegisterRoutes();
} private void RegisterRoutes()
{
// Edit the base address of Service1 by replacing the "Service1" string below
RouteTable.Routes.Add(new ServiceRoute("Service1", new WebServiceHostFactory(), typeof(Service1)));
}
}

这里边是注册了url访问 route,这个访问格式就包含一下集中restful访问类型:

2.访问方式

备注:我默认的url host url为:http://localhost:6001/

访问url:http://localhost:6001/Service1           提交方式:GET      调用函数为:GetCollection

访问url:http://localhost:6001/Service1           提交方式:POST    调用函数为:Create

访问url:http://localhost:6001/Service1/1        提交方式:Get       调用函数为:Get(...)

访问url:http://localhost:6001/Service1/1        提交方式:PUT       调用函数为:Update(...)

访问url:http://localhost:6001/Service1/1        提交方式:PUT       调用函数为:Delete(...)
3.怎么注册访问方式

[WebGet(UriTemplate = "")]

List<SampleItem> GetCollection()

[WebInvoke(UriTemplate = "", Method = "POST")]
SampleItem Create(SampleItem instance)

[WebGet(UriTemplate = "{id}")]
SampleItem Get(string id)

[WebInvoke(UriTemplate = "{id}", Method = "PUT")]
SampleItem Update(string id, SampleItem instance)

[WebInvoke(UriTemplate = "{id}", Method = "DELETE")]
void Delete(string id)

其实我们可以注意到这些"访问url"访问到的“调用的函数”上的注解:

指定了访问url为:http://localhost:6001/Service1

WebGetAttribute

a. 决定了提交方式为Get,

b. UriTemplate ,决定了是否在http://localhost:6001/Service1后边追加其他路径

c. 其他参数RequestFormat = WebMessageFormat.Xml, ResponseFormat = WebMessageFormat.Json等

WebInvokeAttribute

a.UriTemplate ,决定了是否在http://localhost:6001/Service1后边追加其他路径

b.Method,决定了提交方式,可选参数包含Get,Post,Put,Delete,当然也支持Header等其他方式。

c. 其他参数RequestFormat = WebMessageFormat.Xml, ResponseFormat = WebMessageFormat.Json等

  • 关于请求、返回数据格式

REST被受到欢迎原因:

1,分布式架构

2,请求、返回格式支持xml,json。

3,支持异步请求/提交数据,这使得Javascript可以轻松调用RESTful服务。

A.如果手动配置请求、返回数据格式?

wcf rest service template内置了简单的注册方式:在上边我们提到的WebGetAttribute,WebInvokerAttribute中都支持参数RequestFormat,ResponseFormat,可以手动的在代码中制定他们的参数类型:

  // Summary:
// An enumeration that specifies the format of Web messages.
public enum WebMessageFormat
{
// Summary:
// The XML format.
Xml = ,
//
// Summary:
// The JavaScript Object Notation (JSON) format.
Json = ,
}

B.如果动态配置请求、返回数据格式?

相比手动配置之下,动态选择格式还是更灵活。打开之前创建的项目,在web.config中,设置:

<standardEndpoint name="" helpEnabled="true" automaticFormatSelectionEnabled="true"/>

将automaticFormatSelectionEnabled设置为true

设置后,我们就可以通过url访问到当前我们可以访问的路径接口等.

  • 测试客户端

比如我们Restful Service代码如下:

  [WebInvoke(UriTemplate = "Servlet", Method = "POST", ResponseFormat = WebMessageFormat.Json)]
public string Welcome()
{
try
{
// 接收传递的参数信息
string param= HttpContext.Current.Request["param"]; // do somthing // 返回成功操作状态信息
return "{\"status\": 0, \"msg\": \"操作成功!\"}"; }
catch (Exception ex)
{ // 返回失败操作状态信息
return "{\"status\": 1, \"msg\": \"操作失败!\"}"; ;
}
}

客户端调试:

static void Main(string[] args)
{
NameValueCollection responseEntity = new NameValueCollection();
responseEntity.Add("param", ""); string baseUrl = "http://localhost:6001/TestReceiver/Welcome"; try
{
WebClient webClient = new WebClient(); // 采取POST方式必须加的header,如果改为GET方式的话就去掉这句话即可
webClient.Headers.Add("Content-Type", "application/x-www-form-urlencoded"); // 得到返回字符流 
byte[] responseData = webClient.UploadValues(baseUrl, "POST", responseEntity); // 解码返回值 
string byRemoteInfo = Encoding.UTF8.GetString(responseData); Console.WriteLine(byRemoteInfo); }
catch (Exception ex)
{
Console.WriteLine(ex.Message);
Console.WriteLine(ex.StackTrace);
} Console.ReadKey();
}

  上边我们采用的接收参数的方式是HttpContext.Current.Request.Params["param"]的方式来接收参数,确切说,这种方式在vs的调试模式下是可以接收到参数的,但是在iis上就不是这么好说了(具体问题,我不是太清楚是不是因为我的服务器上没有注册WCF组建的方式(验证方案参考:http://blog.csdn.net/findsafety/article/details/9045721),还是其他问题;具体原因我暂时不能去确认,服务器远程不了,只能明天去确认)。

  • 服务器端在函数的参数中写入要接收的参数信息,作为函数的参数,这种方式是可行的。

  demo:

  

 public class Global : HttpApplication
{
void Application_Start(object sender, EventArgs e)
{
RegisterRoutes();
} private void RegisterRoutes()
{
// Edit the base address of Service1 by replacing the "Test" string below
RouteTable.Routes.Add(new ServiceRoute("Test", new WebServiceHostFactory(), typeof(Test)));
}
} [ServiceContract]
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.PerCall)]
// NOTE: If the service is renamed, remember to update the global.asax.cs file
public class Test
{
private Logger logger = LogManager.GetCurrentClassLogger(); [WebInvoke(UriTemplate = "Login", Method = "POST", BodyStyle = WebMessageBodyStyle.WrappedRequest)]
public bool Login(string userName, string userPassword)
{
//模拟登录,实际使用时需数据库访问
if (userName == "huangbo" && userPassword == "")
{
return true;
}
return false;
}
} public class UserInfo
{
public string userName { get; set; }
public string userPassword { get; set; }
} class Program
{
static void Main(string[] args)
{
string baseUrl = "http://localhost:6001/Test/Login"; UserInfo info = new UserInfo
{
userName = "",
userPassword = ""
}; //先访问LoginService,得到授权
var request = (HttpWebRequest)WebRequest.Create(baseUrl);
request.ContentType = "application/json;charset=utf-8";
request.Method = "POST";
request.KeepAlive = true;
//request.ClientCertificates.Add(cert);
//使用JSON.NET将对象序列化成JSON字符串
var requestBody = Newtonsoft.Json.JsonConvert.SerializeObject(info);
byte[] databytes = Encoding.UTF8.GetBytes(requestBody);
request.ContentLength = databytes.Length;
using (var writer = request.GetRequestStream())
{
writer.Write(databytes, , databytes.Length);
}
var response = request.GetResponse();
using (var stream = response.GetResponseStream())
{
using (var reader = new StreamReader(stream))
{
Console.WriteLine(reader.ReadToEnd());
}
}
}
}

参考资料:

http://www.cnblogs.com/xiarifeixue/archive/2011/04/27/restful.html

http://www.cnblogs.com/bearhb/archive/2012/07/11/2586412.html

怎么接收参数:

UserInfo info = new UserInfo
{
userName = “user”,
userPassword = “pwd”
};

string strJson=Newtonsoft.Json.JsonConvert.SerializeObject(message);

byte[] jsonBytes=Encoding.UTF8.GetBytes(strJson);

test client post method: json string UploadData(string url,string method,byte[] bytes);

webClient.UploadData("url","Post",jsonBytes);

mapping WCF Rest Service method:

[WebInvoke(UriTemplate = "", Method = "POST",BodyStyle=WebMessageBodyStyle.WrappedRequest)]

 public bool Login(string userName, string userPassword)
{
  return true;
}

最新文章

  1. docker创建私有仓库
  2. 电脑上不安装Oracle时,C# 调用oracle数据库,Oracle客户工具
  3. java之ubuntu12.04 开发环境配制
  4. Visual Studio 2010中的stdafx.h和targetver.h两个头文件是有什么用?
  5. vs2013安装闪退及vs2010 vs2013打开时提示 未能完成的操作 及vs2013安装时出现图片后闪退
  6. listview可见再加载图片
  7. 【算法题目】包含min函数的栈
  8. go-vim配置
  9. 手把手教你在openshift上搭建wordpress博客(二)
  10. 关于.NET,.NET Framework 和ASP.NET的总结
  11. .Net程序员学用Oracle系列(29):PLSQL 之批量应用和系统包
  12. Java中 equals() 和 == 的区别
  13. vue-router实现登录和跳转到指定页面,vue-router 传参
  14. 从JDK源码角度看java并发的原子性如何保证
  15. python系列-3 pyenv的使用
  16. Redis工具类
  17. Django中的session于cookie的用法
  18. ACM-ICPC 2018 北京赛区网络预赛(9.22)
  19. python 复制/移动文件
  20. CSS(四):字体和文本属性

热门文章

  1. wordpress为dashboard添加样式
  2. LR中的C语言问题
  3. mysql的事务和select...for update
  4. 百度Ueditor编辑器的Html模式自动替换样式的解决方法
  5. xml提取
  6. Java中的匿名对象
  7. Android开发笔记-加载xml资源
  8. postgresql 将查询结果导出到文件
  9. Selenium2学习-029-WebUI自动化实战实例-027-判断元素是否存在
  10. linux命令之tee