- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在开发一个引擎,我们可以在运行时动态复制大量属性。根据情况,我们可能会也可能不会修改属性值。它最初是用反射写的,但是由于性能问题,我们最近在 Reflection.Emit
中重新写了它。重写完成,性能明显好很多,但现在代码正在与手写的 C#
进行基准测试。显然,为了公平起见,用于基准测试的手写 C#
与 IL
具有“相似的功能”(您很快就会明白我的意思) .
一些 IL
引擎已经被签署,因为它以优异的成绩通过了测试,并且与手写的 C#
几乎是 1:1 的。这告诉我:
调用动态方法没有开销
我们的总体概念和实现是正确的
基准测试是正确的
IL
和手写的 C#
正在以完全相同的方式进行测试,因此没有有趣的 JIT
业务在进行(我别想)
我们期望 IL
比手写的稍微慢一些,但到目前为止情况并非如此。在长回合中它可能会慢几毫秒,但您可以在 IL
中使用快捷方式,这样有助于弥补差异。
在一种特殊情况下,它的速度要慢得多。慢 2 倍。
在 C#
中,您将拥有:
class Source
{
public string S1 { get; set; }
public int I1 { get; set; }
public int I2 { get; set; }
public double D1 { get; set; }
public double D2 { get; set; }
public double D3 { get; set; }
}
class Dest
{
public string S1 { get; set; }
public int I1 { get; set; }
public string I2 { get; set; }
public double D1 { get; set; }
public int D2 { get; set; }
public string D3 { get; set; }
}
static Dest Test(Source s)
{
Dest d = new Dest();
object o = s.D3;
if (o != null)
d.D3 = o.ToString();
return d;
}
这就是我所说的类似功能的意思。为了通用,当我们将属性复制到字符串时,我们首先将其装箱,然后调用 Object.ToString()
。 native 地,值类型调用 ToString
不同,因此上面的代码是同类。
如果我注释掉 D3
copy/ToString
并取消注释其他 5 个属性,我将回到 1:1 与 C#
.
你会注意到 I2
是 int
-> string
,但出于某种原因,那个没有同样的问题与 double
-> string
一样。我知道 double ToString()
通常更昂贵,但该费用也应显示在 C# 代码中,但事实并非如此。
我为 D3
副本发出的代码与我为 I2
副本发出的代码相同,为什么 D3
的开销很大复制?
编辑:
编译器发出:
IL_0000: newobj instance void ConsoleApplication3.Dest::.ctor()
IL_0005: ldarg.0
IL_0006: callvirt instance float64 ConsoleApplication3.Source::get_D3()
IL_000b: box [mscorlib]System.Double
IL_0010: stloc.0
IL_0011: dup
IL_0012: ldloc.0
IL_0013: brtrue.s IL_0018
IL_0015: ldnull
IL_0016: br.s IL_001e
IL_0018: ldloc.0
IL_0019: callvirt instance string [mscorlib]System.Object::ToString()
IL_001e: callvirt instance void ConsoleApplication3.Dest::set_D3(string)
IL_0023: ret
我的代码的这个特定部分不会为 Dest 对象发出新的,这是在其他地方完成的。如上面的 C#
所示,dup 正在复制 Dest 对象。
LocalBuilder localBuilderObject = generator.DeclareLocal(_typeOfObject);
Label labelNull = generator.DefineLabel();
Label labelNotNull = generator.DefineLabel();
generator.Emit(OpCodes.Ldarg_0);
generator.Emit(OpCodes.Callvirt, miGetter);
generator.Emit(OpCodes.Box, typeSource);
generator.Emit(OpCodes.Stloc_S, localBuilderObject);
generator.Emit(OpCodes.Dup);
generator.Emit(OpCodes.Ldloc_S, localBuilderObject);
generator.Emit(OpCodes.Brtrue, labelNotNull);
generator.Emit(OpCodes.Ldnull);
generator.Emit(OpCodes.Br, labelNull);
generator.MarkLabel(labelNotNull);
generator.Emit(OpCodes.Ldloc_S, localBuilderObject);
generator.Emit(OpCodes.Callvirt, _miToString);
generator.MarkLabel(labelNull);
generator.Emit(OpCodes.Callvirt,miSetter);
正如我提到的,我将类型框起来,这样我就可以一般地调用 Object::ToString()
而不必担心值类型。 Ref 类型也经过这条路径。 C#
代码被设计成这样,但仍然需要 1/2 的时间???
我整个周末都在处理这个问题。进一步测试显示其他值类型是 1:1。 int
、long
等。由于某种原因,double
导致了问题。
最佳答案
正如您在 C#
编译代码中所见,使用了快速本地访问指令:
IL_000b: box [mscorlib]System.Double
IL_0010: stloc.0
IL_0011: dup
IL_0012: ldloc.0
...
IL_0018: ldloc.0
相反,在您的 IL
生成的代码中,您使用 STLoc.s
和 ldloc.s
,它们也采用本地索引的操作数.
还要确保缓存(如果 C#
运行速度只快两倍,则可能是缓存)为每个 Type
生成的方法。
关于C# EMIT IL 性能问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36961159/
我只是偶然发现了一个似乎广为人知的compsci关键字“emit”。但是我找不到用通用计算机科学术语对其的任何明确定义,也找不到任何特定编程语言中的“emit()”函数或关键字的特定定义。 我在这里找
在 Qt 中,它们都是有效的,并且表现相同: emit someSignal(value); 对比 emit(someSignal(value)); 有什么区别吗? 最佳答案 Is there any
是否可以同步执行并在调用方法本身时返回结果。所以,我想在 $emit 完成后执行下一条语句。其如下: Parent component has method, doC
我是 socket.io 的新手,遇到了一些看起来很奇怪的事情。我实际上不知道 socket.emit 和 io.emit 之间的区别,但我在任何地方都找不到解释。 io.on('connection
有谁知道如何显式实现接口(interface)的属性 使用反射。发射? 最佳答案 有关 TypeBuilder.DefineMethodOverride 的信息,请参阅 MSDN 文档,其中包括使用
我有多个可观察对象进行网络调用,并且仅当所有可观察对象发出错误时,我才需要从组合器可观察对象发出错误。如果至少有一个可观察完成,则应传递结果。 我当前的流功能如下: Observable.fromIt
我想在这里实现的是当 IF 语句为 true 时 fightID 更改为其他值,因此它不会匹配它之前分配的值,因此我无法在战斗结束后立即通过控制台运行 win-fight 发出的消息。但是我现在得到的
当我们需要自定义组件在 VueTable2 中发出事件时,我们必须使用: this.$parent.$emit('myCustomEvent') // instead of this.$emit('m
This is my production webpack config .标题中的两个引号分别指的是 webpack2 和 webpack。两者都因类似的错误而挂起。 这是我触发构建的命令 设置 N
我正在尝试构建一个发布者,它会在其他 5 个发布者中的任何一个发布者发出 true 时发出 true。我已经设法构建了一个工作版本,但感觉非常恶心,使用 CombineLatest4 + Combin
我正在尝试创建用于教育目的的简单 .net 编译器。在解析、扫描和构建 AST 之后,我正在使用 Reflection.Emit.ILGenerator 生成 .net 程序集。 这是我的程序集生成示
以下 node.js 脚本不工作 var EventEmitter = require('events').EventEmitter; var util = require('util'); var
我使用的是Laravel 10,Livewire 2.x,我试图使用Livewire组件中的一个简单函数中的emit或emitTo事件来更新愿望列表计数。。WishlistCount控制器:。当我运行
我收到以下警告: [Vue warn]: Extraneous non-emits event listeners (addData) were passed to component but cou
我不明白 io.emit 和 io.sockets.emit 之间有什么区别。 有时它们的行为相同,但有时它们的行为不同。 最佳答案 socket.emit 仅将消息发送到发送者客户端 io.emit
我刚刚成功地使用 socket.io 连接到一个服务器脚本,所以我很高兴。我对我的脚本生成的奇怪行为不太满意:我在单击按钮时向服务器脚本发送一个发射,然后服务器测试脚本向控制台日志发回一条消息 6x。
这听起来可能有点模糊,但我仍在学习并试图理解它是如何工作的。 我开始使用“侧菜单”启动 ionic 项目,它已经设置了带有“应用程序”抽象状态/ View / Controller 的 urlrout
我遵循了一些 Nodejs 教程并完成了一个简单的 REST Web 服务。此 Web 服务监听/api/accounts 中的电子邮件和密码,然后继续在 Cassandra 集群中进行查找。我使用
我正在尝试建立一种模式,在这种模式下,我的产生某些对象的可观察对象被转换为领域事件,例如围绕可观察对象产生的 Started、Success、Error,如果这有意义的话 public Observa
这个问题已经有答案了: How can I let the javascript catch a signal from node and prompt a window right after th
我是一名优秀的程序员,十分优秀!