tag(s) Tags: C#, Linq
lu 2411 fois
6 pages
Frédéric Mélantois
Comprendre les bases de Linq to objects
La meilleure façon de comprendre le fonctionnement de Linq est sans doute de réaliser soi-même un petit développement similaire en C# 2.0 et de le faire évoluer sous C# 3.0 en intégrant chaque nouveauté du langage.
Par Frédéric Mélantois publié le 21/10/2007 à 22:53
 
Reprenons le code à notre disposition pour le faire évoluer sous Visual Studio 2008 et C# 3.0. Notre classe statique « ComplementEnumerable » est la suivante :

public delegate TResult Func<TSource, TResult>(TSource source);

 

public static class ComplementEnumerable

{

  public static IEnumerable<TResult> Select<TSource, TResult>(IEnumerable<TSource> source, Func<TSource, TResult> selector)

    {

        foreach (TSource s in source)

 

            yield return selector.Invoke(s);

    }

 

    public static float Average(IEnumerable<float> source)

    {

        double num1 = 0;

        long num2 = 0;

        foreach (float num3 in source)

        {

            num1 += num3;

            num2++;

        }

        return (float)(num1 / (double)num2);

    }

 

    public static float Average<TSource>(IEnumerable<TSource> Source, Func<TSource, float> selector)

    {

        return ComplementEnumerable.Average(ComplementEnumerable.Select<TSource, float>(Source, selector));

    }

 

    public static IEnumerable<TSource> Where<TSource>(IEnumerable<TSource> source, Func<TSource, bool> selector)

    {

        foreach (TSource s in source)

            if (selector.Invoke(s))

                yield return s;

    }

}

Pour toute création de fichiers de code C# dans Visual Studio, veillez bien à mettre en commentaire la ligne « using System.Linq ; » afin que nous puissions mener notre démonstration à bien sans l'utilisation du Framework Linq mais uniquement avec les nouveautés du compilateur C# 3.0.
Une très grande avancée du langage C# dans sa version 3.0 est la déduction de type. Elle se généralise grâce à l'évolution du compilateur C# 3.0. L'expression suivante :

IEnumerable<IEnumerable<NoteENT>> t = ComplementEnumerable.Select<EleveENT, IEnumerable<NoteENT>>(l,

                            delegate(EleveENT c) { return c.Notes; });

devient :

var t = ComplementEnumerable.Select<EleveENT, IEnumerable<NoteENT>>(l, delegate(EleveENT c) { return c.Notes; });

Le mot clé "var" nous permet de réaliser une économie syntaxique confortable puisque la partie gauche d'une affectation n'est plus nécessaire. Il ne s'agit aucunement d'un « variant », comme on le connaît dans d'autres langages.
Comme je l'ai montré dans l'article « C# 3.0 Beta, déclarations et initialisations simplifiées, regardons sous le capot ! », il n'existe pas de différence au niveau de l'IL (Intermediate Langage), que le type soit renseigné ou que la syntaxe soit simplifiée par le mot clé « var ».
Le typage implicite offert par le compilateur C# 3.0 mais aussi par l'intellisence de Visual Studio 2008 va beaucoup plus loin que la prise en compte du nouveau mot clé « var ». Il permet encore de simplifier la syntaxe de l'expression précédente. En effet, les parties génériques de cette dernière peuvent être déduites par le compilateur. De sorte que l'on peut écrire :

var u = ComplementEnumerable.Select(l, delegate(EleveENT c) { return c.Notes; });

Par exemple si nous prenons l'expression compliquée suivante :

IEnumerable<IEnumerable<float>> t = ComplementEnumerable.Select<EleveENT, IEnumerable<float>>(l, delegate(EleveENT c)

            {

                return ComplementEnumerable.Select<NoteENT, float>(ComplementEnumerable.Where<NoteENT>(c.Notes,

                    delegate(NoteENT n)

                        { return n.Date >= new DateTime(2007,4,1); }), delegate(NoteENT n)

                            { return n.Note; }

                    );

                   });

