0

0

在Flask应用中安全高效地更新SQLAlchemy用户数据

聖光之護

聖光之護

发布时间:2025-11-29 13:41:11

|

560人浏览过

|

来源于php中文网

原创

在flask应用中安全高效地更新sqlalchemy用户数据

本文详细介绍了在Flask应用中使用SQLAlchemy更新用户数据库中特定字段(如用户积分)的方法。我们将探讨如何安全地查询用户、递增其数值,并利用事务锁机制(`with_for_update`)避免并发问题,确保数据一致性,最终实现用户点击按钮后积分的可靠更新。

在构建基于Flask和SQLAlchemy的Web应用时,动态更新数据库中的用户数据是一项核心功能。例如,当用户在网站上执行特定操作(如点击按钮)时,需要实时更新其积分或状态。本教程将详细指导您如何在Flask环境中,安全且高效地实现这一目标。

1. 初始设置与SQLAlchemy模型

首先,我们需要一个基础的Flask应用框架和SQLAlchemy模型。以下是示例代码中的关键部分:

from flask import Flask, redirect, url_for, render_template, request, session, flash
from datetime import timedelta
from flask_sqlalchemy import SQLAlchemy

app = Flask(__name__)
app.secret_key = "your_secret_key" # 实际应用中请使用更复杂的密钥
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///users.sqlite3'
app.config["SQLALCHEMY_TRACK_MODIFICATIONS"] = False
app.permanent_session_lifetime = timedelta(minutes=5)

db = SQLAlchemy(app)

class Users(db.Model): # 建议类名使用大写开头
    __tablename__ = 'users' # 明确指定表名
    _id = db.Column("id", db.Integer, primary_key=True)
    name = db.Column(db.String(100))
    email = db.Column(db.String(100))
    score = db.Column(db.Integer, default=0) # 设定默认值

    def __init__(self, name, email, score=0):
        self.name = name
        self.email = email
        self.score = score

    def __repr__(self):
        return f""

# 在应用上下文外部创建表(通常在首次运行时执行一次)
# with app.app_context():
#     db.create_all()

在此模型中,Users类代表数据库中的用户表,包含 _id (主键)、name、email 和 score 字段。score字段用于存储用户的积分,并设置了默认值为0。

2. 实现用户积分更新逻辑

我们的目标是当用户点击一个按钮时,其积分能够增加。这通常通过一个POST请求路由来处理。以下是实现此功能的关键代码:

@app.route('/buttonclick', methods=['POST'])
def buttonclick():
    # 1. 获取用户ID
    # 实际应用中,用户ID通常从会话(session)或JWT token中获取,
    # 而非直接从请求体中获取,以防止客户端伪造。
    # 这里为了演示,假设user_id通过JSON传递。
    user_id = request.json.get("user_id")

    if not user_id:
        return 'Error: user_id is required', 400

    try:
        # 2. 根据ID查找用户并加锁
        # with_for_update() 用于对查询结果行施加悲观锁,
        # 防止在当前事务完成之前,其他并发事务修改相同的数据。
        # 这对于递增操作尤其重要,可以避免竞态条件导致的数据不一致。
        user = Users.query.filter_by(_id=user_id).with_for_update().first()

        if user:
            # 3. 更新用户积分
            user.score += 1
            # 4. 提交事务
            db.session.commit()
            print(f"User {user.name} score updated to {user.score}")
            return 'Success', 200
        else:
            return 'Error: User not found', 404
    except Exception as e:
        db.session.rollback() # 发生异常时回滚事务
        print(f"Error updating score: {e}")
        return 'Error: Internal server error', 500

# 其他路由...
@app.route("/clicker")
def clicker():
    # 假设这里渲染的clicker.html包含一个按钮,点击后发送POST请求到/buttonclick
    return render_template("clicker.html")

if __name__ == "__main__":
    with app.app_context():
        db.create_all() # 确保数据库表已创建
    app.run(debug=True)

3. 关键概念解析

3.1 with_for_update() 的作用

在多用户并发访问的场景下,如果多个用户同时点击按钮,尝试更新同一个用户的积分,就可能出现“竞态条件”(Race Condition)。例如:

瑞宝通JAVA版B2B电子商务系统
瑞宝通JAVA版B2B电子商务系统

