How to simplify comprehension of a complex Entity Framework Query
Posted on: July 17, 2012
You may run into the case of where the Linq to Entity query is long and complex.
If you are using a layer of abstraction to access your database (DAL "Data Access Layer) that return `IQueryable you may be surprise that you can divide your query into multiple IQueryable and to merge all Linq to Entity back into a single IQueryable.
The first step is to create your method into your DAL.
1public IQueryable<MyModel> GetMyModel() {2 //Linq to Entity is returned here3}
The second step is to create query in logical group. Let say you desire to have all objects of your MyModel that is over 100$ and another condition is that you want also all those over 20$ if they have a discount.
You can do it with a single Linq to Entity, but in some situation (more complex than the one exposed here) you may found easier to cut the problem in sub query.
This can be done by using multiple IQueryable.
1public IQueryable<MyModel> GetMyModel() {2 var query1 = //Linq to Entity with condition for > 100$3 var query2 = //Linq to Entity with condition for > 20$ && Discount > 0$4 //return query1 + query 25}
The last step is to merge query 1 and query 2. You can do it in two differents way with Linq. The first one is to use the Union keyword and the second is with Concat.
The difference between the two is that union will merge only when not already present when concat will merge everything. Here is a small snippet of code that illustrate this theory.
1var set1 = new int[] { 1, 2, 3 };2var set2 = new int[] { 3, 4, 5 };34var result= set1.AsQueryable().Concat(set2.AsQueryable()); // count == 65var result2 = set1.AsQueryable().Union(set2.AsQueryable()); // count == 5
Our example could look with union like this:
1public IQueryable<MyModel> GetMyModel() {2 var query1 = //Linq to Entity with condition for > 100$3 var query2 = //Linq to Entity with condition for > 20$ && Discount > 0$4 return query1.union(query2); //Both condition5}
Since we are using IQueryable
, none of the query are executed until their execution. That mean that that Sql Server will receive a single query (with multiple sub query).