Django 密码修改后会话失效的解决方案
技术百科
霞舞
发布时间:2026-01-01
浏览: 次 django 默认在用户密码变更时使旧会话失效,导致用户被登出;需调用 `update_session_auth_hash()` 保持登录状态。
在 Django 中,当用户密码被修改(尤其是通过 user.set_password() 或直接更新 user.password 字段)后,框架会自动使所有基于旧密码哈希生成的会话失效——这是出于安全考虑的默认行为:防止密码泄露后攻击者继续利用已存在的会话令牌。因此,即使用户仍在浏览页面,request.user 也会变为匿名(is_anonymous=True),看似“意外登出”。
你当前的视图中存在两个关键问题:
- 未调用 update_session_auth_hash():这是解决该问题的核心。它会重新生成并更新当前会话的认证哈希,使其与新密码匹配;
- 手动赋值 user.password = make_password(...) 不推荐:应优先使用 user.set_password(),它不仅加密密码,还会自动处理盐值、算法升级等细节,并兼容后续的 session hash 更新。
✅ 正确做法如下(已修复):
from django.contrib.auth import update_session_auth_hash
from django.contrib import messages
def psswdReset(request):
if request.method == 'POST':
new_psswd = request.POST.get('new_psswd')
psswd = request.POST.get('psswd')
# 验证原密码(注意:check_password 接收明文 vs 哈希)
if check_password(psswd, request.user.password):
user = request.user # 直接使用 request.user,无需重复查询
user.set_password(new_psswd) # ✅ 推荐方式:自动处理加密与兼容性
user.save()
# ✅ 关键一步:更新当前会话的认证哈希,避免登出
update_session_auth_hash(request, user)
messages.success(request, 'Password changed successfully!')
return render(request, 'User/userPsswdReset.html', {})
return render(request, 'User/userPsswdReset.html', {})⚠️ 注意事项:
- update_session_auth_hash() 必须在 user.save() 之后、响应返回之前调用;
- 它仅影响当前请求的会话,其他设备/浏览器的会话仍会失效(符合安全设计);
- 若使用 authenticate() + login() 流程(如登录页),无需手动调用此函数——login() 内部已自动处理;
- 切勿跳过原密码校验(即 check_password),否则将削弱账户安全性。
总结:Django 的会话自动失效机制是主动的安全防护,而非 bug;正确使用 update_session_auth_hash() 即可在保障安全的前提下提供无缝的密码更新体验。
# 安全防护
# 浏览器
# word
# go
# html
# session
# django
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Win11怎么设置默认邮件应用_Windows11
- 如何在Golang中使用内置函数_Golangle
- 如何使用Golang实现路由分组管理_Golang
- Windows如何使用注册表查找和删除项?(reg
- PHP怎么接收URL中的锚点参数_获取#后面参数值
- 如何使用Golang配置安全开发环境_防止敏感信息
- TestNG的testng.xml配置文件怎么写
- Windows笔记本无法进入睡眠模式怎么办?(电源
- php本地部署后数据库连接报错_1045acces
- Windows音频驱动无声音原因解析_声卡驱动错误
- Windows 11如何开启文件夹加密(EFS)_
- Python与OpenAI接口集成实战_生成式AI
- Python爬虫项目实战教程_Scrapy抓取与存
- Windows7怎么找回经典开始菜单_Window
- Win11怎么关闭键盘按键音_Win11禁用打字声
- Win11色盲模式怎么开_Win11屏幕颜色滤镜设
- 如何使用Golang recover捕获panic
- Python lxml的etree和Element
- 如何使用Golang实现路由参数绑定_使用Mux和
- 如何使用Golang操作指针变量_Golang解引
- Python深度学习实战教程_神经网络模型构建与训
- Windows10怎么用“讲述人”读屏辅助 Win
- c++如何实现一个高性能的环形队列(Ring Bu
- Win11任务栏天气怎么关闭 Win11隐藏天气小
- Windows服务持续崩溃怎样修复_系统服务保护机
- Windows系统文件被保护机制阻止怎么办_权限不
- php8.4匿名类怎么用_php8.4匿名类创建与
- 如何使用Golang实现负载均衡_分发请求到多个服
- 如何使用Golang指针与结构体结合_修改结构体内
- 如何用正则与预处理结合精准拦截拼接式垃圾域名
- 如何提升Golang程序I/O性能_Golang
- Dapper的Execute方法的返回值是什么意思
- Python多进程教程_multiprocessi
- 企业SEO优化选择网站建设模板的技巧
- Win11怎么退出高对比度模式_Win11取消反色
- Python生成器表达式内存优化_惰性计算说明【指
- 跨文件调用类方法怎么用_php作用域操作符与自动加
- c++的mutex和lock_guard如何使用
- WindowsUSB驱动安装异常怎么办_USB驱动
- 如何在Golang中使用encoding/gob序
- windows 10专注助手怎么关闭_window
- Win10怎么卸载金山毒霸_Win10彻底卸载金山
- Mac版Final Cut Pro入门_Mac视频
- php中$this和::能混用吗_对象与静态作用域
- 如何使用Golang搭建Web开发环境_快速启动H
- Linux怎么禁止Root用户远程登录_Linux
- Mac如何使用听写功能_Mac语音输入打字【效率技
- 如何在 Python 测试中动态配置 @backo
- c++如何判断文件是否存在_c++ filesys
- 如何使用Golang log记录不同级别日志_Go

user.save()
# ✅ 关键一步:更新当前会话的认证哈希,避免登出
update_session_auth_hash(request, user)
messages.success(request, 'Password changed successfully!')
return render(request, 'User/userPsswdReset.html', {})
return render(request, 'User/userPsswdReset.html', {})
QQ客服