Ajax跨域请求保证同一个session的问题

本文详细介绍了CORS(跨源资源共享)的工作原理及实现方法,包括如何通过HTTP头来控制跨域请求,以及预请求机制等内容。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


一、CORS概述

跨源资源共享标准通过新增一系列 HTTP 头,让服务器能声明那些来源可以通过浏览器访问该服务器上的各类资源(包括CSS、图片、JavaScript 脚本以及其它类资源)。另外,对那些会对服务器数据造成破坏性影响的 HTTP 请求方法(特别是 GET 以外的 HTTP 方法,或者搭配某些MIME类型的POST请求),标准强烈要求浏览器必须先以 OPTIONS 请求方式发送一个预请求(preflight request),从而获知服务器端对跨源请求所支持 HTTP 方法。在确认服务器允许该跨源请求的情况下,以实际的 HTTP 请求方法发送那个真正的请求。服务器端也可以通知客户端,是不是需要随同请求一起发送信用信息(包括 Cookies 和 HTTP 认证相关数据)。

 

二、CORS原理

例如:域名A(http://a.example)的某 Web 应用程序中通过<img>标签引入了域名B(http://b.foo)站点的某图片资源(http://b.foo/image.jpg)。这就是一个跨域请求,请求http报头包含Origin: http://a.example,如果返回的http报头包含响应头 Access-Control-Allow-Origin: http://a.example (或者Access-Control-Allow-Origin: http://a.example),表示域名B接受域名B下的请求,那么这个图片就运行被加载。否则表示拒绝接受请求。

 

 

三、CORS跨域请求控制方法

 

1.http请求头

 

Origin: 普通的HTTP请求也会带有,在CORS中专门作为Origin信息供后端比对,表明来源域。

Access-Control-Request-Method: 接下来请求的方法,例如PUT, DELETE等等

Access-Control-Request-Headers: 自定义的头部,所有用setRequestHeader方法设置的头部都将会以逗号隔开的形式包含在这个头中

 

2.http响应头

 

然后浏览器再根据服务器的返回值判断是否发送非简单请求。简单请求前面讲过是直接发送,只是多加一个origin字段表明跨域请求的来源。然后服务器处理完请求之后,会再返回结果中加上如下控制字段

Access-Control-Allow-Origin: 允许跨域访问的域,可以是一个域的列表,也可以是通配符"*"。这里要注意Origin规则只对域名有效,并不会对子目录有效。即http://foo.example/subdir/ 是无效的。但是不同子域名需要分开设置,这里的规则可以参照同源策略

Access-Control-Allow-Credentials: 是否允许请求带有验证信息,

Access-Control-Expose-Headers: 允许脚本访问的返回头,请求成功后,脚本可以在

Access-Control-Max-Age: 缓存此次请求的秒数。在这个时间范围内,所有同类型的请求都将不再发送预检请求而是直接使用此次返回的头作为判断依据,非常有用,大幅优化请求次数

Access-Control-Allow-Methods: 允许使用的请求方法,以逗号隔开

Access-Control-Allow-Headers: 允许自定义的头部,以逗号隔开,大小写不敏感

如果程序猿偷懒将Access-Control-Allow-Origin设置为:Access-Control-Allow-Origin: * 允许任何来自任意域的跨域请求,那么久存在被 DDoS攻击的可能。


我们知道,根据浏览器的保护规则,跨域的时候我们创建的sessionId是不会被浏览器保存下来的,这样,当我们在进行跨域访问的时候,我们的sessionId就不会被保存下来,也就是说,每一次的请求,服务器就会以为是一个新的人,而不是同一个人,为了解决这样的办法,下面这种方法可以解决这种跨域的办法。

public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletResponse res = (HttpServletResponse) servletResponse;
        HttpServletRequest request=(HttpServletRequest)servletRequest;
        res.setContentType("textml;charset=UTF-8");
        res.setHeader("Access-Control-Allow-Origin", request.getHeader("Origin"));
        res.setHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE");
        res.setHeader("Access-Control-Max-Age", "0");
        res.setHeader("Access-Control-Allow-Headers", "Origin, No-Cache, X-Requested-With, If-Modified-Since, Pragma, Last-Modified, Cache-Control, Expires, Content-Type, X-E4M-With,userId,token");
        res.setHeader("Access-Control-Allow-Credentials", "true");
        res.setHeader("XDomainRequestAllowed","1");
        filterChain.doFilter(servletRequest,servletResponse);
    }


