我是SQLAlchemy ORM的新手,我正在努力完成多个表上的复杂查询 - 这些查询在Doctrine DQL中相对简单。
我有城市数据对象,这些对象属于国家。一些城市还设置了县ID,但并非所有城市都设置了。除了必要的主键和外键之外,每个记录还具有text_string_id,该ID链接到TextStrings表,该表以不同语言存储城市/县/国家的名称。TextStrings MySQL表如下:
CREATE TABLE IF NOT EXISTS `text_strings` (
`id` INT UNSIGNED NOT NULL,
`language` VARCHAR(2) NOT NULL,
`text_string` varchar(255) NOT NULL,
PRIMARY KEY (`id`, `language`)
)
我希望为每个城市构建一个面包屑导航,格式如下:
country_en_name > city_en_name 或
country_en_name > county_en_name > city_en_name,
具体取决于此城市是否设置了县属性。在Doctrine中,这相对简单。
$query = Doctrine_Query::create()
->select('ci.id, CONCAT(cyts.text_string, \'> \', IF(cots.text_string is not null, CONCAT(cots.text_string, \'> \', \'\'), cits.text_string) as city_breadcrumb')
->from('City ci')
->leftJoin('ci.TextString cits')
->leftJoin('ci.Country cy')
->leftJoin('cy.TextString cyts')
->leftJoin('ci.County co')
->leftJoin('co.TextString cots')
->where('cits.language = ?', 'en')
->andWhere('cyts.language = ?', 'en')
->andWhere('(cots.language = ? OR cots.language is null)', 'en');
使用SQLAlchemy ORM,我正在努力实现相同的功能。我相信我已经正确设置了对象 - 例如:
class City(Base):
__tablename__ = "cities"
id = Column(Integer, primary_key=True)
country_id = Column(Integer, ForeignKey('countries.id'))
text_string_id = Column(Integer, ForeignKey('text_strings.id'))
county_id = Column(Integer, ForeignKey('counties.id'))
text_strings = relation(TextString, backref=backref('cards', order_by=id))
country = relation(Country, backref=backref('countries', order_by=id))
county = relation(County, backref=backref('counties', order_by=id))
我的问题在于查询 - 我尝试了各种方法来生成面包屑,但似乎都没有成功。一些观察结果:
也许在查询中使用类似于CONCAT和IF的操作不太符合Python的风格(使用ORM是否可能?),所以我尝试在SQLAlchemy之外执行这些操作,在记录的Python循环中执行。然而,我在这里遇到了访问单个字段的困难 - 例如,模型访问器似乎无法深入n级,例如City.counties.text_strings.language不存在。
我还尝试使用元组 - 最接近实现的方法是将其拆分为两个查询:
# For cities without a county
for city, country in session.query(City, Country).\
filter(Country.id == City.country_id).\
filter(City.county_id == None).all():
if city.text_strings.language == 'en':
# etc
# For cities with a county
for city, county, country in session.query(City, County, Country).\
filter(and_(City.county_id == County.id, City.country_id == Country.id)).all():
if city.text_strings.language == 'en':
# etc
我将它分成两个查询,因为我无法弄清如何使Suit加入可选项,只是在一个查询中。但这种方法当然很糟糕,更糟糕的是第二个查询没有100%工作-它没有连接所有不同的城市文本字符串以进行后续过滤。
所以我被难住了!如果您可以帮助我找到正确的路径来执行这些复杂的查询,那将不胜感激。