This page is part of the .

    Previous: Writing SELECT statements for ORM Mapped Classes | Next:

    Writing SELECT statements for Inheritance Mappings

    About this Document

    This section makes use of ORM mappings configured using the feature, described at Mapping Class Inheritance Hierarchies. The emphasis will be on as this is the most intricate ORM querying case.

    View the ORM setup for this page.

    A SELECT statement constructed against a class in a joined inheritance hierarchy will query against the table to which the class is mapped, as well as any super-tables present, using JOIN to link them together. The query would then return objects that are of that requested type as well as any sub-types of the requested type, using the discriminator value in each row to determine the correct type. The query below is established against the subclass of Employee, which then returns a result that will contain only objects of type Manager:

    When the SELECT statement is against the base class in the hierarchy, the default behavior is that only that class’ table will be included in the rendered SQL and JOIN will not be used. As in all cases, the column is used to distinguish between different requested sub-types, which then results in objects of any possible sub-type being returned. The objects returned will have attributes corresponding to the base table populated, and attributes corresponding to sub-tables will start in an un-loaded state, loading automatically when accessed. The loading of sub-attributes is configurable to be more “eager” in a variety of ways, discussed later in this section.

    The example below creates a query against the Employee superclass. This indicates that objects of any type, including Manager, Engineer, and Employee, may be within the result set:

    1. >>> from sqlalchemy import select
    2. >>> stmt = select(Employee).order_by(Employee.id)
    3. >>> objects = session.scalars(stmt).all()
    4. BEGIN (implicit)
    5. SELECT employee.id, employee.name, employee.type, employee.company_id
    6. FROM employee ORDER BY employee.id
    7. [...] ()
    8. >>> print(objects)
    9. [Manager('Mr. Krabs'), Engineer('SpongeBob'), Engineer('Squidward')]

    Above, the additional tables for Manager and Engineer were not included in the SELECT, which means that the returned objects will not yet contain data represented from those tables, in this example the .manager_name attribute of the Manager class as well as the .engineer_info attribute of the Engineer class. These attributes start out in the expired state, and will automatically populate themselves when first accessed using :

    1. >>> mr_krabs = objects[0]
    2. >>> print(mr_krabs.manager_name)
    3. SELECT manager.manager_name AS manager_manager_name
    4. FROM manager
    5. WHERE ? = manager.id
    6. [...] (1,)
    7. Eugene H. Krabs

    This lazy load behavior is not desirable if a large number of objects have been loaded, in the case that the consuming application will need to be accessing subclass-specific attributes, as this would be an example of the N plus one problem that emits additional SQL per row. This additional SQL can impact performance and also be incompatible with approaches such as using . Additionally, in our query for Employee objects, since the query is against the base table only, we did not have a way to add SQL criteria involving subclass-specific attributes in terms of Manager or Engineer. The next two sections detail two constructs that provide solutions to these two issues in different ways, the selectin_polymorphic() loader option and the entity construct.

    Using selectin_polymorphic()

    To address the issue of performance when accessing attributes on subclasses, the loader strategy may be used to eagerly load these additional attributes up front across many objects at once. This loader option works in a similar fashion as the relationship loader strategy to emit an additional SELECT statement against each sub-table for objects loaded in the hierarchy, using IN to query for additional rows based on primary key.

    selectinload() accepts as its arguments the base entity that is being queried, followed by a sequence of subclasses of that entity for which their specific attributes should be loaded for incoming rows:

    1. >>> from sqlalchemy.orm import selectin_polymorphic
    2. >>> loader_opt = selectin_polymorphic(Employee, [Manager, Engineer])

    The construct is then used as a loader option, passing it to the Select.options() method of . The example illustrates the use of selectin_polymorphic() to eagerly load columns local to both the Manager and Engineer subclasses:

    1. >>> from sqlalchemy.orm import selectin_polymorphic
    2. >>> loader_opt = selectin_polymorphic(Employee, [Manager, Engineer])
    3. >>> stmt = select(Employee).order_by(Employee.id).options(loader_opt)
    4. >>> objects = session.scalars(stmt).all()
    5. BEGIN (implicit)
    6. SELECT employee.id, employee.name, employee.type, employee.company_id
    7. FROM employee ORDER BY employee.id
    8. [...] ()
    9. SELECT manager.id AS manager_id, employee.id AS employee_id,
    10. employee.type AS employee_type, manager.manager_name AS manager_manager_name
    11. FROM employee JOIN manager ON employee.id = manager.id
    12. WHERE employee.id IN (?) ORDER BY employee.id
    13. [...] (1,)
    14. SELECT engineer.id AS engineer_id, employee.id AS employee_id,
    15. employee.type AS employee_type, engineer.engineer_info AS engineer_engineer_info
    16. FROM employee JOIN engineer ON employee.id = engineer.id
    17. WHERE employee.id IN (?, ?) ORDER BY employee.id
    18. [...] (2, 3)
    19. >>> print(objects)
    20. [Manager('Mr. Krabs'), Engineer('SpongeBob'), Engineer('Squidward')]

    The above example illustrates two additional SELECT statements being emitted in order to eagerly fetch additional attributes such as Engineer.engineer_info as well as Manager.manager_name. We can now access these sub-attributes on the objects that were loaded without any additional SQL statements being emitted:

    1. >>> print(objects[0].manager_name)
    2. Eugene H. Krabs

    Tip

    The loader option does not yet optimize for the fact that the base employee table does not need to be included in the second two “eager load” queries; hence in the example above we see a JOIN from employee to manager and engineer, even though columns from employee are already loaded. This is in contrast to the selectinload() relationship strategy which is more sophisticated in this regard and can factor out the JOIN when not needed.

    The SELECT statements emitted by selectin_polymorphic() are themselves ORM statements, so we may also add other loader options (such as those documented at ) that refer to specific subclasses. For example, if we considered that the Manager mapper had a one to many relationship to an entity called Paperwork, we could combine the use of and selectinload() to eagerly load this collection on all Manager objects, where the sub-attributes of Manager objects were also themselves eagerly loaded:

    1. >>> from sqlalchemy.orm import selectinload
    2. >>> from sqlalchemy.orm import selectin_polymorphic
    3. >>> stmt = (
    4. ... select(Employee)
    5. ... .order_by(Employee.id)
    6. ... .options(
    7. ... selectin_polymorphic(Employee, [Manager, Engineer]),
    8. ... selectinload(Manager.paperwork),
    9. ... )
    10. ... )
    11. >>> objects = session.scalars(stmt).all()
    12. BEGIN (implicit)
    13. SELECT employee.id, employee.name, employee.type, employee.company_id
    14. FROM employee ORDER BY employee.id
    15. [...] ()
    16. SELECT manager.id AS manager_id, employee.id AS employee_id, employee.type AS employee_type, manager.manager_name AS manager_manager_name
    17. FROM employee JOIN manager ON employee.id = manager.id
    18. WHERE employee.id IN (?) ORDER BY employee.id
    19. [...] (1,)
    20. SELECT paperwork.manager_id AS paperwork_manager_id, paperwork.id AS paperwork_id, paperwork.document_name AS paperwork_document_name
    21. FROM paperwork
    22. WHERE paperwork.manager_id IN (?)
    23. [...] (1,)
    24. SELECT engineer.id AS engineer_id, employee.id AS employee_id, employee.type AS employee_type, engineer.engineer_info AS engineer_engineer_info
    25. FROM employee JOIN engineer ON employee.id = engineer.id
    26. WHERE employee.id IN (?, ?) ORDER BY employee.id
    27. [...] (2, 3)
    28. >>> print(objects[0])
    29. Manager('Mr. Krabs')
    30. >>> print(objects[0].paperwork)
    31. [Paperwork('Secret Recipes'), Paperwork('Krabby Patty Orders')]

    Applying selectin_polymorphic() to an existing eager load

    In addition to being able to add loader options to the right side of a selectin_polymorphic() load, we may also indicate on the target of an existing load. As our setup mapping includes a parent Company entity with a Company.employees referring to Employee entities, we may illustrate a SELECT against the Company entity that eagerly loads all Employee objects as well as all attributes on their subtypes as follows, by applying Load.selectin_polymorphic() as a chained loader option; in this form, the first argument is implicit from the previous loader option (in this case ), so we only indicate the additional target subclasses we wish to load:

    1. >>> stmt = select(Company).options(
    2. ... selectinload(Company.employees).selectin_polymorphic([Manager, Engineer])
    3. ... )
    4. >>> for company in session.scalars(stmt):
    5. ... print(f"company: {company.name}")
    6. ... print(f"employees: {company.employees}")
    7. SELECT company.id, company.name
    8. FROM company
    9. [...] ()
    10. SELECT employee.company_id AS employee_company_id, employee.id AS employee_id,
    11. employee.name AS employee_name, employee.type AS employee_type
    12. FROM employee
    13. WHERE employee.company_id IN (?)
    14. [...] (1,)
    15. SELECT manager.id AS manager_id, employee.id AS employee_id, employee.name AS employee_name,
    16. employee.type AS employee_type, employee.company_id AS employee_company_id,
    17. manager.manager_name AS manager_manager_name
    18. FROM employee JOIN manager ON employee.id = manager.id
    19. WHERE employee.id IN (?) ORDER BY employee.id
    20. [...] (1,)
    21. SELECT engineer.id AS engineer_id, employee.id AS employee_id, employee.name AS employee_name,
    22. employee.type AS employee_type, employee.company_id AS employee_company_id,
    23. engineer.engineer_info AS engineer_engineer_info
    24. FROM employee JOIN engineer ON employee.id = engineer.id
    25. WHERE employee.id IN (?, ?) ORDER BY employee.id
    26. [...] (2, 3)
    27. company: Krusty Krab
    28. employees: [Manager('Mr. Krabs'), Engineer('SpongeBob'), Engineer('Squidward')]

    See also

    Eager Loading of Polymorphic Subtypes - illustrates the equivalent example as above using instead

    The behavior of may be configured on specific mappers so that it takes place by default, by using the Mapper.polymorphic_load parameter, using the value "selectin" on a per-subclass basis. The example below illustrates the use of this parameter within Engineer and Manager subclasses:

    1. class Employee(Base):
    2. __tablename__ = "employee"
    3. id = mapped_column(Integer, primary_key=True)
    4. name = mapped_column(String(50))
    5. type = mapped_column(String(50))
    6. __mapper_args__ = {"polymorphic_identity": "employee", "polymorphic_on": type}
    7. class Engineer(Employee):
    8. __tablename__ = "engineer"
    9. id = mapped_column(Integer, ForeignKey("employee.id"), primary_key=True)
    10. engineer_info = mapped_column(String(30))
    11. __mapper_args__ = {
    12. "polymorphic_load": "selectin",
    13. "polymorphic_identity": "engineer",
    14. }
    15. class Manager(Employee):
    16. __tablename__ = "manager"
    17. id = mapped_column(Integer, ForeignKey("employee.id"), primary_key=True)
    18. manager_name = mapped_column(String(30))
    19. __mapper_args__ = {
    20. "polymorphic_load": "selectin",
    21. "polymorphic_identity": "manager",
    22. }

    With the above mapping, SELECT statements against the Employee class will automatically assume the use of as a loader option when the statement is emitted.

    In contrast to selectin_polymorphic() which affects only the loading of objects, the construct affects how the SQL query for a polymorphic structure is rendered, most commonly as a series of LEFT OUTER JOINs to each of the included sub-tables. This join structure is referred towards as the polymorphic selectable. By providing for a view of several sub-tables at once, with_polymorphic() offers a means of writing a SELECT statement across several inherited classes at once with the ability to add filtering criteria based on individual sub-tables.

    is essentially a special form of the aliased() construct. It accepts as its arguments a similar form to that of , which is the base entity that is being queried, followed by a sequence of subclasses of that entity for which their specific attributes should be loaded for incoming rows:

    In order to indicate that all subclasses should be part of the entity, with_polymorphic() will also accept the string "*", which may be passed in place of the sequence of classes to indicate all classes (note this is not yet supported by ):

      The example below illustrates the same operation as illustrated in the previous section, to load all columns for Manager and Engineer at once:

      1. >>> stmt = select(employee_poly).order_by(employee_poly.id)
      2. >>> objects = session.scalars(stmt).all()
      3. BEGIN (implicit)
      4. SELECT employee.id, employee.name, employee.type, employee.company_id,
      5. manager.id AS id_1, manager.manager_name, engineer.id AS id_2, engineer.engineer_info
      6. FROM employee
      7. LEFT OUTER JOIN manager ON employee.id = manager.id
      8. LEFT OUTER JOIN engineer ON employee.id = engineer.id ORDER BY employee.id
      9. [...] ()
      10. >>> print(objects)
      11. [Manager('Mr. Krabs'), Engineer('SpongeBob'), Engineer('Squidward')]

      As is the case with selectin_polymorphic(), attributes on subclasses are already loaded:

      1. >>> print(objects[0].manager_name)
      2. Eugene H. Krabs

      As the default selectable produced by uses LEFT OUTER JOIN, from a database point of view the query is not as well optimized as the approach that selectin_polymorphic() takes, with simple SELECT statements using only JOINs emitted on a per-table basis.

      Filtering Subclass Attributes with with_polymorphic()

      The with_polymorphic() construct makes available the attributes on the included subclass mappers, by including namespaces that allow references to subclasses. The employee_poly construct created in the previous section includes attributes named .Engineer and .Manager which provide the namespace for Engineer and Manager in terms of the polymorphic SELECT. In the example below, we can use the construct to create criteria against both classes at once:

      1. >>> from sqlalchemy import or_
      2. >>> employee_poly = with_polymorphic(Employee, [Engineer, Manager])
      3. >>> stmt = (
      4. ... select(employee_poly)
      5. ... .where(
      6. ... or_(
      7. ... employee_poly.Manager.manager_name == "Eugene H. Krabs",
      8. ... employee_poly.Engineer.engineer_info
      9. ... == "Senior Customer Engagement Engineer",
      10. ... )
      11. ... )
      12. ... .order_by(employee_poly.id)
      13. ... )
      14. >>> objects = session.scalars(stmt).all()
      15. SELECT employee.id, employee.name, employee.type, employee.company_id, manager.id AS id_1,
      16. manager.manager_name, engineer.id AS id_2, engineer.engineer_info
      17. FROM employee
      18. LEFT OUTER JOIN manager ON employee.id = manager.id
      19. LEFT OUTER JOIN engineer ON employee.id = engineer.id
      20. WHERE manager.manager_name = ? OR engineer.engineer_info = ?
      21. ORDER BY employee.id
      22. [...] ('Eugene H. Krabs', 'Senior Customer Engagement Engineer')
      23. >>> print(objects)
      24. [Manager('Mr. Krabs'), Engineer('Squidward')]

      The construct, as a special case of aliased(), also provides the basic feature that does, which is that of “aliasing” of the polymorphic selectable itself. Specifically this means two or more with_polymorphic() entities, referring to the same class hierarchy, can be used at once in a single statement.

      To illustrate this feature, the example below emits a SELECT for two separate polymorphic entities, Employee joined with Engineer, and Employee joined with Manager. Since these two polymorphic entities will both be including the base employee table in their polymorphic selectable, aliasing must be applied in order to differentiate this table in its two different contexts. The two polymorphic entities are treated like two individual tables, and as such typically need to be joined with each other in some way, as illustrated below where the entities are joined on the company_id column along with some additional limiting criteria against the Employee / Manager entity:

      1. >>> manager_employee = with_polymorphic(Employee, [Manager], aliased=True, flat=True)
      2. >>> engineer_employee = with_polymorphic(Employee, [Engineer], aliased=True, flat=True)
      3. >>> stmt = (
      4. ... select(manager_employee, engineer_employee)
      5. ... .join(
      6. ... engineer_employee,
      7. ... engineer_employee.company_id == manager_employee.company_id,
      8. ... )
      9. ... .where(
      10. ... or_(
      11. ... manager_employee.name == "Mr. Krabs",
      12. ... manager_employee.Manager.manager_name == "Eugene H. Krabs",
      13. ... )
      14. ... )
      15. ... .order_by(engineer_employee.name, manager_employee.name)
      16. ... )
      17. >>> for manager, engineer in session.execute(stmt):
      18. ... print(f"{manager} {engineer}")
      19. SELECT
      20. employee_1.id, employee_1.name, employee_1.type, employee_1.company_id,
      21. manager_1.id AS id_1, manager_1.manager_name,
      22. employee_2.id AS id_2, employee_2.name AS name_1, employee_2.type AS type_1,
      23. employee_2.company_id AS company_id_1, engineer_1.id AS id_3, engineer_1.engineer_info
      24. FROM employee AS employee_1
      25. LEFT OUTER JOIN manager AS manager_1 ON employee_1.id = manager_1.id
      26. JOIN
      27. (employee AS employee_2 LEFT OUTER JOIN engineer AS engineer_1 ON employee_2.id = engineer_1.id)
      28. ON employee_2.company_id = employee_1.company_id
      29. WHERE employee_1.name = ? OR manager_1.manager_name = ?
      30. ORDER BY employee_2.name, employee_1.name
      31. [...] ('Mr. Krabs', 'Eugene H. Krabs')
      32. Manager('Mr. Krabs') Manager('Mr. Krabs')
      33. Manager('Mr. Krabs') Engineer('SpongeBob')
      34. Manager('Mr. Krabs') Engineer('Squidward')

      In the above example, the behavior of is that the polymorphic selectables remain as a LEFT OUTER JOIN of their individual tables, which themselves are given anonymous alias names. There is also a right-nested JOIN produced.

      When omitting the with_polymorphic.flat parameter, the usual behavior is that each polymorphic selectable is enclosed within a subquery, producing a more verbose form:

      1. >>> manager_employee = with_polymorphic(Employee, [Manager], aliased=True)
      2. >>> engineer_employee = with_polymorphic(Employee, [Engineer], aliased=True)
      3. >>> stmt = (
      4. ... select(manager_employee, engineer_employee)
      5. ... .join(
      6. ... engineer_employee,
      7. ... engineer_employee.company_id == manager_employee.company_id,
      8. ... )
      9. ... .where(
      10. ... or_(
      11. ... manager_employee.name == "Mr. Krabs",
      12. ... manager_employee.Manager.manager_name == "Eugene H. Krabs",
      13. ... )
      14. ... )
      15. ... .order_by(engineer_employee.name, manager_employee.name)
      16. ... )
      17. >>> print(stmt)
      18. SELECT anon_1.employee_id, anon_1.employee_name, anon_1.employee_type,
      19. anon_1.employee_company_id, anon_1.manager_id, anon_1.manager_manager_name, anon_2.employee_id AS employee_id_1,
      20. anon_2.employee_name AS employee_name_1, anon_2.employee_type AS employee_type_1,
      21. anon_2.employee_company_id AS employee_company_id_1, anon_2.engineer_id, anon_2.engineer_engineer_info
      22. FROM
      23. (SELECT employee.id AS employee_id, employee.name AS employee_name, employee.type AS employee_type,
      24. employee.company_id AS employee_company_id,
      25. manager.id AS manager_id, manager.manager_name AS manager_manager_name
      26. FROM employee LEFT OUTER JOIN manager ON employee.id = manager.id) AS anon_1
      27. JOIN
      28. (SELECT employee.id AS employee_id, employee.name AS employee_name, employee.type AS employee_type,
      29. employee.company_id AS employee_company_id, engineer.id AS engineer_id, engineer.engineer_info AS engineer_engineer_info
      30. FROM employee LEFT OUTER JOIN engineer ON employee.id = engineer.id) AS anon_2
      31. ON anon_2.employee_company_id = anon_1.employee_company_id
      32. WHERE anon_1.employee_name = :employee_name_2 OR anon_1.manager_manager_name = :manager_manager_name_1
      33. ORDER BY anon_2.employee_name, anon_1.employee_name

      The above form historically has been more portable to backends that didn’t necessarily have support for right-nested JOINs, and it additionally may be appropriate when the “polymorphic selectable” used by is not a simple LEFT OUTER JOIN of tables, as is the case when using mappings such as concrete table inheritance mappings as well as when using alternative polymorphic selectables in general.

      Configuring with_polymorphic() on mappers

      As is the case with selectin_polymorphic(), the construct also supports a mapper-configured version which may be configured in two different ways, either on the base class using the mapper.with_polymorphic parameter, or in a more modern form using the Mapper.polymorphic_load parameter on a per-subclass basis, passing the value "inline".

      Warning

      For joined inheritance mappings, prefer explicit use of within queries, or for implicit eager subclass loading use Mapper.polymorphic_load with "selectin", instead of using the mapper-level mapper.with_polymorphic parameter described in this section. This parameter invokes complex heuristics intended to rewrite the FROM clauses within SELECT statements that can interfere with construction of more complex statements, particularly those with nested subqueries that refer to the same mapped entity.

      For example, we may state our Employee mapping using as "inline" as below:

      1. class Employee(Base):
      2. __tablename__ = "employee"
      3. id = mapped_column(Integer, primary_key=True)
      4. name = mapped_column(String(50))
      5. type = mapped_column(String(50))
      6. __mapper_args__ = {"polymorphic_identity": "employee", "polymorphic_on": type}
      7. class Engineer(Employee):
      8. __tablename__ = "engineer"
      9. id = mapped_column(Integer, ForeignKey("employee.id"), primary_key=True)
      10. engineer_info = mapped_column(String(30))
      11. __mapper_args__ = {
      12. "polymorphic_load": "inline",
      13. "polymorphic_identity": "engineer",
      14. }
      15. class Manager(Employee):
      16. __tablename__ = "manager"
      17. id = mapped_column(Integer, ForeignKey("employee.id"), primary_key=True)
      18. manager_name = mapped_column(String(30))
      19. __mapper_args__ = {
      20. "polymorphic_load": "inline",
      21. "polymorphic_identity": "manager",
      22. }

      With the above mapping, SELECT statements against the Employee class will automatically assume the use of with_polymorphic(Employee, [Engineer, Manager]) as the primary entity when the statement is emitted:

      1. print(select(Employee))
      2. SELECT employee.id, employee.name, employee.type, engineer.id AS id_1,
      3. engineer.engineer_info, manager.id AS id_2, manager.manager_name
      4. FROM employee
      5. LEFT OUTER JOIN engineer ON employee.id = engineer.id
      6. LEFT OUTER JOIN manager ON employee.id = manager.id

      When using mapper-level “with polymorphic”, queries can also refer to the subclass entities directly, where they implicitly represent the joined tables in the polymorphic query. Above, we can freely refer to Manager and Engineer directly against the default entity:

      However, if we needed to refer to the Employee entity or its sub entities in separate, aliased contexts, we would again make direct use of with_polymorphic() to define these aliased entities as illustrated in .

      For more centralized control over the polymorphic selectable, the more legacy form of mapper-level polymorphic control may be used which is the Mapper.with_polymorphic parameter, configured on the base class. This parameter accepts arguments that are comparable to the construct, however common use with a joined inheritance mapping is the plain asterisk, indicating all sub-tables should be LEFT OUTER JOINED, as in:

      1. class Employee(Base):
      2. __tablename__ = "employee"
      3. id = mapped_column(Integer, primary_key=True)
      4. type = mapped_column(String(50))
      5. __mapper_args__ = {
      6. "polymorphic_identity": "employee",
      7. "with_polymorphic": "*",
      8. "polymorphic_on": type,
      9. }
      10. class Engineer(Employee):
      11. __tablename__ = "engineer"
      12. id = mapped_column(Integer, ForeignKey("employee.id"), primary_key=True)
      13. engineer_info = mapped_column(String(30))
      14. __mapper_args__ = {
      15. "polymorphic_identity": "engineer",
      16. }
      17. class Manager(Employee):
      18. __tablename__ = "manager"
      19. id = mapped_column(Integer, ForeignKey("employee.id"), primary_key=True)
      20. manager_name = mapped_column(String(30))
      21. __mapper_args__ = {
      22. "polymorphic_identity": "manager",
      23. }

      Overall, the LEFT OUTER JOIN format used by with_polymorphic() and by options such as may be cumbersome from a SQL and database optimizer point of view; for general loading of subclass attributes in joined inheritance mappings, the selectin_polymorphic() approach, or its mapper level equivalent of setting to "selectin" should likely be preferred, making use of with_polymorphic() on a per-query basis only as needed.

      Joining to specific sub-types or with_polymorphic() entities

      As a with_polymorphic() entity is a special case of , in order to treat a polymorphic entity as the target of a join, specifically when using a relationship() construct as the ON clause, we use the same technique for regular aliases as detailed at , most succinctly using PropComparator.of_type(). In the example below we illustrate a join from the parent Company entity along the one-to-many relationship Company.employees, which is configured in the to link to Employee objects, using a with_polymorphic() entity as the target:

      1. >>> employee_plus_engineer = with_polymorphic(Employee, [Engineer])
      2. >>> stmt = (
      3. ... select(Company.name, employee_plus_engineer.name)
      4. ... .join(Company.employees.of_type(employee_plus_engineer))
      5. ... .where(
      6. ... or_(
      7. ... employee_plus_engineer.name == "SpongeBob",
      8. ... employee_plus_engineer.Engineer.engineer_info
      9. ... == "Senior Customer Engagement Engineer",
      10. ... )
      11. ... )
      12. ... )
      13. >>> for company_name, emp_name in session.execute(stmt):
      14. ... print(f"{company_name} {emp_name}")
      15. SELECT company.name, employee.name AS name_1
      16. FROM company JOIN (employee LEFT OUTER JOIN engineer ON employee.id = engineer.id) ON company.id = employee.company_id
      17. WHERE employee.name = ? OR engineer.engineer_info = ?
      18. [...] ('SpongeBob', 'Senior Customer Engagement Engineer')
      19. Krusty Krab SpongeBob
      20. Krusty Krab Squidward

      More directly, is also used with inheritance mappings of any kind to limit a join along a relationship() to a particular sub-type of the ’s target. The above query could be written strictly in terms of Engineer targets as follows:

      1. >>> stmt = (
      2. ... select(Company.name, Engineer.name)
      3. ... .join(Company.employees.of_type(Engineer))
      4. ... .where(
      5. ... or_(
      6. ... Engineer.name == "SpongeBob",
      7. ... Engineer.engineer_info == "Senior Customer Engagement Engineer",
      8. ... )
      9. ... )
      10. ... )
      11. >>> for company_name, emp_name in session.execute(stmt):
      12. ... print(f"{company_name} {emp_name}")
      13. SELECT company.name, employee.name AS name_1
      14. FROM company JOIN (employee JOIN engineer ON employee.id = engineer.id) ON company.id = employee.company_id
      15. WHERE employee.name = ? OR engineer.engineer_info = ?
      16. [...] ('SpongeBob', 'Senior Customer Engagement Engineer')
      17. Krusty Krab SpongeBob
      18. Krusty Krab Squidward

      It can be observed above that joining to the Engineer target directly, rather than the “polymorphic selectable” of with_polymorphic(Employee, [Engineer]) has the useful characteristic of using an inner JOIN rather than a LEFT OUTER JOIN, which is generally more performant from a SQL optimizer point of view.

      The use of illustrated with the Select.join() method in the previous section may also be applied equivalently to , such as selectinload() and .

      As a basic example, if we wished to load Company objects, and additionally eagerly load all elements of Company.employees using the with_polymorphic() construct against the full hierarchy, we may write:

      1. >>> all_employees = with_polymorphic(Employee, "*")
      2. >>> stmt = select(Company).options(selectinload(Company.employees.of_type(all_employees)))
      3. >>> for company in session.scalars(stmt):
      4. ... print(f"company: {company.name}")
      5. ... print(f"employees: {company.employees}")
      6. SELECT company.id, company.name
      7. FROM company
      8. [...] ()
      9. SELECT employee.company_id AS employee_company_id, employee.id AS employee_id,
      10. employee.name AS employee_name, employee.type AS employee_type, manager.id AS manager_id,
      11. manager.manager_name AS manager_manager_name, engineer.id AS engineer_id,
      12. engineer.engineer_info AS engineer_engineer_info
      13. FROM employee
      14. LEFT OUTER JOIN manager ON employee.id = manager.id
      15. LEFT OUTER JOIN engineer ON employee.id = engineer.id
      16. WHERE employee.company_id IN (?)
      17. [...] (1,)
      18. company: Krusty Krab
      19. employees: [Manager('Mr. Krabs'), Engineer('SpongeBob'), Engineer('Squidward')]

      The above query may be compared directly to the version illustrated in the previous section Applying selectin_polymorphic() to an existing eager load.

      See also

      - illustrates the equivalent example as above using selectin_polymorphic() instead

      Single Table Inheritance Setup

      This section discusses single table inheritance, described at Single Table Inheritance, which uses a single table to represent multiple classes in a hierarchy.

      .

      In contrast to joined inheritance mappings, the construction of SELECT statements for single inheritance mappings tends to be simpler since for an all-single-inheritance hierarchy, there’s only one table.

      Regardless of whether or not the inheritance hierarchy is all single-inheritance or has a mixture of joined and single inheritance, SELECT statements for single inheritance differentiate queries against the base class vs. a subclass by limiting the SELECT statement with additional WHERE criteria.

      As an example, a query for the single-inheritance example mapping of Employee will load objects of type Manager, Engineer and Employee using a simple SELECT of the table:

      1. >>> stmt = select(Employee).order_by(Employee.id)
      2. >>> for obj in session.scalars(stmt):
      3. ... print(f"{obj}")
      4. BEGIN (implicit)
      5. SELECT employee.id, employee.name, employee.type
      6. FROM employee ORDER BY employee.id
      7. [...] ()
      8. Manager('Mr. Krabs')
      9. Engineer('SpongeBob')
      10. Engineer('Squidward')

      When a load is emitted for a specific subclass, additional criteria is added to the SELECT that limits the rows, such as below where a SELECT against the Engineer entity is performed:

      1. >>> stmt = select(Engineer).order_by(Engineer.id)
      2. >>> objects = session.scalars(stmt).all()
      3. SELECT employee.id, employee.name, employee.type, employee.engineer_info
      4. FROM employee
      5. WHERE employee.type IN (?) ORDER BY employee.id
      6. [...] ('engineer',)
      7. >>> for obj in objects:
      8. ... print(f"{obj}")
      9. Engineer('SpongeBob')
      10. Engineer('Squidward')

      Optimizing Attribute Loads for Single Inheritance

      The default behavior of single inheritance mappings regarding how attributes on subclasses are SELECTed is similar to that of joined inheritance, in that subclass-specific attributes still emit a second SELECT by default. In the example below, a single Employee of type Manager is loaded, however since the requested class is Employee, the Manager.manager_name attribute is not present by default, and an additional SELECT is emitted when it’s accessed:

      1. >>> mr_krabs = session.scalars(select(Employee).where(Employee.name == "Mr. Krabs")).one()
      2. BEGIN (implicit)
      3. SELECT employee.id, employee.name, employee.type
      4. FROM employee
      5. WHERE employee.name = ?
      6. [...] ('Mr. Krabs',)
      7. >>> mr_krabs.manager_name
      8. SELECT employee.manager_name AS employee_manager_name
      9. FROM employee
      10. WHERE employee.id = ? AND employee.type IN (?)
      11. [...] (1, 'manager')
      12. 'Eugene H. Krabs'

      To alter this behavior, the same general concepts used to eagerly load these additional attributes used in joined inheritance loading apply to single inheritance as well, including use of the option as well as the with_polymorphic() option, the latter of which simply includes the additional columns and from a SQL perspective is more efficient for single-inheritance mappers:

      1. >>> employees = with_polymorphic(Employee, "*")
      2. >>> stmt = select(employees).order_by(employees.id)
      3. >>> objects = session.scalars(stmt).all()
      4. BEGIN (implicit)
      5. SELECT employee.id, employee.name, employee.type,
      6. employee.manager_name, employee.engineer_info
      7. FROM employee ORDER BY employee.id
      8. [...] ()
      9. >>> for obj in objects:
      10. ... print(f"{obj}")
      11. Manager('Mr. Krabs')
      12. Engineer('SpongeBob')
      13. Engineer('Squidward')
      14. >>> objects[0].manager_name
      15. 'Eugene H. Krabs'

      With the above mapping, the Manager and Engineer classes will have their columns included in SELECT statements against the Employee entity automatically:

      1. >>> print(select(Employee))
      2. SELECT employee.id, employee.name, employee.type,
      3. employee.manager_name, employee.engineer_info
      4. FROM employee

      Inheritance Loading API

      function sqlalchemy.orm.with_polymorphic(base: Union[_O, Mapper[_O]], classes: Union[Literal[‘*‘], Iterable[Type[Any]]], selectable: Union[Literal[False, None], ] = False, flat: bool = False, polymorphic_on: Optional[ColumnElement[Any]] = None, aliased: bool = False, innerjoin: bool = False, adapt_on_names: bool = False, _use_mapper_path: bool = False) → [_O]

      Produce an AliasedClass construct which specifies columns for descendant mappers of the given base.

      Using this method will ensure that each descendant mapper’s tables are included in the FROM clause, and will allow filter() criterion to be used against those tables. The resulting instances will also have those columns already loaded so that no “post fetch” of those columns will be required.

      See also

      - full discussion of with_polymorphic().

      • Parameters:

        • base – Base class to be aliased.

        • classes – a single class or mapper, or list of class/mappers, which inherit from the base class. Alternatively, it may also be the string '*', in which case all descending mapped classes will be added to the FROM clause.

        • aliased – when True, the selectable will be aliased. For a JOIN, this means the JOIN will be SELECTed from inside of a subquery unless the flag is set to True, which is recommended for simpler use cases.

        • flat – Boolean, will be passed through to the FromClause.alias() call so that aliases of objects will alias the individual tables inside the join, rather than creating a subquery. This is generally supported by all modern databases with regards to right-nested joins and generally produces more efficient queries. Setting this flag is recommended as long as the resulting SQL is functional.

        • selectable

          a table or subquery that will be used in place of the generated FROM clause. This argument is required if any of the desired classes use concrete table inheritance, since SQLAlchemy currently cannot generate UNIONs among tables automatically. If used, the selectable argument must represent the full set of tables and columns mapped by every mapped class. Otherwise, the unaccounted mapped columns will result in their table being appended directly to the FROM clause which will usually lead to incorrect results.

          When left at its default value of False, the polymorphic selectable assigned to the base mapper is used for selecting rows. However, it may also be passed as None, which will bypass the configured polymorphic selectable and instead construct an ad-hoc selectable for the target classes given; for joined table inheritance this will be a join that includes all target mappers and their subclasses.

        • polymorphic_on – a column to be used as the “discriminator” column for the given selectable. If not given, the polymorphic_on attribute of the base classes’ mapper will be used, if any. This is useful for mappings that don’t have polymorphic loading behavior by default.

        • adapt_on_names

          Passes through the aliased.adapt_on_names parameter to the aliased object. This may be useful in situations where the given selectable is not directly related to the existing mapped selectable.

          New in version 1.4.33.

      function sqlalchemy.orm.selectin_polymorphic(base_cls: _EntityType[Any], classes: Iterable[Type[Any]]) → _AbstractLoad

      Indicate an eager load should take place for all attributes specific to a subclass.

      This uses an additional SELECT with IN against all matched primary key values, and is the per-query analogue to the setting on the mapper.polymorphic_load parameter.

      New in version 1.2.

      See also

      ORM Querying Guide