gpt4 book ai didi

c# - 为什么在 C# 中调试泛型时不能执行 (A)o,但 (o as A) 可以?

转载 作者:行者123 更新时间:2023-12-03 22:58:10 26 4
gpt4 key购买 nike

假设我有一个泛型类和一些其他代码,如下所示:

public class S<T>
{
public void Method1(T o)
{
}
}

public class A
{
public string Name { get; set; }
}

public static void Test()
{
var s = new S<A>();
var a = new A() { Name = "John" };
s.Method1(a);
}

调试s.Method1(a)并步入S<T>.Method1(T o) ,我发现我无法评估强制转换操作 - (GenericRuntimeCastWhenDebugging.A)oDEBUG CONSOLE使用 VSCode 或在 Immediate 中时使用VS2017时,窗口出现错误:

(GenericRuntimeCastWhenDebugging.A)o

error CS0030: Cannot convert type 'T' to 'GenericRuntimeCastWhenDebugging.A'

但是(o as GenericRuntimeCastWhenDebugging.A)在这种情况下工作正常并显示结果:

(o as GenericRuntimeCastWhenDebugging.A)

{GenericRuntimeCastWhenDebugging.A}

我希望在这种情况下(GenericRuntimeCastWhenDebugging.A)o应该也能正常工作!有人知道为什么吗?

编辑:我明白T o不允许转换为 A通过(A)o在我给出的示例中的编译时,以及 (A)o 之间的区别和(o as A)在 C# 中。我对在调试期间第一种强制转换方式感兴趣 (GenericRuntimeCastWhenDebugging.A)o 引发错误,但第二种方式转换 (o as GenericRuntimeCastWhenDebugging.A) 转换成功并返回正确的A对象回来。因此在调试期间,运行 (o as GenericRuntimeCastWhenDebugging.A).NameDEBUG CONSOLE也很好并且返回值:John .

(o as GenericRuntimeCastWhenDebugging.A).Name

"John"

此外,事实证明,在编译时我实际上可以编写以下代码,它运行良好并且 n2设置为John 。不需要像装箱/拆箱这样的技巧。

public class S<T>
{
public void Method1(T o)
{
// var n1 = ((A)o).Name; Cannot compile
var n2 = (o as A).Name;
}
}

最佳答案

问题是没有为T -> A定义显式转换。 。这是因为您的泛型类型没有任何限制。使用as之所以有效,是因为它明确询问是否 o可以强制转换,而不是明确地 o可以施放。

您可以限制TAA 的子类/实现与以下内容。

public class S<T> where T : A
{
public void Method1(T o)
{
var a = (A)o;
}
}

现在这是有效的,因为编译器确信这不会生成运行时异常。

或者,你可以写:

public class S<T>
{
public void Method1(T o)
{
var a = (A)(object)o;
}
}

它将编译,因为任何1都可以转换(或可装箱)为 object ,您可以转换(或取消装箱) object任何类型。但请注意,这可能是一个坏主意,因为没有什么可以阻止某人写 new S<int>().Method1(5)

1 任何东西在编译时都是(不可)装箱/(不可)可转换的 - 但这并不能保护您免受运行时异常的影响。

关于c# - 为什么在 C# 中调试泛型时不能执行 (A)o,但 (o as A) 可以?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43993481/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com