加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

ajax跨域调用webservice

发布时间:2020-12-16 01:39:33 所属栏目:百科 来源:网络整理
导读:最近ajax访问webservice遇到跨域的问题,网上搜索资料,总结如下(很多都是觉得人家总结不错的复制下来) 用JSON来传数据,靠JSONP来跨域 先上我的已实现代码: 前端代码: $.ajax({ type:"get", url:"http://localhost/Service1.asmx/ getElevatorStatusJsonD

最近ajax访问webservice遇到跨域的问题,网上搜索资料,总结如下(很多都是觉得人家总结不错的复制下来)

<<用JSON来传数据,靠JSONP来跨域>>

先上我的已实现代码:

前端代码:

$.ajax({

type:"get",

url:"http://localhost/Service1.asmx/getElevatorStatusJsonData?jsoncallback=?",

dataType:"jsonp",

jsonp:"json",

data:"",

success:function(result){

vardata=eval(result);

for(vari=0;i<data.length;i++){

alert(data[i].ID+"--"+data[i].Name);

}

},

error:function(a,b,c){

alert(c);

}

});


服务端代码:

c#


///<summary>

///获取状态数据信息

///</summary>

///<returns></returns>

[WebMethod]

publicvoidgetElevatorStatusJsonData()

{

List<List<DeviceInfo>>elevatordatas=newList<List<DeviceInfo>>();

List<SendDicdate>searchList=XmlSerializeHelper.XmlDeserializeFromFile<List<SendDicdate>>(@ConfigUtil.servicePath+ConfigUtil.getConfigByKey("xmlPath")+"查询指令信息.xml",Encoding.UTF8);

foreach(SendDicdateiteminsearchList)

{

stringkey=item.portno+"-"+item.bordrate+"-"+item.sendtype;

List<DeviceInfo>deviceInfoList=(List<DeviceInfo>)Context.Cache.Get(key);

elevatordatas.Add(deviceInfoList);

}


Stringresult="";

DataContractJsonSerializerjson=newDataContractJsonSerializer(elevatordatas.GetType());

using(MemoryStreamstream=newMemoryStream())

{

json.WriteObject(stream,elevatordatas);

result=Encoding.UTF8.GetString(stream.ToArray());

}

Stringjsoncallback=HttpContext.Current.Request["jsoncallback"];

result=jsoncallback+'('+result+')';

HttpContext.Current.Response.Write(result);

HttpContext.Current.Response.End();


}



以上是调用c#服务端的实现代码,下面的是java端的,参数可能有差异,但原理是相通的




java:

String callbackFunName = context.Request["callbackparam];

context.Response.Write(callbackFunName+"([{"name":"John"}])");


PS:客户端的jsonp参数是用来通过url传参,传递jsonpCallback参数的参数名,比较拗口,通俗点讲:

jsonp:""
jsonpCallback:""


顺带一提:
在chrome浏览器里,还可以在服务端设置header信息
context.Response.AddHeader("Access-Control-Allow-Origin","*");
来达到跨域请求的目的,并且不需要设置ajax以下参数
dataType:"jsonp",jsonp:"callbackparam",jsonpCallback:"jsonpCallback1"

以正常ajax请求方式就可以获得数据。



下面是原理,看别人讲解的,感觉很有道理:

  1、一个众所周知的问题,Ajax直接请求普通文件存在跨域无权限访问的问题,甭管你是静态页面、动态网页、web服务、WCF,只要是跨域请求,一律不准;

  2、不过我们又发现,Web页面上调用js文件时则不受是否跨域的影响(不仅如此,我们还发现凡是拥有”src”这个属性的标签都拥有跨域的能力,比如<script>、<img>、<iframe>);

  3、于是可以判断,当前阶段如果想通过纯web端(ActiveX控件、服务端代理、属于未来的HTML5之Websocket等方式不算)跨域访问数据就只有一种可能,那就是在远程服务器上设法把数据装进js格式的文件里,供客户端调用和进一步处理;

  4、恰巧我们已经知道有一种叫做JSON的纯字符数据格式可以简洁的描述复杂数据,更妙的是JSON还被js原生支持,所以在客户端几乎可以随心所欲的处理这种格式的数据;

  5、这样子解决方案就呼之欲出了,web客户端通过与调用脚本一模一样的方式,来调用跨域服务器上动态生成的js格式文件(一般以JSON为后缀),显而易见,服务器之所以要动态生成JSON文件,目的就在于把客户端需要的数据装入进去。

  6、客户端在对JSON文件调用成功之后,也就获得了自己所需的数据,剩下的就是按照自己需求进行处理和展现了,这种获取远程数据的方式看起来非常像AJAX,但其实并不一样。

  7、为了便于客户端使用数据,逐渐形成了一种非正式传输协议,人们把它称作JSONP,该协议的一个要点就是允许用户传递一个callback 参数给服务端,然后服务端返回数据时会将这个callback参数作为函数名来包裹住JSON数据,这样客户端就可以随意定制自己的函数来自动处理返回数 据了。



聪明的开发者很容易想到,只要服务端提供的js脚本是动态生成的就行了呗,这样调用者可以传一个参数过去告诉服务端“我想要一段调用XXX函数的js代码,请你返回给我”,于是服务器就可以按照客户端的需求来生成js脚本并响应了。
<!DOCTYPEhtmlPUBLIC"-//W3C//DTDXHTML1.0Transitional//EN""http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><htmlxmlns="http://www.w3.org/1999/xhtml"><head><title></title><scripttype="text/javascript">//得到航班信息查询结果后的回调函数varflightHandler=function(data){
alert('你查询的航班结果是:piao价'+data.price+'元,'+'余piao'+data.tickets+'张。');
};//提供jsonp服务的url地址(不管是什么类型的地址,最终生成的返回值都是一段javascript代码)varurl="http://flightQuery.com/jsonp/flightResult.aspx?code=CA1998&callback=flightHandler";//创建script标签,设置其属性varscript=document.createElement('script');
script.setAttribute('src',url);//把script标签加入head,此时调用开始document.getElementsByTagName('head')[0].appendChild(script);</script></head><body></body></html>

是不是有点奇怪?为什么我这次没有写flightHandler这个函数呢?而且竟然也运行成功了!哈哈,这就是jQuery的功劳了,jquery在处
理jsonp类型的ajax时(还是忍不住吐槽,虽然jquery也把jsonp归入了ajax,但其实它们真的不是一回事儿),自动帮你生成回调函数并
把数据取出来供success属性方法来调用,是不是很爽呀?
<!DOCTYPEhtmlPUBLIC"-//W3C//DTDXHTML1.0Transitional//EN""http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><htmlxmlns="http://www.w3.org/1999/xhtml"><head><title>UntitledPage</title><scripttype="text/javascript"src=jquery.min.js"></script><scripttype="text/javascript">
jQuery(document).ready(function(){
$.ajax({
type:"get",async:false,url:"http://flightQuery.com/jsonp/flightResult.aspx?code=CA1998",dataType:"jsonp",jsonp:"callback",//传递给请求处理程序或页面的,用以获得jsonp回调函数名的参数名(一般默认为:callback)jsonpCallback:"flightHandler",//自定义的jsonp回调函数名称,默认为jQuery自动生成的随机函数名,也可以写"?",jQuery会自动为你处理数据success:function(json){
alert('您查询到航班信息:piao价:'+json.price+'元,余piao:'+json.tickets+'张。');
},error:function(){
alert('fail');
}
});
});</script></head><body></body></html>

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读