- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我最近一直在优化/对一些代码进行基准测试并遇到了这个方法:
public void SomeMethod(Type messageType)
{
if (messageType == typeof(BroadcastMessage))
{
// ...
}
else if (messageType == typeof(DirectMessage))
{
// ...
}
else if (messageType == typeof(ClientListRequest))
{
// ...
}
}
这是从其他地方的性能关键循环调用的,所以我很自然地假设所有这些 typeof(...)
调用都增加了不必要的开销(我知道这是一种微优化)并且可以移动到类中的私有(private)字段。 (我知道有更好的方法来重构这段代码,但是,我仍然想知道这里发生了什么。)
根据我的基准测试,情况完全不是这样(使用 BenchmarkDotNet )。
[DisassemblyDiagnoser(printAsm: true, printSource: true)]
[RyuJitX64Job]
public class Tests
{
private Type a = typeof(string);
private Type b = typeof(int);
[Benchmark]
public bool F1()
{
return a == typeof(int);
}
[Benchmark]
public bool F2()
{
return a == b;
}
}
我机器上的结果(Window 10 x64、.NET 4.7.2、RyuJIT、发布版本):
编译成ASM的函数:
F1
mov rcx,offset mscorlib_ni+0x729e10
call clr!InstallCustomModule+0x2320
mov rcx,qword ptr [rsp+30h]
cmp qword ptr [rcx+8],rax
sete al
movzx eax,al
F2
mov qword ptr [rsp+30h],rcx
mov rcx,qword ptr [rcx+8]
mov rdx,qword ptr [rsp+30h]
mov rdx,qword ptr [rdx+10h]
call System.Type.op_Equality(System.Type, System.Type)
movzx eax,al
我不知道如何解释 ASM,因此无法理解这里发生的事情的重要性。简而言之,为什么 F1 更快?
最佳答案
您发布的程序集表明 mjwills 的评论正如预期的那样是正确的。正如链接文章所指出的,抖动对于某些比较可能很聪明,这就是其中之一。
让我们看看您的第一个片段:
mov rcx,offset mscorlib_ni+0x729e10
rcx 是调用成员函数的“this 指针”。在这种情况下,“this 指针”将是某个 CLR 预分配对象的地址,具体是什么我不知道。
call clr!InstallCustomModule+0x2320
现在我们调用那个对象的一些成员函数;我不知道是什么。您拥有调试信息的最近 公共(public)函数是 InstallCustomModule,但很明显我们没有在这里调用 InstallCustomModule;我们正在调用距 InstallCustomModule 0x2320 字节的函数。
看看 InstallCustomModule+0x2320 的代码做了什么会很有趣。
无论如何,我们进行调用,返回值进入 rax。继续:
mov rcx,qword ptr [rsp+30h]
cmp qword ptr [rcx+8],rax
这看起来像是从 this
中获取 a
的值,并将它与函数返回的任何值进行比较。
其余代码非常普通:将比较的 bool 结果移入返回寄存器。
简而言之,第一个片段相当于:
return ReferenceEquals(SomeConstantObject.SomeUnknownFunction(), this.a);
显然这里有根据的猜测是常量对象和未知函数是具有特殊用途的助手,可以快速获取常用类型对象,如 typeof(int)。
第二个有根据的猜测是,抖动自行决定模式“将 Type 类型的字段与 typeof(something) 进行比较”最好作为对象之间的直接引用比较。
现在您可以亲眼看到第二个片段的作用。它只是:
return Type.op_Equality(this.a, this.b);
它所做的只是调用一个辅助方法来比较两种类型的值相等性。请记住,CLR 不保证所有等效类型对象的引用相等。
现在应该清楚为什么第一个片段更快了。 抖动对第一个片段的了解要多得多。例如,它知道 typeof(int) 将始终返回相同的引用,因此您可以进行廉价的引用比较。它知道 typeof(int) 永远不会为 null。它知道 typeof(int) 的确切类型——记住,Type
不是密封的;您可以创建自己的 Type
对象。
在第二个片段中,抖动只知道它有两个 Type
类型的操作数。它不知道它们的运行时类型,也不知道它们的无效性;就其所知,您自己对 Type
进行了子类化,并组成了两个引用不相等但值相等的实例。它必须回落到最保守的位置并调用一个开始在列表中向下移动的辅助方法:它们都是空的吗?其中一个为空,另一个为非空?他们引用相等吗?等等。
看起来缺乏这些知识会让您付出巨大的代价……半纳秒。我不会为此担心。
关于c# - 为什么 typeA == typeB 比 type == typeof(Type B) 慢?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54874400/
我有一个这样的宏: #include #include #define m_test_type(e) \ do {
我知道 JS 中的严格等于评估两件事:平等和相似 Object.is() 是我能找到的最接近的比较来收集更多的洞察力,它在我的调查中没有提供进一步的帮助。 谁能更好地理解 JS 的内部结构?数组是一个
我正在学习 JavaScript,我在代码中看到使用 typeof 和 typeof() 是一样的,例如: 两种情况下的结果都是数字: console.log(typeof 1); console.l
据我所知,检查 undefined variable 的首选方法是 typeof a === 'undefined'。 但为什么它比 typeof a == 'undefined' 更好?它会在哪些地
今天我在 Linux 内核中遇到了这个宏 (include/linux/kernel.h) #define DIV_ROUND_CLOSEST(x, divisor)( \ {
我收到错误 Type 'typeof Questions' is not assignable to type 'typeof Model'.同时在模型中添加了关系。 问题.model.ts impo
只是四处看看我通常掩盖的地方,然后注意到了这一点。 typeof('apple'); //"string" typeof 'apple'; //"string" 好的,首先,第二个示例是如何工作的?我
我正在尝试检查以下内容 typeof( ICollection<> ).GetTypeInfo().IsAssignableFrom( targetProperty.PropertyType.GetT
所以我明白 typeof 可以是一个运算符还是一个函数。但是当我这样做时 console.log(typeof(typeof)); 我收到了这条消息 未捕获的语法错误:意外的标记 ')' 那么我在这里
我正在使用 node/typescript 为 Discord 开发一个机器人。当我在我的源代码上运行 typescript 编译器时,我收到了这个错误: node_modules/@types/re
我有typeof(List)作为类型对象,但我需要 typeof(List<>)我可以从中使用 MakeGenericType()检索 List 的类型对象,是否可能? 更新:伙计们,谢谢。这似乎是一
我是js的新手,正在努力学习js,你们能告诉我为什么typeof typeof x返回string,提供下面的代码片段,如果我理解这个简单的概念,它将对我有更多帮助: var x=null; cons
简短的例子: #include #include using namespace boost; template BOOST_TYPEOF_TPL(T() + U()) add2(const T&
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
检查 typeof 运算符结果的两个版本之间有什么区别(编译器/解释器/juju wise 等)吗? 我问是因为我多次看到第一个版本,好像它遵循了一个概念,而第二个版本更具可读性并且更好地描述了我的意
这一定很简单,但我找不到解决方案。我有一个我想现在 undefined object 。如何将 typeof 更改为未定义?谢谢! 凯尔 最佳答案 你不能让对象本身undefined,但是你可以让引用
这两种说法有什么区别? if (typeof errorMessage !== undefined) {} 和 if (typeof (errorMessage) !== undefined) {}
有一个简单的问题,我想我知道答案但我不想听来自其他一些人。 假设我将创建一个通用函数: public string GetTypeName() { } 它应该返回类型 T 的名称,这很简单: retu
正如标题所说的那样,typeof (Array, null) 返回 object 而 typeof(null, Array) 返回 函数。 它返回第二个参数的类型。 为什么? 最佳答案 因为 type
这是一个令人尴尬的问题,但我觉得我在过去几个小时里尝试了一切。 我只想在我的属性中添加以下属性 #using #using ... using namespace System::Draw
我是一名优秀的程序员,十分优秀!