在ajax 请求是也要加相应的东西
$.ajax({
url:url,
//加上这句话
xhrFields: {
           withCredentials: true
       },
       crossDomain: true,

success:function(result){
alert("test");
},
error:function(){
}
});


这样我们再跨域测试的时候,就会发现我们的sessionId是一样的了,这样就实现了跨域并且保证在同一个session下


相关博客: Access-Control-Allow-CredentialsHTTP访问控制(CORS)
在使用 `$.ajax` 发起 **请求** 时,要保持 **Session 会话状态**(即前后端之间能识别同一个用户),需要前后端都进行一些特定配置。否则,浏览器出于安全策略限制,默认不会发送或保留 Cookie,导致 Session 无法维持。 --- ### ✅ 前端配置:启用携带 Cookie 你需要在 `$.ajax` 请求中设置: ```javascript $.ajax({ url: 'https://api.example.com/login', type: 'POST', xhrFields: { withCredentials: true // 允许携带 cookie }, crossDomain: true, success: function(data) { console.log('登录成功'); } }); ``` #### 关键参数说明: - `withCredentials: true`:允许请求携带 Cookie; - `crossDomain: true`:显式声明这是一个请求; - 如果你使用了 `dataType: 'json'` 或其他类型,请确保后端返回的响应头也正确设置。 --- ### ✅ 后端配置:允许并设置 Cookie 后端必须设置以下 HTTP 响应头来允许前端访问,并且允许 Cookie 被携带。 #### 示例(Node.js + Express): ```js const express = require('express'); const session = require('express-session'); const cors = require('cors'); const app = express(); app.use(cors({ origin: 'https://your-frontend-domain.com', // 允许的源 credentials: true // 允许携带凭证(Cookie) })); app.use(session({ secret: 'keyboard cat', resave: false, saveUninitialized: true, cookie: { httpOnly: true, secure: false, // 开发环境设为 false;生产建议启用 HTTPS + secure: true sameSite: 'none' // 如果是请求,需设置 sameSite: 'none' } })); app.post('/login', (req, res) => { req.session.user = { id: 1, name: 'Alice' }; res.send('Logged in'); }); app.listen(3000); ``` #### 必须设置的响应头: ```http Access-Control-Allow-Origin: https://your-frontend-domain.com Access-Control-Allow-Credentials: true Set-Cookie: connect.sid=xxx; Path=/; HttpOnly; SameSite=None ``` > ⚠️ 注意: > - `Access-Control-Allow-Origin` **不能是** `*`,必须指定具体名; > - `SameSite=None` 需配合 `Secure` 使用(即启用 HTTPS); > - 如果是开发环境,可以先不启用 `Secure`。 --- ### ✅ 浏览器限制和注意事项 | 条件 | 是否影响 Session | |------|------------------| | 使用隐私浏览模式 | Cookie 不持久,每次刷新可能丢失 Session | | 禁用第三方 Cookie | 某些浏览器(如 Safari)默认阻止 Cookie | | 清除浏览器缓存 | 删除了 Cookie,Session 也会失效 | --- ### ✅ 完整流程图解 ``` 前端发起 $.ajax({ withCredentials: true }) ↓ CORS 请求到达后端后端设置 Set-Cookie 并允许 Access-Control-Allow-Credentials ↓ 浏览器保存 Cookie(如果符合规则) ↓ 下次请求自动携带 Cookie(Session ID) ↓ 后端识别 Session,保持登录状态 ``` --- ### ✅ 总结 | 步骤 | 内容 | |------|------| | 前端 | 设置 `withCredentials: true` 和 `crossDomain: true` | | 后端 | 设置 CORS 允许来源 + `credentials: true` + 正确设置 Cookie 属性 | | 协议 | 前后端协议、名、端口一致或正确处理 | | Cookie | 设置 `SameSite=None; Secure`(生产环境) | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值