- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
通常,将结构 S
视为接口(interface) I
会触发结构的自动装箱,如果经常这样做会对性能产生影响。但是,如果我编写一个采用类型参数 T : I
的泛型方法并使用 S
调用它,那么编译器是否会省略装箱,因为它知道类型 S
并没有使用接口(interface)?
这段代码表明了我的观点:
interface I{
void foo();
}
struct S : I {
public void foo() { /* do something */ }
}
class Y {
void doFoo(I i){
i.foo();
}
void doFooGeneric<T>(T t) where T : I {
t.foo(); // <--- Will an S be boxed here??
}
public static void Main(string[] args){
S x;
doFoo(x); // x is boxed
doFooGeneric(x); // x is not boxed, at least not here, right?
}
}
doFoo
方法在 I
类型的对象上调用 foo()
,所以一旦我们用 S< 调用它
,S
将被装箱。 doFooGeneric
方法做同样的事情。然而,一旦我们用 S
调用它,可能不需要自动装箱,因为运行时知道如何在 S
上调用 foo()
.但这会完成吗?或者运行时会盲目地将 S
装箱到 I
以调用接口(interface)方法?
最佳答案
void doFooGeneric<T>(T t) where T : I {
t.foo(); // <--- Will an S be boxed here??
}
那里将避免拳击!
结构类型 S
是密封的。对于类型参数的值类型版本 T
你的方法doFooGeneric
上面,C# 编译器给出了直接调用相关结构成员的代码,没有装箱。
这很酷。
有关一些技术细节,请参阅 Sameer 的回答。
好的,所以我想出了一个这样的例子。如果有人有一些更好的例子,我会对更好的例子感兴趣:
using System;
using System.Collections.Generic;
namespace AvoidBoxing
{
static class Program
{
static void Main()
{
var myStruct = new List<int> { 10, 20, 30, }.GetEnumerator();
myStruct.MoveNext(); // moves to '10' in list
//
// UNCOMMENT ONLY *ONE* OF THESE CALLS:
//
//UseMyStruct(ref myStruct);
//UseMyStructAndBox(ref myStruct);
Console.WriteLine("After call, current is now: " + myStruct.Current); // 10 or 20?
}
static void UseMyStruct<T>(ref T myStruct) where T : IEnumerator<int>
{
myStruct.MoveNext();
}
static void UseMyStructAndBox<T>(ref T myStruct)
{
((IEnumerator<int>)myStruct).MoveNext();
}
}
}
这里是 myStruct
的类型是一个可变值类型,它保留对 List<>
的引用, 并且还持有记住 List<>
中的索引的“计数器”我们已经达到了现在。
我不得不使用 ref
,否则当传递给任一方法时,值类型将按值复制!
当我取消对 UseMyStruct
的调用的注释时(仅),此方法将我们的值类型中的“计数器”向前移动一个位置。如果它在值类型的盒装副本中这样做,我们将不会在结构的原始实例中看到它。
要查看装箱有何不同,请尝试调用 UseMyStructAndBox
相反(再次评论 UseMyStruct
)。它在类型转换上创建了一个盒子,并且 MoveNext
发生在副本上。所以输出是不同的!
致那些对 ref
不满意(或感到困惑)的人, 就写出 Current
而不是从方法内部。然后我们可以去掉ref
.示例:
static void F<T>(T t) where T : IEnumerator<int>
{
t.MoveNext(); // OK, not boxed
Console.WriteLine(t.Current);
}
static void G<T>(T t) where T : IEnumerator<int>
{
((IEnumerator<int>)t).MoveNext(); // We said "Box!", it will box; 'Move' happens to a copy
Console.WriteLine(t.Current);
}
关于c# - 在这种情况下,C# 泛型会阻止结构的自动装箱吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31457682/
我是 Java 新手,这是我的代码, if( a.name == b.name && a.displayname == b.displayname && a.linknam
在下面的场景中,我有一个 bool 值。根据结果,我调用完全相同的函数,唯一的区别是参数的数量。 var myBoolean = ... if (myBoolean) { retrieve
我是一名研究 C++ 的 C 开发人员: 我是否正确理解如果我抛出异常然后堆栈将展开直到找到第一个异常处理程序?是否可以在不展开的情况下在任何 throw 上打开调试器(即不离开声明它的范围或任何更高
在修复庞大代码库中的错误时,我观察到一个奇怪的情况,其中引用的动态类型从原始 Derived 类型更改为 Base 类型!我提供了最少的代码来解释问题: struct Base { // some
我正在尝试用 C# 扩展给定的代码,但由于缺乏编程经验,我有点陷入困境。 使用 Visual Studio 社区,我尝试通过控制台读出 CPU 核心温度。该代码使用开关/外壳来查找传感器的特定名称(即
这可能是一个哲学问题。 假设您正在向页面发出 AJAX 请求(这是使用 Prototype): new Ajax.Request('target.asp', { method:"post", pa
我有以下 HTML 代码,我无法在所有浏览器中正常工作: 我试图在移动到
我对 Swift 很陌生。我如何从 addPin 函数中检索注释并能够在我的 addLocation 操作 (buttonPressed) 中使用它。我正在尝试使用压力触摸在 map 上添加图钉,在两
我设置了一个详细 View ,我是否有几个 Nib 文件根据在 Root View Controller 的表中选择的项目来加载。 我发现,对于 Nibs 的类,永远不会调用 viewDidUnloa
我需要动态访问 json 文件并使用以下代码。在本例中,“bpicsel”和“temp”是变量。最终结果类似于“data[0].extit1” var title="data["+bpicsel+"]
我需要使用第三方 WCF 服务。我已经在我的证书存储中配置了所需的证书,但是在调用 WCF 服务时出现以下异常。 向 https://XXXX.com/AHSharedServices/Custome
在几个 SO 答案(1、2)中,建议如果存在冲突则不应触发 INSERT 触发器,ON CONFLICT DO NOTHING 在触发语句中。也许我理解错了,但在我的实验中似乎并非如此。 这是我的 S
如果进行修改,则会给出org.hibernate.NonUniqueObjectException。在我的 BidderBO 类(class)中 @Override @Transactional(pr
我使用 indexOf() 方法来精细地查找数组中的对象。 直到此刻我查了一些资料,发现代码应该无法正常工作。 我在reducer中尝试了上面的代码,它成功了 let tmp = state.find
假设我有以下表格: CREATE TABLE Game ( GameID INT UNSIGNED NOT NULL, GameType TINYINT UNSIGNED NOT NU
代码: Alamofire.request(URL(string: imageUrl)!).downloadProgress(closure: { (progress) in
我是一名优秀的程序员,十分优秀!