溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

jsonp和CORS跨域請(qǐng)求問(wèn)題怎么解決

發(fā)布時(shí)間:2021-12-18 10:20:48 來(lái)源:億速云 閱讀:180 作者:iii 欄目:編程語(yǔ)言

這篇文章主要講解了“jsonp和CORS跨域請(qǐng)求問(wèn)題怎么解決”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“jsonp和CORS跨域請(qǐng)求問(wèn)題怎么解決”吧!

JSONP

jsonp 的原理很簡(jiǎn)單,利用了【前端請(qǐng)求靜態(tài)資源的時(shí)候不存在跨域問(wèn)題】這個(gè)思路。

但是 只支持 get,只支持 get,只支持 get。

注意一點(diǎn),既然這個(gè)方法叫 jsonp,后端數(shù)據(jù)一定要使用 json 數(shù)據(jù),不能隨便的搞個(gè)字符串什么的,不然你會(huì)覺(jué)得結(jié)果莫名其妙的。

前端 jQuery 寫(xiě)法

$.ajax({

  type: "get",

  url: baseUrl + "/jsonp/get",

  dataType: "jsonp",

  success: function(response) {

    $("#response").val(JSON.stringify(response));

  }

});

dataType: "jsonp"。除了這個(gè),其他配置和普通的請(qǐng)求是一樣的。

后端 SpringMVC 配置

如果你也使用 SpringMVC,那么配置一個(gè) jsonp 的 Advice 就可以了,這樣我們寫(xiě)的每一個(gè) Controller 方法就完全不需要考慮客戶(hù)端到底是不是 jsonp 請(qǐng)求了,Spring 會(huì)自動(dòng)做相應(yīng)的處理。

@ControllerAdvice

public class JsonpAdvice extends AbstractJsonpResponseBodyAdvice {

    public JsonpAdvice(){

        // 這樣如果請(qǐng)求中帶 callback 參數(shù),Spring 就知道這個(gè)是 jsonp 的請(qǐng)求了

        super("callback");

    }

}

以上寫(xiě)法要求 SpringMVC 版本不低于 3.2,低于 3.2 的我只能說(shuō),你們?cè)撋?jí)了。

后端非 SpringMVC 配置

以前剛工作的時(shí)候,Struts2 還紅遍天,幾年的光景,SpringMVC 就基本統(tǒng)治下來(lái)了國(guó)內(nèi)市場(chǎng)。

偷懶一下,這里貼個(gè)偽代碼吧,在我們的方法返回前端之前調(diào)一下 wrap 方法:

public Object wrap(HttpServletRequest request){

    String callback = request.getParameter("callback");

    if(StringUtils.isBlank(callback)){

        return result;

    } else {

        return callback+"("+JSON.toJSONString(result)+")";

    }

}

CORS

Cross-Origin Resource Sharing

畢竟 jsonp 只支持 get 請(qǐng)求,肯定不能滿(mǎn)足我們的所有的請(qǐng)求需要,所以才需要搬出 CORS。

國(guó)內(nèi)的 web 開(kāi)發(fā)者還是比較苦逼的,用戶(hù)死不升級(jí)瀏覽器,老板還死要開(kāi)發(fā)者做兼容。

CORS 支持以下瀏覽器,目前來(lái)看,瀏覽器的問(wèn)題已經(jīng)越來(lái)越不重要了,連淘寶都不支持 IE7 了~~~

Chrome 3+

Firefox 3.5+

Opera 12+

Safari 4+

Internet Explorer 8+

前端 jQuery 寫(xiě)法

直接看代碼吧:

$.ajax({

    type: "POST",

    url: baseUrl + "/jsonp/post",

    dataType: 'json',

    crossDomain: true,

    xhrFields: {

        withCredentials: true

    },

    data: {

        name: "name_from_frontend"

    },

    success: function (response) {

        console.log(response)// 返回的 json 數(shù)據(jù)

        $("#response").val(JSON.stringify(response));

    }

});

dataType: "json",這里是 json,不是 jsonp,不是 jsonp,不是 jsonp。

crossDomain: true,這里代表使用跨域請(qǐng)求

xhrFields: {withCredentials: true},這樣配置就可以把 cookie 帶過(guò)去了,不然我們連 session 都沒(méi)法維護(hù),很多人都栽在這里。當(dāng)然,如果你沒(méi)有這個(gè)需求,也就不需要配置這個(gè)了。

后端 SpringMVC 配置

