Métodos de extensión (Guía de programación de C#

)
Visual Studio 2010 Otras versiones

y

Visual Studio 2008

Los métodos de extensión permiten "agregar" métodos a los tipos existentes sin necesidad de crear un nuevo tipo derivado y volver a compilar o sin necesidad de modificar el tipo original. Los métodos de extensión constituyen un tipo especial de método estático, pero se les llama como si se tratasen de métodos de instancia en el tipo extendido. En el caso del código de cliente escrito en C# y Visual Basic, no existe ninguna diferencia aparente entre llamar a un método de extensión y llamar a los métodos realmente definidos en un tipo. Los métodos de extensión más comunes son los operadores de consulta estándares LINQque agregan la funcionalidad de la consulta a los tipos System.Collections.Generic.IEnumerable<T> y los System.Collections.IEnumerable existentes. utilizar los operadores de consulta estándares, primero introdúzcalos en el ámbito con una directiva using System.Linq. Después, cualquier tipo que implemente IEnumerable<T> parecerá tener métodos de instancia como GroupBy, OrderBy, Average, etc. Puede ver estos métodos adicionales con la característica de finalización de instrucciones IntelliSense al escribir un "punto" después de una instancia de un tipo IEnumerable<T> como List<T> o Array. En el ejemplo siguiente se muestra cómo llamar al método de operador de consulta estándar OrderBy en una matriz de enteros. La expresión entre paréntesis es una expresión lambda. Muchos operadores de consulta estándar usan expresiones lambda como parámetros, pero no es obligatorio para los métodos de extensión. Para obtener más información, vea Expresiones lambda (Guía de programación de C#).

VB C# C++

Observe que se define dentro de una clase estática no anidada y no genérica: VB C# C++ F# JScript . 39. Los métodos de extensión sólo se incluyen en el ámbito cuando el espacio de nombres se importa explícitamente al código fuente con una directiva using.OrderBy(g => g). En el ejemplo siguiente se muestra un método de extensión definido para la clase System. } } } //Output: 10 15 21 26 39 45 Los métodos de extensión se definen como métodos estáticos pero se llaman utilizando la sintaxis de los métodos de instancia. 15. 26 }.String.Console.Write(i + " "). var result = ints. El primer parámetro especifica en qué tipo actúa el método y va precedido del modificador this. foreach (var i in result) { System.F# JScript class ExtensionMethods2 { static void Main() { int[] ints = { 10. 45. 21.

