Maison > Article > développement back-end > Spécification C# 2.0 (Génériques 4)
Connecté à Generic Three
20.6 Méthodes génériques
Les méthodes génériques sont des méthodes liées à un type spécifique. En plus des paramètres réguliers, une méthode générique nomme également un ensemble de paramètres de type qui doivent être fournis lors de l'utilisation de la méthode. Les méthodes génériques peuvent être déclarées dans une déclaration de classe, de structure ou d'interface, et elles peuvent elles-mêmes être génériques ou non génériques. Si une méthode générique est déclarée dans une déclaration de type générique, le corps de la méthode peut référencer les paramètres de type de la méthode et contenir les paramètres de type déclarés.
class-member-declaration:(类成员声明:) … generic-method-declaration (泛型方法声明) struct-member-declaration:(结构成员声明:) … generic-method-declaration(泛型方法声明) interface-member-declaration:(接口成员声明:) … interface-generic-method-declaration(接口泛型方法声明) 泛型方法的声明可通过在方法的名字之后放置类型参数列表而实现。 generic-method-declaration:(泛型方法声明:) generic-method-header method-body(泛型方法头 方法体) generic-method-header:(泛型方法头:)
attributs opt méthode-modificateurs opt return-type membre-name type-parameter-list (formal-parameter-list opt ) type-parameter-constraints-clause opt
(Attribut facultatif modificateur de méthode facultatif type de retour nom du membre liste de paramètres de type (liste de paramètres formels facultative) déclaration de contrainte de paramètre de type facultative)
interface-generic-method-declaration : (déclaration de méthode générique d'interface :)
attributs opt new opt identifiant de type de retour type-parameter-list
(formal-parameter-list opt) type-parameter-constraints-clauses opt ;
(attributs optionnel nouvel identifiant de type de retour facultatif Liste de paramètres de type (liste de paramètres formelle est facultatif) L'instruction de contrainte de paramètre de type est facultative ;
La liste de paramètres de type et l'instruction de contrainte de paramètre de type ont la même syntaxe et la même fonction que la portée du paramètre de type déclarée par la liste de paramètres de type, elle peut être utilisée pour. les instructions de formulaire comprenant les valeurs de retour, les corps de méthode et les contraintes des paramètres de type, mais pas les attributs
Les noms des paramètres de type d'une méthode ne peuvent pas être les mêmes que les noms des paramètres réguliers de la même méthode. 🎜>
L'exemple suivant trouve le premier élément d'un tableau s'il satisfait au délégué de test donné. Les délégués génériques sont décrits dans le §20.4. Tous les autres modificateurs sont valides sur les méthodes génériques. Signatures de méthodes génériques
À des fins de comparaison de signatures, toutes les contraintes de paramètre de type seront ignorées, tout comme le nom du paramètre de type, mais le nombre de paramètres de type est également ignoré, tout comme la position de l'élément du type. paramètre de gauche à droite. L'exemple suivant montre l'effet de cette règle. La signature de méthode ci-dessous est régie par une règle similaire à celle utilisée dans les déclarations de type générique (§20.1.8). le même nom et le même nombre d'arguments de type ne peuvent pas avoir de types de paramètres qui entourent la liste des arguments de type, lorsqu'ils produisent deux méthodes avec la même signature dans le même ordre. Lorsqu'ils sont appliqués à deux méthodes dans le même ordre, par exemple, <.>
public delegate bool Test<T>(T item); public class Finder { public static T Find<T>(T[] items , Test<T> test){ foreach(T item in items) { if(test(item)) return item; } throw new InvalidOperationException(“Item not found”); } }
20.6.2 Méthode générique virtuelle
Les méthodes génériques peuvent être déclarées à l'aide de modificateurs abstraits, virtuels et de remplacement. Lors de la mise en correspondance des remplacements de méthode et des implémentations d'interface, des signatures correspondant aux règles décrites au §20.6.1 seront utilisées. Lorsqu'une méthode générique remplace une méthode générique déclarée dans une classe de base, ou implémente une méthode dans une interface de base, les contraintes données pour chaque paramètre de type doivent être les mêmes dans les deux déclarations, où les paramètres de type de méthode seront Les positions d'origine sont identifiées à partir de de gauche à droite. La réécriture de la
classe abstraite Baseclass A{} class B {} interface IX { T F1<T>(T[] a , int i); //错误,因为返回类型和类型参数名字无关紧要, void F1<U>(U[] a ,int i); //这两个声明都有相同的签名 void F2<T><int x>; //OK,类型参数的数量是签名的一部分 void F2(int x); void F3<T>(T t) where T: A // 错误,约束不在签名考虑之列 void F3<T>(T t) where T:B: }
20.6.3 Appel de méthodes génériques
泛型方法调用可以显式指定类型实参列表,或者省略类型实参列表,而依靠类型推断来确定类型实参。方法调用的确切编译时处理,包括泛型方法调用,在§20.9.5中进行了描述。当泛型方法不使用类型参数列表调用时,类型推断将按§20.6.4中所描述的进行。
下面的例子展示在类型推断和类型实参替代参数列表后,重载决策是如何发生的。
class Test { static void F<T>(int x , T y) { Console.WriteLine(“One”); } static void F<T>(T x , long y) { Console.WriteLine(“two”); } static void Main() { F<int>(5,324); //ok,打印“one” F<byte>(5,324); //ok, 打印“two” F<double>(5,324); //错误,模糊的 F(5,324); //ok,打印“one” F(5,324L); //错误,模糊的 } }
20.6.4类型实参推断
当不指定类型实参而调用泛型方法时,类型推断(type inference)处理将试图为调用推断类型实参。类型推断的存在可以让调用泛型方法时,采用更方便的语法,并且可以避免程序员指定冗余的类型信息。例如,给定方法声明
class Util { static Random rand = new Random(); static public T Choose<T>(T first , T second) { return (rand.Next(2) == 0)?first:second } }
不显式指定类型实参而调用 Choose方法也是可以的。
int i = Util.Choose(5,123); //调用Choose<int> string s = Util.Choose(“foo”,”bar”);//调用Choose<string>
通过类型推断,类型实参int和string 将由方法的实参确定。
类型推断作为方法调用(§20.9.5)编译时处理的一部分而发生,并且在调用的重载决策之前发生。当在一个方法调用中指定特定的方法组时,类型实参不会作为方法调用的一部分而指定,类型推断将被应用到方法组中的每个泛型方法。如果类型推断成功,被推断的类型实参将被用于确定后续重载决策的实参类型。如果重载决策选择将要调用的泛型方法,被推断的类型实参将被用作调用的实际类型实参。如果特定方法类型推断失败,这个方法将不参与重载决策。类型推断失败自身将不会产生编译时错误。但当重载决策没能找到适用的方法时,将会导致编译时错误。
如果所提供的实参个数与方法的参数个数不同,推断将立刻失败。否则,类型推断将为提供给方法的每个正式实参独立地发生。假定这个实参的类型为A,对应参数为类型P。类型推断将按下列步骤关联类型A和P而。
如果以下任何一条成立,将不能从实参推断任何东西(但类型推断是成功的)
- P和方法的任何类型参数无关[1]
- 实参是null字符
- 实参是一个匿名方法
- 实参是一个方法组
如果P是一个数组类型,A是一个同秩(rank)的数组类型,那么使用A和P的元素类型相应地替换A和P,并重复这个步骤。
如果P是一个数组类型,而A 是一个不同秩的数组类型,那么泛型方法的类型推断失败。
如果P是方法的类型参数,那么对于这个实参的类型推断成功,并且A是那个类型实参所推断的类型。
否则,P必须是一个构造类型。如果对于出现在P中的每个方法类型参数MX ,恰好可以确定一个类型TX,使用每个TX替换每个MX ,这将产生一个类型,对于这个类型,A可以通过标准的隐式转换而被转换,那么对于这个实参的类型推断成功,对于每个MX,TX 就是推断的类型。方法类型参数约束(如果有的话),因为类型推断的原因将会被忽略。如果对于一个给定的MX 没有TX存在或者多于一个TX存在 ,那么泛型方法的类型推断将会失败(多于一个TX 存在的情形只可能发生在,P是一个泛型接口类型,并且A实现了接口的多个构造版本)。
如果所有的方法实参都通过先前的算法进行了处理,那么从实参而来的所有推断都将被汇聚。这组推断必须有如下的属性。
方法的每个类型参数必须有一个为其推断的类型实参。简而言之,这组推断必须是完整的([/b]complete[/b])[/b];
如果类型参数出现多于一次,那么对那个类型参数的所有的推断都必须推断相同的类型实参。简而言之,这组接口必须是一致的([/b]consistent[/b])[/b]。
如果能够找到一组完整而一致的推断类型实参,那么对于一个给定的泛型方法和实参列表,类型推断就可以说是成功的。
如果泛型方法使用参数数组(§10.5.1.4)声明,那么类型推断针对方法将以其通常的方式执行。如果类型推断成功,结果方法是可用的,那么方法将以其通常形式对于重载决策是可行的。否则,类型推断将针对方法的扩展形式(§7.4.2.1)执行。
[1]也就是P并不是方法的类型参数列表所列类型之一。
以上就是C# 2.0 Specification (泛型四)的内容,更多相关内容请关注PHP中文网(www.php.cn)!