對(duì)于大部分的 web 項(xiàng)目,一般都會(huì)有 mvc 相關(guān)的配置類(lèi),此類(lèi)繼承自 WebMvcConfigurerAdapter。如果你也使用 SpringMVC 4.2 以上的版本的話,直接像下面這樣添加這個(gè)方法就可以了:

@Configuration

public class WebConfig extends WebMvcConfigurerAdapter {

    @Override

    public void addCorsMappings(CorsRegistry registry) {

        registry.addMapping("/**/*").allowedOrigins("*");

    }

}

如果很不幸你的項(xiàng)目中 SpringMVC 版本低于 4.2,那么需要「曲線救國(guó)」一下:

public class CrossDomainFilter extends OncePerRequestFilter {

    @Override

    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {

        response.addHeader("Access-Control-Allow-Origin", "*");// 如果提示 * 不行,請(qǐng)往下看

        response.addHeader("Access-Control-Allow-Credentials", "true");

        response.addHeader("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE");

        response.addHeader("Access-Control-Allow-Headers", "Content-Type");

        filterChain.doFilter(request, response);

    }

}

在 web.xml 中配置下 filter:

<filter>

    <filter-name>CrossDomainFilter</filter-name>

    <filter-class>com.javadoop.filters.CrossDomainFilter</filter-class>

</filter>

<filter-mapping>

    <filter-name>CrossDomainFilter</filter-name>

    <url-pattern>/*</url-pattern>

</filter-mapping>

有很多項(xiàng)目用 shiro 的,也可以通過(guò)配置 shiro 過(guò)濾器的方式,這里就不介紹了。

注意了,我說(shuō)的是很籠統(tǒng)的配置,對(duì)于大部分項(xiàng)目是可以這么籠統(tǒng)地配置的。文中類(lèi)似 “*” 這種配置讀者應(yīng)該都能知道怎么配。

如果讀者發(fā)現(xiàn)瀏覽器提示不能用 ‘*’ 符號(hào),那讀者可以在上面的 filter 中根據(jù) request 對(duì)象拿到請(qǐng)求頭中的 referer(request.getHeader("referer")),然后動(dòng)態(tài)地設(shè)置 "Access-Control-Allow-Origin":

String referer = request.getHeader("referer");

if (StringUtils.isNotBlank(referer)) {

    URL url = new URL(referer);

    String origin = url.getProtocol() + "://" + url.getHost();

    response.addHeader("Access-Control-Allow-Origin", origin);

} else {

    response.addHeader("Access-Control-Allow-Origin", "*");

}

2018-04-28:今天終于知道為什么有時(shí)候會(huì)提示我們 * 不支持了,原來(lái)是只要前端寫(xiě)了 withCredentials: true 那么瀏覽器就會(huì)提示這個(gè),一種辦法就是這里說(shuō)的使用動(dòng)態(tài)構(gòu)造 origin 的方式,另一種辦法就是跨域不傳 cookie,讓前端把 cookie 要傳的信息(如 sessionId/accessKey) 放到 header 中或者直接寫(xiě)在 request 的參數(shù)里。

前端非 jQuery 寫(xiě)法

jQuery 一招鮮吃遍天的日子是徹底不在了,這里就說(shuō)說(shuō)如果不使用 jQuery 的話,怎么解決 post 跨域的問(wèn)題。大部分的 js 庫(kù)都會(huì)提供相應(yīng)的方案的,大家直接找相應(yīng)的文檔看看就知道怎么用了。

來(lái)一段原生 js 介紹下:

function createCORSRequest(method, url) {

    var xhr = new XMLHttpRequest();

    if ("withCredentials" in xhr) {

        // 如果有 withCredentials 這個(gè)屬性,那么可以肯定是 XMLHTTPRequest2 對(duì)象??吹谌齻€(gè)參數(shù)

        xhr.open(method, url, true);

    } else if (typeof XDomainRequest != "undefined") {

        // 此對(duì)象是 IE 用來(lái)跨域請(qǐng)求的

        xhr = new XDomainRequest();

        xhr.open(method, url);

    } else {

        // 如果是這樣,很不幸,瀏覽器不支持 CORS

        xhr = null;

    }

    return xhr;

}

var xhr = createCORSRequest('GET', url);

if (!xhr) {

    throw new Error('CORS not supported');

}

其中,Chrome,F(xiàn)irefox,Opera,Safari 這些「程序員友好」的瀏覽器使用的是 XMLHTTPRequest2 對(duì)象。IE 使用的是 XDomainRequest。

感謝各位的閱讀,以上就是“jsonp和CORS跨域請(qǐng)求問(wèn)題怎么解決”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)jsonp和CORS跨域請(qǐng)求問(wèn)題怎么解決這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI