CA2214: No llamar a métodos reemplazables en constructores
Propiedad | Value |
---|---|
Identificador de la regla | CA2214 |
Título | No llamar a métodos reemplazables en constructores |
Categoría | Uso |
La corrección es problemática o no problemática | Poco problemático |
Habilitado de forma predeterminada en .NET 8 | No |
Causa
El constructor de un tipo no sellado llama a un método virtual definido en su clase.
Descripción de la regla
Cuando se llama a un método virtual, el tipo real que ejecuta el método no se selecciona hasta el tiempo de ejecución. Cuando un constructor llama a un método virtual, es posible que no se haya ejecutado el constructor de la instancia que invoca el método. Esto podría provocar errores o un comportamiento inesperado, si un método virtual invalidado se basa en la inicialización y en otra configuración del constructor.
Cómo corregir infracciones
Para corregir una infracción de esta regla, no llame a los métodos virtuales de un tipo desde dentro de los constructores del tipo.
Cuándo suprimir las advertencias
No suprima las advertencias de esta regla. Se debe rediseñar el constructor para eliminar la llamada al método virtual.
Ejemplo
En el ejemplo siguiente se muestra el efecto de infringir esta regla. La aplicación de prueba crea una instancia de DerivedType
, que provoca la ejecución del constructor de su clase base (BadlyConstructedType
). El constructor de BadlyConstructedType
llama al método virtual DoSomething
de forma incorrecta. Como muestra el resultado, DerivedType.DoSomething()
se ejecuta antes de que se ejecute el constructor de DerivedType
.
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 Main2214()
{
DerivedType derivedInstance = new DerivedType();
}
}
Imports System
Namespace ca2214
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 Main2214()
Dim derivedInstance As New DerivedType()
End Sub 'Main
End Class
End Namespace
Este ejemplo produce el siguiente resultado:
Calling base ctor.
Derived DoSomething is called - initialized ? No
Calling derived ctor.