从多对多查询sqlalchemy

时间:2018-08-11 17:35:03

标签: python python-3.x sqlalchemy many-to-many relationships

我有这个数据库方案,多对多关系:

Base = declarative_base()

association_table = Table('association', Base.metadata,
    Column('book_id', Integer, ForeignKey('book.book_id')),
    Column('author_id', Integer, ForeignKey('author.author_id')),
)

class Book(Base):
    __tablename__ = 'book'
    book_id = Column(Integer, primary_key=True)
    title = Column(String(50), nullable = False)
    authors = relationship('Author', secondary = association_table, backref=backref('books', lazy= 'dynamic'))

class Author(Base):
    __tablename__ = 'author'
    author_id = Column(Integer, primary_key=True)
    name = Column(String(50), nullable = False)

我要从中查询数据的地方。我知道如何从我创建的对象中查询,例如:

a=Author(name='Author One')
b=Book(title='Book Title One')
session.add(b)
session.add(a)
session.commit() 
a.books.append(b)
session.commit()
# repeat above steps with new variables a2 and b2
# a.books.append(b2) 
for i in a.books:
   print(i.title)

但是,如何直接从表格中查询哪些作者与特定书籍有关?即,下次我要使用数据库,但不再有对象时。我尝试过的事情:

for u in session.query(Book.title).\
        filter(Book.book_id==Author.author_id).\
        filter(Author.name=='Author One').\
        all():
    print(u) # doesn't seem to work, returns only one element.

x = session.query(Author).filter(Book.title.any(name='Book Title One')).all()
# gives AttributeError: Neither 'AnnotatedColumn' object nor 'Comparator' object has an attribute 'any' error.

但是它们似乎都失败了,或者返回了不正确的金额。

1 个答案:

答案 0 :(得分:1)

您可以通过查询数据库在新会话中重新创建对象。例如,您可以获取作者

  author = session.query(Author).filter_by(name='Author One').one()

然后只是遍历他们的图书收藏:

  print('Author\'s books:')
  for b in author.books:
      print(b.title)

Author's books:
Book One
Book Three

或使用作者对象查询Book模型:

  query = session.query(Book).filter(Book.authors.contains(author))
  print('Books\' authors:')
  for b in query:
      print(b.title, ', '.join(a.name for a in b.authors))

Book's authors:
Book One Author One
Book Three Author Two, Author One

如果您不想获取作者对象,则可以使用如下作者名称来查询Book模型:

  query = session.query(Book).filter(Book.authors.any(name='Author One'))
  print('Books by Author One:')
  for b in query:
      print(b.title, ', '.join(a.name for a in b.authors))

Books by Author One:
Book One Author One
Book Three Author Two, Author One