Celle-ci se trouve très allégée :

var r = ComplementEnumerable.Select(l, delegate(EleveENT c)

    {

        return ComplementEnumerable.Select(ComplementEnumerable.Where(c.Notes, delegate(NoteENT n)

        { return n.Date >= new DateTime(2007,4,1); }), delegate(NoteENT n)

        { return n.Note; }

    );

    });

La déduction de type par le compilateur apporte donc un véritable confort du point de vue de la syntaxe dans l'utilisation de types à base de génériques. L'environnement de Visual Studio 2008 possède une intellisence prenant parfaitement en compte cette simplification.
A ce stade de notre exposé, il serait intéressant de comparer l'IL généré par le compilateur C# 2.0 concernant le code présenté dans le premier chapitre et l'IL généré par le compilateur C# 3.0 dans ce chapitre. Pour cela, vous pouvez utiliser l'utilitaire « ILDASM » fourni par le Framework ou bien l'utilitaire Reflector dont vous limiterez l'affichage uniquement au code IL. Vous aurez la surprise de constater que le code IL est identique dans les deux cas. C'est donc que les nouveautés C# 3.0 présentées jusqu'à présent et notamment les simplifications d'écriture n'ont pas induit de changement en ce qui concerne l'IL. Pour vous en persuader complètement, je vous invite à recompiler le code IL issu des sources C# 3.0, en changeant les assemblies de références vers celles du framework 2.0. Utilisez pour cela l'utilitaire « ILASM » du SDK 2.0. La compilation se passe correctement.
Si par les déductions de types, la syntaxe a été très fortement simplifiée, les méthodes anonymes restent encore assez lourdes à écrire. A partir de cette constatation, l'équipe en charge de C# 3.0 a donc imaginé une simplification d'écriture en expression Lambda. C'est ainsi que :

delegate(NoteENT n) { return n.Note; }

devient

(NoteENT f) => f.Note

A partir de cet allégement, on peut donc commencer à modifier la syntaxe de l'expression suivante :

var r = ComplementEnumerable.Select(l, delegate(EleveENT c)

    {

        return ComplementEnumerable.Select(ComplementEnumerable.Where(c.Notes, delegate(NoteENT n)

        { return n.Date >= new DateTime(2007,4,1); }), (NoteENT f) => f.Note);

 

    });

en :

var r = ComplementEnumerable.Select(l, c =>

                ComplementEnumerable.Select(ComplementEnumerable.Where(c.Notes, n =>

                n.Date >= new DateTime(2007,4,1)),f => f.Note)

            );

Ce code montre de façon évidente que la déduction de type par le compilateur combinée à l'utilisation des expressions Lambda, oeuvre vers une grande simplification d'écriture.
Une nouvelle fois, j'invite le lecteur à comparer l'IL généré dans le cadre du premier chapitre et l'IL généré par le compilateur C# 3.0 après avoir ajouté les expressions lambda à notre code. Vous constaterez encore qu'il n'y a pas de différence ! C'est donc que l'emploi d'une méthode anonyme ou d'une expression Lambda produit le même code IL.
Bien que la déduction de types par le compilateur et l'emploi d'expressions Lambda nous fassent bénéficier d'une grosse simplification de l'écriture C#, la réduction du code avec la nouvelle version de C# 3.0 n'est pas encore totalement complète. Nous pouvons encore réduire l'écriture ! C'est ce que nous allons voir au prochain chapitre.

 Commentaire - Comprendre les bases de Linq to objects 

Discussion démarée par Matthieu Mezil le 22/10/2007 à 11:37 , 1 commentaire(s).

 Dernières Publications      

Windows Media Center et WCF : développez votre maison intelligente
  Le développement d'applications pour Windows Media Center est facilité avec l'arrivée du SDK 5.3. Même si l'on sent un modèle objet bien lourd derrière, il devient plus facile d'exposer les fonctionnalités de WMC sous la forme de services WCF.
