私は ドキュメントを通して を見ましたが、SQLAlchemyでORクエリを実行する方法を見つけることができないようです。このクエリを実行したいだけです。
SELECT address FROM addressbook WHERE city='boston' AND (lastname='bulger' OR firstname='whitey')
のようなものでなければなりません
addr = session.query(AddressBook).filter(City == "boston").filter(????)
tutorial から:
from sqlalchemy import or_
filter(or_(User.name == 'ed', User.name == 'wendy'))
SQLAlchemyは、ビット単位の演算子&
、|
、および~
をオーバーロードするため、or_()
およびand_()
を使用したい読みにくいプレフィックス構文の代わりに( Bastienの答え のように)これらの演算子を使用できます:
.filter((AddressBook.lastname == 'bulger') | (AddressBook.firstname == 'whitey'))
括弧はビット単位演算子の優先順位のため、オプションではありません。
したがって、クエリ全体は次のようになります。
addr = session.query(AddressBook) \
.filter(AddressBook.city == "boston") \
.filter((AddressBook.lastname == 'bulger') | (AddressBook.firstname == 'whitey'))
or_演算子は、ORクエリコンポーネントの数が不明な場合に役立ちます。
たとえば、オプションのフィルターがほとんどないRESTサービスを作成し、いずれかのフィルターがtrueを返す場合にレコードを返すと仮定します。一方、リクエストでパラメータが定義されていなかった場合、クエリは変更されません。 or_関数がなければ、次のようなことをしなければなりません。
query = Book.query
if filter.title and filter.author:
query = query.filter((Book.title.ilike(filter.title))|(Book.author.ilike(filter.author)))
else if filter.title:
query = query.filter(Book.title.ilike(filter.title))
else if filter.author:
query = query.filter(Book.author.ilike(filter.author))
Or_関数を使用して、次のように書き換えることができます。
query = Book.query
not_null_filters = []
if filter.title:
not_null_filters.append(Book.title.ilike(filter.title))
if filter.author:
not_null_filters.append(Book.author.ilike(filter.author))
if len(not_null_filters) > 0:
query = query.filter(or_(*not_null_filters))
これは本当に役に立ちました。これは、特定のテーブルの実装です。
def sql_replace(self, tableobject, dictargs):
#missing check of table object is valid
primarykeys = [key.name for key in inspect(tableobject).primary_key]
filterargs = []
for primkeys in primarykeys:
if dictargs[primkeys] is not None:
filterargs.append(getattr(db.RT_eqmtvsdata, primkeys) == dictargs[primkeys])
else:
return
query = select([db.RT_eqmtvsdata]).where(and_(*filterargs))
if self.r_ExecuteAndErrorChk2(query)[primarykeys[0]] is not None:
# update
filter = and_(*filterargs)
query = tableobject.__table__.update().values(dictargs).where(filter)
return self.w_ExecuteAndErrorChk2(query)
else:
query = tableobject.__table__.insert().values(dictargs)
return self.w_ExecuteAndErrorChk2(query)
# example usage
inrow = {'eqmtvs_id': eqmtvsid, 'datetime': dtime, 'param_id': paramid}
self.sql_replace(tableobject=db.RT_eqmtvsdata, dictargs=inrow)