Por lo tanto. '. Por ejemplo. Sin embargo.RemoveEmptyEntries).namespace ExtensionMethods { public static class MyExtensions { public static int WordCount(this String str) { return str. También se puede llamar desde una aplicación con esta sintaxis: string s = "Hello Extension Methods".WordCount().'. simplemente agregue una directiva using para el espacio de nombres en el que se definen los métodos.Length. Para habilitar los métodos de extensión para un tipo determinado. para utilizar los operadores de consulta estándar. los métodos de extensión no pueden tener acceso a las variables privadas del tipo que extienden. Dado que se llama a los métodos de extensión con sintaxis de método de instancia. En el código.Split(new char[] { ' '. no se infringe realmente el principio de encapsulación. agregue esta directiva using a su código: . el lenguaje intermedio (IL) generado por el compilador convierte el código en una llamada en el método estático. De hecho. } } } El método de extensión WordCount se puede incluir en el ámbito con esta directiva using: using ExtensionMethods. el método de extensión se invoca con sintaxis de método de instancia. probablemente llamará a métodos de extensión con mayor frecuencia de lo que implementará los suyos propios. StringSplitOptions. '?' }. vea Cómo: Implementar e invocar un método de extensión personalizado (Guía de programación de C#). no se requieren conocimientos especiales para usarlos desde el código de cliente. En general. Para obtener más información. int i = s.

using System. busca primero una coincidencia entre los métodos de instancia del tipo. Nunca se llamará a un método de extensión que tenga el mismo nombre y firma que un método de interfaz o clase. La clase estática Extensions contiene métodos de extensión definidos para cualquier tipo que implementa IMyInterface.Linq.) Observará que los operadores de consulta estándar ahora aparecen en IntelliSense como métodos adicionales disponibles para la mayoría de los tipos IEnumerable<T>. siguen estando disponibles. Las clases A. Cuando el compilador encuentra una invocación de método. Nota Aunque no aparezcan operadores de consulta estándar en IntelliSense para String. si un tipo tiene un método denominado Process(int i) y hay un método de extensión con la misma firma. pero no para invalidarlas. los métodos de extensión siempre tienen menos prioridad que los métodos de instancia definidos en el propio tipo.Core. el compilador siempre establecerá el enlace con el método de instancia. . en caso de que exista. Si el compilador no encuentra un método de instancia con una firma coincidente. Enlazar métodos de extensión en tiempo de compilación Puede utilizar métodos de extensión para extender una clase o una interfaz. En otras palabras.dll. En el ejemplo siguiente se muestra cómo el compilador determina con qué método de extensión o método de instancia establecerá el enlace. (Puede que tenga que agregar también una referencia a System. Nunca se llama al método de extensión MethodB porque su nombre y firma coinciden exactamente con métodos ya implementados por las clases. B y C implementan la interfaz. establecerá el enlace con un método de extensión coincidente. Si no la encuentra. buscará cualquier método de extensión definido para el tipo y establecerá el enlace con el primer método de extensión que encuentre. Ejemplo En el ejemplo siguiente se muestran las reglas que sigue el compilador de C# para determinar si debe establecer un enlace entre una llamada a método y un método de instancia del tipo o un método de extensión. En tiempo de compilación.

void MethodB().MethodA(this IMyInterface myInterface. Extensions System. namespace DefineIMyInterface { using System. because each .MethodA(this IMyInterface myInterface. } // This method is never called in ExtensionMethodsDemo1. } } // Define namespace { using using extension methods for IMyInterface. string s) { Console. int i) { Console. public interface IMyInterface { // Any class that implements IMyInterface must define a method // that matches the following signature.VB C# C++ F# JScript // Define an interface named IMyInterface.WriteLine ("Extension. // The following extension methods can be accessed by instances of any // class that implements IMyInterface. int i)").WriteLine ("Extension. string s)"). DefineIMyInterface. } public static void MethodA(this IMyInterface myInterface. public static class Extension { public static void MethodA(this IMyInterface myInterface.

B. C c = new C(). } } class B : IMyInterface { public void MethodB() { Console. B b = new B(). // For a. and C implements a method named MethodB // that has a matching signature. using Extensions.WriteLine("B.WriteLine ("Extension. } } class C : IMyInterface { public void MethodB() { Console.// of the three classes A. } public void MethodA(object obj) { Console. using DefineIMyInterface.MethodB()").MethodA with an int argument // -.MethodB()"). and class C. namespace ExtensionMethodsDemo1 { using System. class B. and c.MethodB with no argument.MethodB()"). class A : IMyInterface { public void MethodB() { Console. } public void MethodA(int i) { Console. A a = new A(). // A contains no MethodA.MethodA(int i)").MethodA with a string argument // -. } } class ExtMethodDemo { static void Main(string[] args) { // Declare an instance of class A.WriteLine("A. } } } // Define three classes that implement IMyInterface.WriteLine("B.MethodB(this IMyInterface myInterface)"). call the following methods: // -.WriteLine("C. public static void MethodB(this IMyInterface myInterface) { Console.MethodA(object obj)").WriteLine("C. b. so each call to MethodA resolves to . and then use them to test // the extension methods.

debe hacerlo creando un nuevo tipo derivado del existente.MethodA(object.MethodB() Instrucciones generales En general.MethodA("hello").MethodB().MethodA(object.MethodA(object) // C.MethodA(1).MethodA(this IMyInterface myInterface.MethodA(int) b.MethodA(1). but // class Extension does. corre el riesgo de que un cambio en la implementación del tipo interrumpa su método de extensión. int) a. Para obtener más información. c. } } } /* Output: Extension. recuerde los dos puntos siguientes: . string s) A. // C contains an instance method that matches each of the following // method calls.MethodA("hello"). // Extension. sólo cuando sea necesario. b.MethodA(this IMyInterface myInterface.MethodB(). Siempre que sea posible.MethodB(). // B. a. Al utilizar un método de extensión para extender un tipo cuyo código fuente no se puede cambiar. vea Herencia (Guía de programación de C#). a.MethodA(object obj) C.MethodA("hello").MethodB() Extension.MethodA(this IMyInterface myInterface.MethodB() // B has methods that match the signatures of the following // nethod calls. int i) Extension.MethodA(object.MethodB() // B has no matching method for the following call. c. // Extension. // Extension.// the extension method that has a matching signature. recomendamos que implemente métodos de extensión en contadas ocasiones.MethodB() */ // C.MethodA(1). string s) C.MethodA(object obj) C. string) b. string) // A has a method that matches the signature of the following call // to MethodB. // A.MethodA(int i) B. // B. c. cuando el código de cliente debe extender un tipo existente. Si implementa métodos de extensión para un tipo determinado.MethodB() B.MethodA(object) // C.

Los métodos de extensión se incluyen en el ámbito en el nivel de espacio de nombres. Vea también Referencia Expresiones lambda (Guía de programación de C#) Conceptos Guía de programación de C# Información general sobre operadores de consulta estándar Otros recursos Conversion rules for Instance parameters and their impact Extension methods Interoperability between languages Extension methods and Curried Delegates Extension method Binding and Error reporting Historial de cambios . los incluirá a todos en el ámbito. Por ejemplo. si tiene varias clases estáticas que contienen métodos de extensión en un espacio de nombres único denominado Extensions. la directiva using Extensions.y y Nunca se llamará a un método de extensión si tiene la misma firma que un método definido en el tipo.

Sign up to vote on this title
UsefulNot useful