Flask + SQLAlchemy:类不可迭代

5

当我试图将SQLAlchemy查询渲染到模板中时,我遇到了问题,错误提示如下:

TypeError: 'Users' object is not iterable

这个数据库是sqlite类型的。我一直在跟随youtube上的“发现真正的Python”教程,它帮了我很大忙。唯一改变的是,在教程中他使用“BlogPosts”,而我使用“Users”。

我试图理解他如何在模板中迭代用户。

我的Models.py文件如下:

from app import db

class Users(db.Model):
    __tablename__="users"

    id= db.Column(db.Integer, primary_key=True)
    username=db.Column(db.String, nullable=False)
    password=db.Column(db.String, nullable=False)

    def __init__(self, username, password):
        self.username=username
        self.password=password

    def __repr__(self):
        return '<username{}'.format(self.username)

还有我的“app.py”文件。

from flask import Flask, render_template, redirect, url_for, request, session, flash
from functools import wraps
#import sqlite3
from flask.ext.sqlalchemy import SQLAlchemy


app= Flask(__name__)

app.secret_key="this is the secret key"
#app.database="users.db"

app.config['SQLALCHEMY_DATABASE_URI']='sqlite:////users2.db'

db=SQLAlchemy(app)

from models import *

def login_required(f):
    @wraps(f)
    def wrap(*args, **kwargs):
        if 'logged_in' in session:
            return f(*args, **kwargs)
        else:
            flash('You must be logged in to complete this action')
            return redirect(url_for('login'))
    return wrap


@app.route('/')
def index():
    users= db.session.query(Users).all()
    return render_template('index.html', users=users)


@app.route('/testing')
def testing():
    users= db.session.query(Users).all()
    return render_template('testquery.html', users=users)


@app.route('/welcome')
def welcome():
    return render_template('index.html')


@app.route('/login', methods =['GET', 'POST'])
def login():
    error= None
    #form = LoginForm(request.form)
    if request.method == 'POST':
    #if request.form.validate_on_submit():
        if request.form['username'] !='admin' or request.form['password'] !='admin':
            error= 'invalid credentials'
        else:
            session ['logged_in']=True
            flash('login successful')
            return redirect(url_for('loggedin'))
    #else:
        #render_template('login.html', form=form, error=error)
    return render_template('login.html', error=error)


@app.route('/loggedin')
@login_required
def loggedin():
    return render_template('loggedin.html')


@app.route('/logout')
def logout():
    session.pop('logged_in', None)
    flash('You have been logged out')
    return redirect(url_for('welcome'))


#def connect_db():
#   return sqlite3.connect(app.database)

if __name__=='__main__':
    app.run(debug=True)

我一直在使用的测试方法是通过路由"/testing",而testquery.html文件则是我一直在尝试进行测试但遇到错误的文件。
<!DOCTYPE html>
<html>
<head>
    <title></title>
</head>
    <body>


        <table>
        {%for user in users%}
            <tr>
                <td>{{user.username}}</td>
                <td>{{user.password}}</td>
            </tr>

        </table>
        {%endfor%}

    </body>
</html>

我尝试了在另一个堆栈问题中看到的修复方法:

def __init__(self, username, password):
    self.username=[username]
    self.password=[password]

毫无效果,这没有改变任何事情。 有人知道我需要做什么才能使它可迭代吗?

请包含错误的完整回溯。 - Martijn Pieters
1个回答

5
在测试中,你没有用户列表。这里只有一个用户: testing
users = db.session.query(Users).first()

这个代码只返回数据库中的第一个用户。因此,您无法遍历用户。将其重命名为user(删除s)并在模板中删除循环:

@app.route('/testing')
def testing():
    user = db.session.query(Users).first()
    return render_template('testquery.html', user=user)

并且

<table>
  <tr>
    <td>{{user.username}}</td>
    <td>{{user.password}}</td>
  </tr>
</table>

修改了一些内容进行测试,但最初我使用的是“all()”。 - TheTimes
@TheTimes:你在/路由中做了什么。 - Martijn Pieters
好的,我基本上只是按照你建议的去做,然后它就可以工作了。然后我又改回了我原来的代码,神奇地它也能工作了 :/ 我讨厌这种事情发生,虽然我确定一些东西已经改变了,但它看起来像编程魔法。 - TheTimes

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接