ドキュメント およびsqlalchemy.Column
クラスのコメントによると、クラスsqlalchemy.schema.Index
を使用して、複数の列を含むインデックスを指定する必要があります。
ただし、この例では、次のようにTableオブジェクトを直接使用して実行する方法を示しています。
meta = MetaData()
mytable = Table('mytable', meta,
# an indexed column, with index "ix_mytable_col1"
Column('col1', Integer, index=True),
# a uniquely indexed column with index "ix_mytable_col2"
Column('col2', Integer, index=True, unique=True),
Column('col3', Integer),
Column('col4', Integer),
Column('col5', Integer),
Column('col6', Integer),
)
# place an index on col3, col4
Index('idx_col34', mytable.c.col3, mytable.c.col4)
宣言型ORM拡張を使用する場合、どうすればよいですか?
class A(Base):
__table= 'table_A'
id = Column(Integer, , primary_key=True)
a = Column(String(32))
b = Column(String(32))
列「a」と「b」にインデックスが必要です。
これらはColumn
オブジェクトにすぎず、index = Trueフラグは正常に機能します。
class A(Base):
__table= 'table_A'
id = Column(Integer, primary_key=True)
a = Column(String(32), index=True)
b = Column(String(32), index=True)
複合インデックスが必要な場合は、ここでも通常どおりTable
が存在するため、宣言する必要はありません。すべてが同じように機能します(宣言Aaの最近の0.6または0.7になっていることを確認してください)クラス宣言の完了後にColumn
として解釈されるラッパー):
class A(Base):
__table= 'table_A'
id = Column(Integer, primary_key=True)
a = Column(String(32))
b = Column(String(32))
Index('my_index', A.a, A.b)
0.7では、Index
はTable
引数にも含めることができます。これは、宣言で__table_args__
を介して行われます。
class A(Base):
__table= 'table_A'
id = Column(Integer, primary_key=True)
a = Column(String(32))
b = Column(String(32))
__table_args__ = (Index('my_index', "a", "b"), )
@zzzeekの answer を完了するには。
DESCで複合インデックスを追加し、ORM宣言メソッドを使用する場合は、次のようにします。
さらに、SQSAlchemyの Functional Indexes ドキュメンテーションに苦労し、_mytable.c.somecol
_を置き換える方法を見つけようとしました。
_from sqlalchemy import Index Index('someindex', mytable.c.somecol.desc())
_
モデルプロパティを使用して、.desc()
を呼び出すだけです。
_from flask_sqlalchemy import SQLAlchemy
db = SQLAlchemy()
class GpsReport(db.Model):
__table= 'gps_report'
id = db.Column(db.Integer, db.Sequence('gps_report_id_seq'), nullable=False, autoincrement=True, server_default=db.text("nextval('gps_report_id_seq'::regclass)"))
timestamp = db.Column(db.DateTime, nullable=False, primary_key=True)
device_id = db.Column(db.Integer, db.ForeignKey('device.id'), primary_key=True, autoincrement=False)
device = db.relationship("Device", back_populates="gps_reports")
# Indexes
__table_args__ = (
db.Index('gps_report_timestamp_device_id_idx', timestamp.desc(), device_id),
)
_
Alembicを使用している場合、私はFlask-Migrateを使用しています。次のようなものが生成されます。
_from alembic import op
import sqlalchemy as sa
# Added manually this import
from sqlalchemy.schema import Sequence, CreateSequence
def upgrade():
# ### commands auto generated by Alembic - please adjust! ###
# Manually added the Sequence creation
op.execute(CreateSequence(Sequence('gps_report_id_seq')))
op.create_table('gps_report',
sa.Column('id', sa.Integer(), server_default=sa.text("nextval('gps_report_id_seq'::regclass)"), nullable=False),
sa.Column('timestamp', sa.DateTime(), nullable=False))
sa.Column('device_id', sa.Integer(), autoincrement=False, nullable=False),
op.create_index('gps_report_timestamp_device_id_idx', 'gps_report', [sa.text('timestamp DESC'), 'device_id'], unique=False)
def downgrade():
# ### commands auto generated by Alembic - please adjust! ###
op.drop_index('gps_report_timestamp_device_id_idx', table_name='gps_report')
op.drop_table('gps_report')
# Manually added the Sequence removal
op.execute(sa.schema.DropSequence(sa.Sequence('gps_report_id_seq')))
# ### end Alembic commands ###
_
最後に、PostgreSQLデータベースに次のテーブルとインデックスが必要です。
_psql> \d gps_report;
Table "public.gps_report"
Column | Type | Collation | Nullable | Default
-----------------+-----------------------------+-----------+----------+----------------------------------------
id | integer | | not null | nextval('gps_report_id_seq'::regclass)
timestamp | timestamp without time zone | | not null |
device_id | integer | | not null |
Indexes:
"gps_report_pkey" PRIMARY KEY, btree ("timestamp", device_id)
"gps_report_timestamp_device_id_idx" btree ("timestamp" DESC, device_id)
Foreign-key constraints:
"gps_report_device_id_fkey" FOREIGN KEY (device_id) REFERENCES device(id)
_