[python] Flask-SQLalchemy는 행의 정보를 업데이트합니다.

행의 정보를 어떻게 업데이트 할 수 있습니까?

예를 들어 ID가 5 인 행의 이름 열을 변경하고 싶습니다.



답변

Flask-SQLAlchemy 설명서에 표시된 자습서를 사용하여 객체를 검색합니다 . 변경하려는 엔티티가 있으면 엔티티 자체를 변경하십시오. 그런 다음 db.session.commit().

예를 들면 :

admin = User.query.filter_by(username='admin').first()
admin.email = 'my_new_email@example.com'
db.session.commit()

user = User.query.get(5)
user.name = 'New Name'
db.session.commit()

Flask-SQLAlchemy는 SQLAlchemy를 기반으로하므로 SQLAlchemy 문서 도 확인하세요 .


답변

update에서 반환하는 SQLAlchemy의 BaseQuery 개체에 대한 메서드 가 filter_by있습니다.

admin = User.query.filter_by(username='admin').update(dict(email='my_new_email@example.com')))
db.session.commit()

update개체를 변경하는 것보다 사용 하면 업데이트 할 개체가 많을 때 이점이 있습니다.

당신이 부여하려면 add_user모든 권한을 admin의,

rows_changed = User.query.filter_by(role='admin').update(dict(permission='add_user'))
db.session.commit()

공지 filter_by키워드 인수를가 (하나만 사용 =에 반대) filter식을 취한다.


답변

모델의 피클 속성을 수정하면 작동하지 않습니다. 업데이트를 트리거하려면 절인 속성을 교체해야합니다.

from flask import Flask
from flask.ext.sqlalchemy import SQLAlchemy
from pprint import pprint

app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqllite:////tmp/users.db'
db = SQLAlchemy(app)


class User(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(80), unique=True)
    data = db.Column(db.PickleType())

    def __init__(self, name, data):
        self.name = name
        self.data = data

    def __repr__(self):
        return '<User %r>' % self.username

db.create_all()

# Create a user.
bob = User('Bob', {})
db.session.add(bob)
db.session.commit()

# Retrieve the row by its name.
bob = User.query.filter_by(name='Bob').first()
pprint(bob.data)  # {}

# Modifying data is ignored.
bob.data['foo'] = 123
db.session.commit()
bob = User.query.filter_by(name='Bob').first()
pprint(bob.data)  # {}

# Replacing data is respected.
bob.data = {'bar': 321}
db.session.commit()
bob = User.query.filter_by(name='Bob').first()
pprint(bob.data)  # {'bar': 321}

# Modifying data is ignored.
bob.data['moo'] = 789
db.session.commit()
bob = User.query.filter_by(name='Bob').first()
pprint(bob.data)  # {'bar': 321}


답변

값을 할당하고 커밋하면 JSON 및 Pickled 속성을 제외한 모든 데이터 유형에 대해 작동합니다. 피클 유형이 위에서 설명되었으므로 JSON을 업데이트하는 약간 다르지만 쉬운 방법을 기록해 보겠습니다.

class User(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(80), unique=True)
    data = db.Column(db.JSON)

def __init__(self, name, data):
    self.name = name
    self.data = data

모델이 위와 같다고 가정 해 봅시다.

user = User("Jon Dove", {"country":"Sri Lanka"})
db.session.add(user)
db.session.flush()
db.session.commit()

이렇게하면 { “country”: “Sri Lanka”} 데이터가있는 MySQL 데이터베이스에 사용자가 추가됩니다.

데이터 수정은 무시됩니다. 작동하지 않은 내 코드는 다음과 같습니다.

user = User.query().filter(User.name=='Jon Dove')
data = user.data
data["province"] = "south"
user.data = data
db.session.merge(user)
db.session.flush()
db.session.commit()

JSON을 새 dict에 복사하는 고통스러운 작업 (위와 같이 새 변수에 할당하지 않음)을 거치는 대신에 간단한 방법을 찾았습니다. JSON이 변경된 시스템에 플래그를 지정하는 방법이 있습니다.

다음은 작업 코드입니다.

from sqlalchemy.orm.attributes import flag_modified
user = User.query().filter(User.name=='Jon Dove')
data = user.data
data["province"] = "south"
user.data = data
flag_modified(user, "data")
db.session.merge(user)
db.session.flush()
db.session.commit()

이것은 매력처럼 작동했습니다. 여기 에이 방법과 함께 제안 된 또 다른 방법이 있습니다
.


답변