Queryable.OuterJoin
1.0.2
Switching to namespace from FSC to Fluentsoft
dotnet add package Queryable.OuterJoin --version 1.0.2
NuGet\Install-Package Queryable.OuterJoin -Version 1.0.2
<PackageReference Include="Queryable.OuterJoin" Version="1.0.2" />
paket add Queryable.OuterJoin --version 1.0.2
#r "nuget: Queryable.OuterJoin, 1.0.2"
// Install Queryable.OuterJoin as a Cake Addin #addin nuget:?package=Queryable.OuterJoin&version=1.0.2 // Install Queryable.OuterJoin as a Cake Tool #tool nuget:?package=Queryable.OuterJoin&version=1.0.2
Queryable Class
Definitions
Namespace: FSC.System.Linq
Assembly: fsc.system.linq.queryable.dll
Package: Queryable.OuterJoin
Provides a set of static (Shared in Visual Basic) methods for querying data structures that implement IQueryable<T>.
public static class Queryable
Methods
Methods | Description |
---|---|
LeftOuterJoin | Correlates all records from the left table, and the matching records from the right table based on matching keys |
RightOuterJoin | Correlates all records from the right table, and the matching records from the left table based on matching keys |
FullOuterJoin | Correlates all records from the right table, and all records from the left table based on matching keys |
Select<T, TResult, T1, T2>(IQueryable<T> source, Expression<Func<T1, T2, TResult>> selector) | Projects each element of a sequence into a new form. |
IQueryable<TResult> Select<T, TResult, T1, T2, T3>(IQueryable<T> source, Expression<Func<T1, T2, T3, TResult>> selector) | Projects each element of a sequence into a new form. |
IQueryable<T> Where<T, T1, T2, T3>(this IQueryable<T> source, Expression<Func<T1, T2, T3, bool>> predicate) | Filters a sequence of values based on a predicate with three arguments taken from source value by splitting. |
IQueryable<T> Where<T, T1, T2>(IQueryable<T> source, Expression<Func<T1, T2, bool>> predicate) | Filters a sequence of values based on a predicate with two arguments taken from source value by splitting. |
Correlates all records from the left table, and the matching records from the right table based on matching keys
public static IQueryable<TResult> LeftOuterJoin<TOuter, TInner, TKey, TResult>(this
IQueryable<TOuter> outer,
IEnumerable<TInner> inner,
Expression<Func<TOuter, TKey>> outerKey,
Expression<Func<TInner, TKey>> innerKey,
Expression<Func<TOuter, TInner?, TResult>> resultSelector);
Type Parameters
TOuter
The type of the elements of the first sequence.
TInner
The type of the elements of the second sequence.
TKey
The type of the keys returned by the key selector functions.
TResult
The type of the result elements.
Parameters
outer
IQueryable<TOuter>
The first sequence to join.
inner
IEnumerable<TInner>
The sequence to join to the first sequence.
outerKeySelector
Expression<Func<TOuter,TKey>>
A function to extract the join key from each element of the first sequence.
innerKeySelector
Expression<Func<TInner,TKey>>
A function to extract the join key from each element of the second sequence.
resultSelector
Expression<Func<TOuter,TInner,TResult>>
A function to create a result element from two matching elements.
Returns
IQueryable<TResult>
An IQueryable<T> that has elements of type TResult obtained by performing an left outer join on two sequences.
Example
ctx.Departments.LeftOuterJoin(ctx.Employees,
z => z.ID,
z => z.DepartmentID,
(department, employee) => new
{
Department = department.Name,
Employee = employee.Name,
})
public static IQueryable<TResult> RughtOuterJoin<TOuter, TInner, TKey, TResult>(this
IEnumerable<TOuter> outer,
IQueryable<TInner> inner,
Expression<Func<TOuter, TKey>> outerKey,
Expression<Func<TInner, TKey>> innerKey,
Expression<Func<TOuter, TInner?, TResult>> resultSelector);
Type Parameters
TOuter
The type of the elements of the first sequence.
TInner
The type of the elements of the second sequence.
TKey
The type of the keys returned by the key selector functions.
TResult
The type of the result elements.
Parameters
outer
IEnumerable<TOuter>
The first sequence to join.
inner
IQueryable<TInner>
The sequence to join to the first sequence.
outerKeySelector
Expression<Func<TOuter,TKey>>
A function to extract the join key from each element of the first sequence.
innerKeySelector
Expression<Func<TInner,TKey>>
A function to extract the join key from each element of the second sequence.
resultSelector
Expression<Func<TOuter,TInner,TResult>>
A function to create a result element from two matching elements.
Returns
IQueryable<TResult>
An IQueryable<T> that has elements of type TResult obtained by performing an right outer join on two sequences.
Example
ctx.Departments.RightOuterJoin(ctx.Employees,
z => z.ID,
z => z.DepartmentID,
(department, employee) => new
{
Department = department.Name,
Employee = employee.Name,
})
public static IQueryable<TResult> FullOuterJoin<TOuter, TInner, TKey, TResult>(this
IQueryable<TOuter> outer,
IQueryable<TInner> inner,
Expression<Func<TOuter, TKey>> outerKey,
Expression<Func<TInner, TKey>> innerKey,
Expression<Func<TOuter, TInner?, TResult>> resultSelector);
Type Parameters
TOuter
The type of the elements of the first sequence.
TInner
The type of the elements of the second sequence.
TKey
The type of the keys returned by the key selector functions.
TResult
The type of the result elements.
Parameters
outer
IQueryable<TOuter>
The first sequence to join.
inner
IQueryable<TInner>
The sequence to join to the first sequence.
outerKeySelector
Expression<Func<TOuter,TKey>>
A function to extract the join key from each element of the first sequence.
innerKeySelector
Expression<Func<TInner,TKey>>
A function to extract the join key from each element of the second sequence.
resultSelector
Expression<Func<TOuter,TInner,TResult>>
A function to create a result element from two matching elements.
Returns
IQueryable<TResult>
An IQueryable<T> that has elements of type TResult
obtained by performing an full outer join on two sequences.
Example
ctx.Departments.FullOuterJoin(ctx.Employees,
z => z.ID,
z => z.DepartmentID,
(department, employee) => new
{
Department = department.Name,
Employee = employee.Name,
})
Initial challenge
It all started as a problem a lot of people are facing related to the implementation of Left Outer Join in Entity Framework. Here is a little reminder of the problem.
Everybody knows Entity Framework (EF for simplicity) supports joins between two collections based on a key and equality:
[EntityTypeConfiguration(typeof(DepartmentConfig))]
public class Department
{
public int ID { get; set; }
public string Name { get; set; } = null!;
public ICollection<Employee> Employees { get; set; } = new List<Employee>();
}
[EntityTypeConfiguration(typeof(EmployeeConfig))]
public class Employee
{
public int ID { get; set; }
public string Name { get; set; } = null!;
public int? DepartmentID { get; set; }
public Department? Department { get; set;}
}
public class DepartmentEmployee
{
public Department Department { get; set; }
public Employee Employee { get; set; }
}
public class TestDbContext : DbContext
{
public DbSet<Department> Departments { get; set; } = null!;
public DbSet<Employee> Employees { get; set; } = null!;
public TestDbContext(DbContextOptions<TestDbContext> dbContextOptions) : base(dbContextOptions)
{
}
}
The Inner Join request looks like:
await using var ctx = _fixture.CreateDbContext(init: true);
var query = ctx.Departments.Join(ctx.Employees,
z => z.ID,
z => z.DepartmentID,
(department, employee) => new
{
Department = department.Name,
Employee = employee.Name,
});
_output.WriteLine(query.ToQueryString());
The SQL output:
SELECT d.Name AS Department, e.Name AS Employee
FROM Departments AS d
INNER JOIN Employees AS e ON d.ID = e.department_id
As a result, we will have a collection of Departments with matching Employees. If an Departments does not have an associated Employee – it will be left out of result.
Time to talk about Left Outer Join. By “the book” the Left Outer Join is a combination of using the GroupJoin() method along with SelectMany() and DefaultIfEmpty() methods. For above context it will look like:
var query = ctx.Departments
.GroupJoin(ctx.Employees,
d => d.ID,
e => e.DepartmentID,
(dep, emps) => new { dep, emps })
.SelectMany(
g=> g.emps.DefaultIfEmpty(),
(a, e) => new
{
Department = a.dep.Name,
Employee = e.Name
});
_output.WriteLine(query.ToQueryString());
The Sql output:
SELECT d.Name AS Department, e.Name AS Employee
FROM Departments AS d
LEFT JOIN Employees AS e ON d.ID = e.department_id
The EF provider is able to recognize the construct and use a LEFT OUTER JOIN in the database query. We have what we need. Really?
Let’s analyze the last expression.
GroupJoin()
correlates the elements of Department with Employee by DepartmentID. Its result is a collection of contain anonymous objects with a Department and all associated Employee items.DefaultIfEmpty()
– returns the default item, in our case null, if the collection is empty or all items in a collection. In our case if no Employee are associated with a Department, then it's employee’s collection will contain one item with null value.SelectMany()
– for each department it calls result selector for each associated Employee. Because Employee’s collection uses DefaultIfEmpty() – it will call at least once.
Now we have an expression that will generate Left Outer Join in database query. Is that expression good and usable? No! Let me explain what I mean.
Recall the expression with a simple join from above. There is an easy way to create an Expression to get result. For instance, this one:
public static Expression<Func<Department, Employee, DepartmentEmployee>> Expression =
(department, employee) => new DepartmentEmployee
{
Department = department,
Employee = employee
};
Using this expression I can write the Join:
result = ctx.Departments.Join(
ctx.Employees,
d => d.ID,
e => e.DepartmentID,
Expression);
This is much better.It would be nice to have the ability to use a similar expression with that expression above for a Left Outer Join. Unfortunately – this is not possible.
Well, “Not Possible” is not in my vocabulary. Let’s make it possible!
We need an extension method which looks like this one:
public static IQueryable<TResult> OuterJoin<TOuter, TInner, TKey, TResult>(this IQueryable<TOuter> outer,
IQueryable<TInner> inner,
Expression<Func<TOuter, TKey>> outerKey,
Expression<Func<TInner, TKey>> innerKey,
Expression<Func<TOuter, TInner?, TResult>> resultSelector)
{
...
}
It has very similar signature as the Join() method. Let’s try to make a body of this method using expression for Left Outer Join:
var query = uter.GroupJoin(inner, outerKey, innerKey, (o, i) => new { o, i }
.SelectMany(g => g.i.DefaultIfEmpty(), (tmp, i) => new { Outer = tmp.o, Inner = i });
The type of query is IQueryable<{TOuter, TInner }>
. It has the fields we need, but it is an anonymous
type and it is only known only inside the scope of q2. Inside a scope we can access to its items but
we cannot use it to call a result selector expression from outside.
To make it possible to call an external expression we will need some kind of expression transformer:
public static Expression<Func<T, TResult>> ExpressionTransformer<T, TOuter, TInner, TResult>(
Expression<Func<TOuter, TInner?, TResult>> resultSelector)
{
…
}
which transforms the expression with the anonymous type from q2 to use resultSelector expression.
It is not so difficult to make by providing a transformation to split the anonymous type’s fields into
separate instances and here the SplitParameter
comes to the scene:
public static class SplitParameter<T1, T2>
{
public static Expression<Func<T, TResult>> Translate<T, TResult>(
Expression<Func<T1, T2, TResult>> expression)
{ … }
public static IQueryable<TResult> Select<T, TResult>(
IQueryable<T> source,
Expression<Func<T1, T2, TResult>> selector)
{
return source.Select(Translate<T, TResult>(selector));
}
}
Using this method it's easy to implement a method extension for Left Outer Join:
public static IQueryable<TResult> OuterJoin<TOuter, TInner, TKey, TResult>(this IQueryable<TOuter> outer,
IQueryable<TInner> inner,
Expression<Func<TOuter, TKey>> outerKey,
Expression<Func<TInner, TKey>> innerKey,
Expression<Func<TOuter, TInner?, TResult>> resultSelector)
{
return SplitParameter<TOuter, TInner?>.Select(
outer
.GroupJoin(
inner,
outerKey,
innerKey,
(o, i) => new { o, i })
.SelectMany(z => z.i.DefaultIfEmpty(),
(t, i) => new { t.o, i }),
resultSelector);
}
The last piece of the puzzle is using the ExpressionVisitor to transform expressions:
public static Expression<Func<T, TResult>> Translate<T, TResult>(
Expression<Func<T1, T2, TResult>> expression)
{
return new SplitParameterVisitor<T, TResult>(expression).Translate();
}
private class SplitParameterVisitor<T, TResult> : ExpressionVisitor
{
private readonly MemberExpression _arg1Expression;
private readonly MemberExpression _arg2Expression;
private readonly Expression<Func<T1, T2, TResult>> _expression;
private readonly ParameterExpression _p1;
private readonly ParameterExpression _p2;
private readonly ParameterExpression _parameterExpression;
public SplitParameterVisitor(Expression<Func<T1, T2, TResult>> expression)
{
_expression = expression;
_p1 = expression.Parameters[0];
_p2 = expression.Parameters[1];
var type = typeof(T);
const BindingFlags FLAGS = BindingFlags.Instance
| BindingFlags.Public
| BindingFlags.GetProperty;
var properties = type.GetProperties(FLAGS).ToArray();
var prop1Name = properties
.Where(z => z.PropertyType.IsAssignableTo(typeof(T1)))
.Select(z => z.Name)
.FirstOrDefault();
var prop2Name = typeof(T1) == typeof(T2)
? properties.Where(z => z.PropertyType.IsAssignableTo(typeof(T2)))
.Skip(1)
.Select(z => z.Name).FirstOrDefault()
: properties.Where(z => z.PropertyType.IsAssignableTo(typeof(T2)))
.Select(z => z.Name).FirstOrDefault();
_parameterExpression = Expression.Parameter(type);
_arg1Expression = Expression.Property(_parameterExpression, prop1Name);
_arg2Expression = Expression.Property(_parameterExpression, prop2Name);
}
public Expression<Func<T, TResult>> Translate()
{
return Expression.Lambda<Func<T, TResult>>(base.Visit(_expression.Body), _parameterExpression);
}
protected override Expression VisitParameter(ParameterExpression node)
{
if (node == _p1)
{
return _arg1Expression;
}
if (node == _p2)
{
return _arg2Expression;
}
return base.VisitParameter(node);
}
}
Here in the visitor’s constructor, we are looking for properties of type T1 and T2 and making expression Property:
_parameterExpression = Expression.Parameter(type);
_arg1Expression = Expression.Property(_parameterExpression, prop1Name);
_arg2Expression = Expression.Property(_parameterExpression, prop2Name);
In addition, expression parameter created based on type T as well as getting parameters from argument expression:
_p1 = expression.Parameters[0];
_p2 = expression.Parameters[1];
At last, in the VisitParameter override method, we replace all occurrence of _p1 and _p2 nodes by arg1Expression and arg2Expression respectively.
The last call is to the Translate method, which just executes an expression visitor.
Finally, we can write the Left Outer Join expression:
query = ctx.Departments.OuterJoin(ctx.Employees,
z => z.ID,
z => z.DepartmentID,
(department, employee) => new
{
Department = department.Name,
Employee = employee.Name,
});
_output.WriteLine(query.ToQueryString());
);
The Sql output:
SELECT d.Name AS Department, e.Name AS Employee
FROM Departments AS d
LEFT JOIN Employees AS e ON d.ID = e.department_id
and this is excatly the same SQL request from above.
Puzzle Soved!
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 is compatible. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- No dependencies.
-
net6.0
- No dependencies.
-
net7.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.