par Frédéric Colin posté le 23/06/2008 à 08:04, lu 295 fois, #0
Notions avancées avec Biztalk Server 2006 R2
  Utilisation des notions d'interchange, corrélation et convoi avec BizTalk Server 2006 R2
par Kader Yildirim posté le 09/06/2008 à 08:04, lu 288 fois, #0
Lucene Persistence Engine pour Evaluant Universal Storage Services
  Suite à l'article de Laurent Kempé, voici un moteur de stockage pour EUSS permettant l'indexation d'entités métier avec Lucene.
par Nicolas Penin posté le 01/06/2008 à 23:38, lu 510 fois, #1
Tags: C#, Linq
XMLA Trivia : Découverte du XMLA
  Le XMLA (XML for Analysis) est un langage normalisé par plusieurs éditeurs BI pour simplifier l'accès aux données aux cubes et aux métadonnées des bases multidimensionnelles.
par Renaud Harduin posté le 25/05/2008 à 11:57, lu 523 fois, #0
Exploiter les données CSV via Linq en toute simplicité
  A partir du requêteur dynamique fourni en exemple avec Visual Studio 2008, nous allons essayer de remplir les propriétés d'un ensemble d'objets à partir des données d'un fichier CSV. Nous enrichirons aussi le parseur de nos propres fonctions.
par Frédéric Mélantois posté le 17/05/2008 à 11:41, lu 1713 fois, #0
Comment manipuler simplement le contenu d'un fichier WordML ?
  Manipulations autour du format WordML
par Fabien Reinle posté le 14/05/2008 à 23:55, lu 781 fois, #0
Polymorphisme et contrats de données WCF
  WCF aborde les types polymorphes du point de vue de la sérialisation. En effet, la connaissance du type réel potentiel est rendue nécessaire dès la description du contrat de données. Une fois n'est pas coutume, j'ai réalisé l'exemple en VB.NET.
par Frédéric Colin posté le 14/05/2008 à 08:40, lu 1881 fois, #2
A la découverte de BizTalk Server 2006 3/3
  Développer un assembleur pour BizTalk Server 2006 R2
par Kader Yildirim posté le 06/05/2008 à 13:20, lu 511 fois, #0

 Dernières Actualités      

Deep Earth – Une belle utilisation de Virtual Earth et de Silverlight Deep Zoom
  Ce projet très intéressant est disponible sur Codeplex et vous pouvez voir une démo sur la page suivante . Bien entendu comme touts les projets sur Codeplex vous avez accès aux sources....
Tags: Silverlight
Sortie de JetBrains ReSharper 4.0 en version finale, l’outil ultime pour Visual Studio
  Après plusieurs mois de Early Access Program (EAP) , JetBrains met enfin à disposition la version finale de son outil ReSharper 4.0 . Cette nouvelle version est disponible pour Visual Studio 2005 &...
Tags: Visual Studio 2008, Visual Studio 2005, Outils
BoutDuTunnel v1.4
  BoutDuTunnel est un petit logiciel de tunneling réseau écrit en C#. Il permet par exemple d’accéder aux services ftp/smtp/pop/telnet/nntp/… sur des réseaux qui n’autorisent...
BI Framework & sample sur CodePlex
  Après plusieurs demandes, je me suis décidé à déposer l'ensemble des sources et du BI Framework MS proposés dans mes articles sur codeplex : http://www.codeplex.com/BILAB Je le mettrais à jour au fil des...
Injection de code et API de profiling .NET
  Si vous êtes intéressés par la sécurité du Framework, par le reverse engineering et la manipulation/injection de code .NET et les packers, alors jetez un coup d’œil...
NDepend pour l'analyse statique de code .NET
  Pour ceux qui ne connaissent pas NDepend , il s’agit d’un outil d’analyse statique de code .NET qui permet de remonter des informations à toute une équipe de développement. NDepend aide à travailler sur...
Tags: Outils