Accueil > > > ENTITY FRAMEWORK - AVOIR UN INCLUDE TYPÉ
ENTITY FRAMEWORK - AVOIR UN INCLUDE TYPÉ
Information sur la source
Description
Entity Framework propose une méhtode Include permettant d'inclure des objets connexes lors du chargement de certaines entités. Cette méthode prend un String en paramètre, ce qui ne permet pas d'avoir une vérification lors de la compilation. Le bout de code ci-dessous permet de générer différentes méthodes Include permettant d'avoir des includes typés. Il existe de nombreuses solutions permettant d'attendre cette objectif, cette solution permet d'inclure plusieurs profondeurs d'objets connexes, y compris lorsqu'un de ces objets est une collection. Il est ainsi possible de faire : var q = entities.Orders.Include(o => o.Customer.Addresses, a => a.Country).Where(o => o.CustomerId == 3); La source est composé de 2 fichiers, un fichier ExpressionExtensions contenant quelques méthodes d'extension permettant de connaitre le nom d'une propriété à partir d'une lambda et d'un fichier T4 générant les différentes méthodes Include.
Source
- // ExpressionExtensions.cs
-
- public static class ExpressionExtensions
- {
- public static String GetPropertyName<T, TProperty>(this Expression<Func<T, TProperty>> expression)
- {
- return GetPropertyName(expression);
- }
- public static String GetPropertyName(this Expression expression)
- {
- LambdaExpression lambdaExpression = expression as LambdaExpression;
- if (lambdaExpression != null)
- {
- expression = lambdaExpression.Body;
- }
-
- MemberExpression memberExpression = expression as MemberExpression;
- if (memberExpression != null)
- {
- String parentPropertyName = String.Empty;
- if (memberExpression.Expression is MemberExpression)
- {
- parentPropertyName = ExpressionExtensions.GetPropertyName(memberExpression.Expression) + ".";
- }
- return parentPropertyName + memberExpression.Member.Name;
- }
-
- throw new NotImplementedException();
- }
- }
-
- // ObjectQueryExtensions.Include.tt
-
- <#@ template debug="true" hostSpecific="true" #>
- <#@ output extension=".cs" #>
- <#@ Assembly Name="System.Core.dll" #>
- <#@ Assembly Name="System.Windows.Forms.dll" #>
- <#@ import namespace="System" #>
- <#@ import namespace="System.IO" #>
- <#@ import namespace="System.Diagnostics" #>
- <#@ import namespace="System.Linq" #>
- <#@ import namespace="System.Collections" #>
- <#@ import namespace="System.Collections.Generic" #>
- <#
- Byte maxDepth = 6;
- #>
-
- using System;
- using System.Collections.Generic;
- using System.Data.Objects;
- using System.Data.Objects.DataClasses;
- using System.Linq;
- using System.Linq.Expressions;
-
- namespace Magelia.WebStore.Data.Entities
- {
- public static partial class ObjectQueryExtensions
- {
- internal static String GetPath(params LambdaExpression[] lambdas)
- {
- return String.Join(".",
- lambdas
- .Select(l => l.GetPropertyName())
- .ToArray()
- );
- }
-
- <#
- for(Byte i = 1; i < maxDepth; i++)
- {
- int combinationCount = (int)Math.Pow(2, i);
- for (int j = 0; j < combinationCount; j++)
- {
- WriteMethod(i, j);
- }
- }
- #>
- }
- }
-
- <#+
- void WriteMethod(Byte depth, int combinationValue)
- {
- String[] genericsType = new String[depth+1];
- genericsType[0] = "TEntity";
- int i = depth-1;
- while (i >= 0)
- {
- String genericType = "T" + i.ToString();
- if ((combinationValue & (1 << i)) != 0)
- {
- genericType = "IEnumerable<" + genericType + ">";
- }
- genericsType[i+1] = genericType;
- i--;
- }
- String[] parametersType = new String[depth];
- for(i = 0; i < depth; i++)
- {
- parametersType[i] = String.Format("Expression<Func<T{0}, {1}>> selector{2}", (i == 0) ? "Entity" : (i-1).ToString(), genericsType[i+1], i);
- }
- #>
- public static ObjectQuery<TEntity> Include<TEntity, <#= String.Join(", ", Enumerable.Range(0, depth).Select(j => "T" + j.ToString()).ToArray()) #>>(this ObjectQuery<TEntity> objectSet, <#= String.Join(", ", parametersType) #>)
- where TEntity : class
- {
- return objectSet.Include(GetPath(<#= String.Join(", ", Enumerable.Range(0, depth).Select(j => "selector" + j.ToString()).ToArray()) #>));
- }
- <#+
- }
- #>
-
// ExpressionExtensions.cs
public static class ExpressionExtensions
{
public static String GetPropertyName<T, TProperty>(this Expression<Func<T, TProperty>> expression)
{
return GetPropertyName(expression);
}
public static String GetPropertyName(this Expression expression)
{
LambdaExpression lambdaExpression = expression as LambdaExpression;
if (lambdaExpression != null)
{
expression = lambdaExpression.Body;
}
MemberExpression memberExpression = expression as MemberExpression;
if (memberExpression != null)
{
String parentPropertyName = String.Empty;
if (memberExpression.Expression is MemberExpression)
{
parentPropertyName = ExpressionExtensions.GetPropertyName(memberExpression.Expression) + ".";
}
return parentPropertyName + memberExpression.Member.Name;
}
throw new NotImplementedException();
}
}
// ObjectQueryExtensions.Include.tt
<#@ template debug="true" hostSpecific="true" #>
<#@ output extension=".cs" #>
<#@ Assembly Name="System.Core.dll" #>
<#@ Assembly Name="System.Windows.Forms.dll" #>
<#@ import namespace="System" #>
<#@ import namespace="System.IO" #>
<#@ import namespace="System.Diagnostics" #>
<#@ import namespace="System.Linq" #>
<#@ import namespace="System.Collections" #>
<#@ import namespace="System.Collections.Generic" #>
<#
Byte maxDepth = 6;
#>
using System;
using System.Collections.Generic;
using System.Data.Objects;
using System.Data.Objects.DataClasses;
using System.Linq;
using System.Linq.Expressions;
namespace Magelia.WebStore.Data.Entities
{
public static partial class ObjectQueryExtensions
{
internal static String GetPath(params LambdaExpression[] lambdas)
{
return String.Join(".",
lambdas
.Select(l => l.GetPropertyName())
.ToArray()
);
}
<#
for(Byte i = 1; i < maxDepth; i++)
{
int combinationCount = (int)Math.Pow(2, i);
for (int j = 0; j < combinationCount; j++)
{
WriteMethod(i, j);
}
}
#>
}
}
<#+
void WriteMethod(Byte depth, int combinationValue)
{
String[] genericsType = new String[depth+1];
genericsType[0] = "TEntity";
int i = depth-1;
while (i >= 0)
{
String genericType = "T" + i.ToString();
if ((combinationValue & (1 << i)) != 0)
{
genericType = "IEnumerable<" + genericType + ">";
}
genericsType[i+1] = genericType;
i--;
}
String[] parametersType = new String[depth];
for(i = 0; i < depth; i++)
{
parametersType[i] = String.Format("Expression<Func<T{0}, {1}>> selector{2}", (i == 0) ? "Entity" : (i-1).ToString(), genericsType[i+1], i);
}
#>
public static ObjectQuery<TEntity> Include<TEntity, <#= String.Join(", ", Enumerable.Range(0, depth).Select(j => "T" + j.ToString()).ToArray()) #>>(this ObjectQuery<TEntity> objectSet, <#= String.Join(", ", parametersType) #>)
where TEntity : class
{
return objectSet.Include(GetPath(<#= String.Join(", ", Enumerable.Range(0, depth).Select(j => "selector" + j.ToString()).ToArray()) #>));
}
<#+
}
#>
Conclusion
Plus d'information sur mon blog : http://blogs.developpeur.org/cyril/archive/2012/01 /17/include-typ-et-entity-framework.aspx
Historique
- 17 janvier 2012 16:34:26 :
- ajout lien blog explicatif
Sources du même auteur
Sources de la même categorie
Commentaires et avis
Discussions en rapport avec ce code source dans le forum
include ... [ par maevacmoi ]
Hello !Je voudrais faire un site web (C#.net) avec des "include" comme en PHP, c'est à dire inclure une ou plusieures pages au sein d'une même page (s
Feuille de style et "include" [ par Kati83 ]
Bonjour à tous,Je voudrais vous poser 2 questions (pas très compliquées).La 1è : Si j'ai une page aspx qui contient beaucoup d'html, et pour le reste
Obtenir les include des fichiers xsd (xsd:include) d'un schema [ par franckypoune ]
Bonjour,J'ai un schema xsd qui contient des références d'autres schemas xsd qui contiennent des types plus généraux.Afin de ré
Question avec include [ par crahier ]
Bonjour,Je suis confronté à petit problème.
c# 3.0 et linq [ par mathmax ]
Bonjour,J'aimerais pouvoir utiliser c# 3.0 et linq. Pour celà, j'ai téléchargé et installé :.NET <span style="back
[LINQ] - créer une requete dynamique (pas en dur) [ par romagny13 ]
Bonjour,une question a propos de Linqvoilaj'ai essayé de me faire une pet
WCF + Linq : Transmettre une arborescence d'objets [ par sebmafate ]
Bonjour, J'utilise LinqToSql pour récupérer les informations de ma base de données et construire mes objets métiers. J'ai par exemple une class
[LINQ] Problème de requête erreur : (nullreferencexception) [ par teddyalbina ]
Salut !J'ai besoin de votre aide concernant une requete LINQ (c la première fois que j'utilise cette techno d'ailleur). Ma requete sert simplement a r
[LINQ] besoin d'un coup de main avec linq et ooxml [ par teddyalbina ]
Bonjour a tous j'ai besoin de votre aide, pour une requête linq. Je veux récupérer dans un document OOXML les métadonnées ( titre, auteur etc...). Ma
Question Linq très bête ... [ par oximoron ]
J'utilise linq est j'ai un soucis très bête:J'ai une table Dossier avec 3 champs id et chemin et acceson va dire rempli avec un champ qui à un 'guid',
|
Derniers Blogs
VMMAP EN MODE INSTRUMENTATION SUR SYSTèME 64BIT : ATTENTION à LA PLATEFORME CIBLE DU BUILD .NETVMMAP EN MODE INSTRUMENTATION SUR SYSTèME 64BIT : ATTENTION à LA PLATEFORME CIBLE DU BUILD .NET par coq
Si vous tentez d'utiliser VMMap en mode instrumentation* vous avez peut-être rencontré une erreur relativement peu explicite au lancement du profiling : Unable to launch application: "E:\Temp\ApplicationToProfile\ApplicationToProfile\bi...
Cliquez pour lire la suite de l'article par coq ETENDRE LE TEAM WEB ACCESS DE TFS 2012 - STEP 0ETENDRE LE TEAM WEB ACCESS DE TFS 2012 - STEP 0 par Philess
L'extensibilité du Team Web Access
Le Web Access (site d'équipe) de Team Foundation Server a été complètement réécrit dans la version 2012 avec pas moins de 400.000 lignes de JavaScript. Ce nouveau modèle a été pensé pour offrir de grandes...
Cliquez pour lire la suite de l'article par Philess SIMULER FACILEMENT L'ENVOI DE MAILSIMULER FACILEMENT L'ENVOI DE MAIL par JeremyJeanson
il m'a été demandé, à plusieurs reprises, comment je faisais pour simuler l'envoi de mail lors de mes démos de Workflow Foundation. Ma solution est plutôt simple : j'utilise la configuration par défaut du SmtpClient et j'oriente les mails vers un dossier ...
Cliquez pour lire la suite de l'article par JeremyJeanson VOTEZ POUR LE TOP 10 DES INFLUENCEURS SHAREPOINT FRANCOPHONES !VOTEZ POUR LE TOP 10 DES INFLUENCEURS SHAREPOINT FRANCOPHONES ! par Patrick Guimonet
Si ce n'est déjà fait (comme plus de 600 personnes déjà), il est encore temps de voter pour le concours TOP 10 des influenceurs SharePoint francophones ! Il est organisé par harmon.ie et accessible ici : http://harmon.ie/top-...
Cliquez pour lire la suite de l'article par Patrick Guimonet
Logiciels
Easy-Planning (4.5.0.11)EASY-PLANNING (4.5.0.11)Easy-Planning permet de créer des plannings sous la représentation de diagrammes et est adapté a... Cliquez pour télécharger Easy-Planning CVEasy (3.1.0.51)CVEASY (3.1.0.51)PHMSD-CVEasy est un logiciel d'aide à la rédaction de CV d'une simplicité déconcertante.
PHMSD-C... Cliquez pour télécharger CVEasy LettresFaciles 2011 (8.6.0.31)LETTRESFACILES 2011 (8.6.0.31)LettresFaciles est un logiciel facilitant la création et la rédaction de lettres types.
Son inte... Cliquez pour télécharger LettresFaciles 2011 sDEVIS-FACTURES vlPRO (8.4.2.62)SDEVIS-FACTURES VLPRO (8.4.2.62)sDEVIS-FACTURES vlPRO a été mis au point pour les particuliers, créateurs, entrepreneurs, artisa... Cliquez pour télécharger sDEVIS-FACTURES vlPRO Devis-Factures PHMSD (2.1.0.11)DEVIS-FACTURES PHMSD (2.1.0.11)Configuration minimale
Nécessite Windows™ 2000, XP, Windows 7, 8, Vista (Service Pack à... Cliquez pour télécharger Devis-Factures PHMSD
|