瑞宝通B2B系统使用当前流行的JAVA语言开发,以MySQL为数据库,采用B/S J2EE架构。融入了模型化、模板、缓存、AJAX、SEO等前沿技术。与同类产品相比,系统功能更加强大、使用更加简单、运行更加稳 定、安全性更强,效率更高,用户体验更好。系统开源发布,便于二次开发、功能整合、个性修改。 由于使用了JAVA开发语言,无论是在Linux/Unix,还是在Windows服务器上,均能良好运行

下载
  1. 用户A读取积分 score = 10。
  2. 用户B读取积分 score = 10。
  3. 用户A将积分加1,更新为 score = 11。
  4. 用户B将积分加1,更新为 score = 11。 最终积分只增加了1,而不是预期的2。

with_for_update() 方法是SQLAlchemy提供的一种悲观锁机制。当它与查询一起使用时,它会在数据库层面锁定被查询的行,直到当前事务提交或回滚。这意味着,在当前事务持有锁期间,其他试图修改或锁定相同行的事务将被阻塞,直到锁释放。这有效地避免了上述竞态条件,确保了数据的一致性。

注意:with_for_update() 仅在支持行级锁的数据库(如PostgreSQL、MySQL的InnoDB引擎)上有效。SQLite在默认情况下对整个数据库文件加锁,因此在SQLite中,其行为可能有所不同,但概念上仍是为了防止并发问题。

3.2 事务管理 (db.session.commit() 和 db.session.rollback())

SQLAlchemy通过db.session管理数据库会话和事务。

  • db.session.add(obj):将新对象添加到会话中,等待被持久化。
  • db.session.delete(obj):将对象标记为删除。
  • db.session.commit():提交当前会话中的所有更改(新增、修改、删除)到数据库,使其永久生效。
  • db.session.rollback():撤销当前会话中的所有未提交更改,将数据库恢复到事务开始时的状态。这在处理异常时非常重要,可以避免部分数据更新导致的数据不一致。

4. 注意事项与最佳实践

  • 用户认证与授权:在实际生产环境中,user_id不应该直接从客户端请求中获取。它应该从已认证的用户会话(session)或JWT令牌中提取,以确保只有当前登录的用户才能修改自己的数据,并防止恶意用户伪造user_id。
    # 示例:从会话中获取用户ID
    # if 'user_id' in session:
    #     user_id = session['user_id']
    # else:
    #     return 'Unauthorized', 401
  • 错误处理:完善的错误处理机制至关重要。当数据库操作失败时(例如,用户不存在、数据库连接问题),应捕获异常并回滚事务,同时向客户端返回适当的错误信息。
  • 前端交互:前端页面(clicker.html)需要通过JavaScript发送一个POST请求到/buttonclick路由,并在请求体中包含user_id(如果采用这种方式)。
    // 示例前端JavaScript (使用fetch API)
    document.getElementById('myButton').addEventListener('click', function() {
        fetch('/buttonclick', {
            method: 'POST',
            headers: {
                'Content-Type': 'application/json'
            },
            body: JSON.stringify({ user_id: 1 }) // 替换为实际的用户ID
        })
        .then(response => response.text())
        .then(data => console.log(data))
        .catch(error => console.error('Error:', error));
    });
  • 数据库迁移:随着应用的发展,数据库模型可能会发生变化。使用Alembic等数据库迁移工具可以帮助您管理数据库模式的演进。
  • 性能考量:with_for_update()会锁定数据库行,在高并发场景下可能导致性能瓶颈。对于不要求严格实时一致性、且更新操作非常频繁的场景,可以考虑使用乐观锁(通过版本号字段)或队列(异步处理更新)等其他方案。但对于简单的积分递增,悲观锁通常是安全且易于实现的方案。

总结

本教程详细阐述了在Flask应用中,利用SQLAlchemy更新用户积分数据的完整流程。我们不仅学习了如何通过ORM模型进行数据查询和修改,更重要的是,深入理解了with_for_update()悲观锁机制在并发更新场景下的重要性,以及事务管理(commit和rollback)在确保数据一致性方面的作用。遵循这些最佳实践,可以帮助您构建健壮、可靠的Web应用程序。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

556

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

733

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

477

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

414

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

991

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

553

2023.09.20

xml格式相关教程
xml格式相关教程

本专题整合了xml格式相关教程汇总,阅读专题下面的文章了解更多详细内容。

0

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.8万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 801人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号