CA2214 : N'appelez pas de méthodes substituables dans les constructeurs
TypeName |
DoNotCallOverridableMethodsInConstructors |
CheckId |
CA2214 |
Catégorie |
Microsoft.Usage |
Modification avec rupture |
Modification sans rupture |
Cause
Le constructeur d'un type non-sealed (non scellé) appelle une méthode virtuelle définie dans sa classe.
Description de la règle
Lorsqu'une méthode virtuelle est appelée, le type réel qui exécute la méthode n'est pas sélectionné avant le moment de l'exécution. Lorsqu'un constructeur appelle une méthode virtuelle, il est possible que le constructeur de l'instance qui appelle la méthode n'ait pas été exécuté.
Comment corriger les violations
Pour corriger une violation de cette règle, n'appelez pas les méthodes virtuelles d'un type à partir des constructeurs de ce type.
Quand supprimer les avertissements
Ne supprimez aucun avertissement de cette règle. Le design du constructeur doit être refondu pour éliminer l'appel à la méthode virtuelle.
Exemple
L'exemple suivant montre l'effet de la violation de cette règle. L'application de test crée une instance de DerivedType, qui entraîne l'exécution du constructeur de sa classe de base (BadlyConstructedType). Le constructeur de BadlyConstructedType appelle la méthode virtuelle DoSomething de manière incorrecte. Alors que la sortie s'affiche, DerivedType.DoSomething() s'exécute, et ce avant l'exécution du constructeur de DerivedType.
Imports System
Namespace UsageLibrary
Public Class BadlyConstructedType
Protected initialized As String = "No"
Public Sub New()
Console.WriteLine("Calling base ctor.")
' Violates rule: DoNotCallOverridableMethodsInConstructors.
DoSomething()
End Sub 'New
' This will be overridden in the derived type.
Public Overridable Sub DoSomething()
Console.WriteLine("Base DoSomething")
End Sub 'DoSomething
End Class 'BadlyConstructedType
Public Class DerivedType
Inherits BadlyConstructedType
Public Sub New()
Console.WriteLine("Calling derived ctor.")
initialized = "Yes"
End Sub 'New
Public Overrides Sub DoSomething()
Console.WriteLine("Derived DoSomething is called - initialized ? {0}", initialized)
End Sub 'DoSomething
End Class 'DerivedType
Public Class TestBadlyConstructedType
Public Shared Sub Main()
Dim derivedInstance As New DerivedType()
End Sub 'Main
End Class
End Namespace
using System;
namespace UsageLibrary
{
public class BadlyConstructedType
{
protected string initialized = "No";
public BadlyConstructedType()
{
Console.WriteLine("Calling base ctor.");
// Violates rule: DoNotCallOverridableMethodsInConstructors.
DoSomething();
}
// This will be overridden in the derived type.
public virtual void DoSomething()
{
Console.WriteLine ("Base DoSomething");
}
}
public class DerivedType : BadlyConstructedType
{
public DerivedType ()
{
Console.WriteLine("Calling derived ctor.");
initialized = "Yes";
}
public override void DoSomething()
{
Console.WriteLine("Derived DoSomething is called - initialized ? {0}", initialized);
}
}
public class TestBadlyConstructedType
{
public static void Main()
{
DerivedType derivedInstance = new DerivedType();
}
}
}
Cet exemple génère la sortie suivante :