Non-trivial queries often require data of several entity types (tables). In the SQL world, you can make that happen by “joining” two or more tables using a join condition.
Let’s consider an entity User, which has a one-to-many relation to an Address entity. Then we want to query for users living on “Sesame Street”: We would have to join the Address entity with the User entity using the user ID and define a WHERE condition on the Address entity:
1 2 3 4 | QueryBuilder<User> queryBuilder = userDao.queryBuilder(); queryBuilder.join(Address.class, AddressDao.Properties.userId) .where(AddressDao.Properties.Street.eq("Sesame Street")); List<User> users = queryBuilder.list(); |
The join requires the target entity class as a parameter and a join property of each entity. In the example, just the join property of the Address entity is defined, because the primary key property is used by default. In other words, the query results in users, which have an Address entity with a userId equal to the User entity’s ID and also have a specific street.
QueryBuilder Join API
Because you can omit the join property when the primary key property can be used, there are three overloaded join methods available in QueryBuilder:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | /** * Expands the query to another entity type by using a JOIN. * The primary key property of the primary entity for * this QueryBuilder is used to match the given destinationProperty. */ public <J> Join<T, J> join(Class<J> destinationEntityClass, Property destinationProperty) /** * Expands the query to another entity type by using a JOIN. * The given sourceProperty is used to match the primary * key property of the given destinationEntity. */ public <J> Join<T, J> join(Property sourceProperty, Class<J> destinationEntityClass) /** * Expands the query to another entity type by using a JOIN. * The given sourceProperty is used to match the given * destinationProperty of the given destinationEntity. */ public <J> Join<T, J> join(Property sourceProperty, Class<J> destinationEntityClass, Property destinationProperty) |
Chained Joins
In addition, greenDAO allows to chain joins across multiple tables. Here, you define a join using another join and a destination entity. In this case, the destination entity of the first join becomes the starting entity for the second join.
The QueryBuilder API for chained joins looks like this:
1 2 3 4 5 6 7 8 9 10 11 | /** * Expands the query to another entity type by using a JOIN. * The given sourceJoin's property is used to match the * given destinationProperty of the given destinationEntity. * Note that destination entity of the given join is used * as the source for the new join to add. In this way, * it is possible to compose complex "join of joins" across * several entities if required. */ public <J> Join<T, J> join(Join<?, T> sourceJoin, Property sourceProperty, Class<J> destinationEntityClass, Property destinationProperty) |
Let’s look at another example with three entities: City, Country, and Continent. If we want to query all cities in Europe with a population of at least one million, it would look like this:
1 2 3 4 5 6 | QueryBuilder<City> qb = cityDao.queryBuilder().where(Properties.Population.ge(1000000)); Join country = qb.join(Properties.CountryId, Country.class); Join continent = qb.join(country, CountryDao.Properties.ContinentId, Continent.class, ContinentDao.Properties.Id); continent.where(ContinentDao.Properties.Name.eq("Europe")); List<City> bigEuropeanCities = qb.list(); |
Self Join / Tree example
A join can also be used with relations referencing a single entity. For example, we want to find all persons, whose grandfather’s name is “Lincoln”. Let’s assume we have a Person entity with a fatherId property pointing to the same entity. Then, the query is built like this:
1 2 3 4 5 | QueryBuilder<Person> qb = personDao.queryBuilder(); Join father = qb.join(Person.class, Properties.FatherId); Join grandfather = qb.join(father, Properties.FatherId, Person.class, Properties.Id); grandfather.where(Properties.Name.eq("Lincoln")); List<Person> lincolnDescendants = qb.list(); |
As you can see, joins are a powerful tool to build queries spanning multiple entity types or relations.