- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我最近在这里阅读了这个问题 Why is it faster to process a sorted array than an unsorted array?并找到了绝对令人着迷的答案,它完全改变了我在处理基于数据的分支时对编程的看法。
我目前有一个相当基本但功能齐全的解释型英特尔 8080 仿真器,它是用 C 语言编写的,操作的核心是一个 256 长的 switch-case 表,用于处理每个操作码。我最初的想法是,这显然是最快的工作方法,因为操作码编码在整个 8080 指令集中并不一致,解码会增加很多复杂性、不一致和一次性情况。充满预处理器宏的 switch-case 表非常简洁且易于维护。
不幸的是,在阅读上述帖子后,我突然想到我计算机中的分支预测器绝对无法预测 switch case 的跳跃。因此,每次导航 switch-case 时,管道都必须被完全删除,导致几个周期延迟,否则这个程序应该是一个非常快的程序(我的代码中甚至没有乘法)。
我敢肯定你们中的大多数人都在想“哦,这里的解决方案很简单,转向动态重新编译”。是的,这看起来确实会削减大部分开关盒并显着提高速度。不幸的是,我的主要兴趣是模拟旧的 8 位和 16 位时代的控制台(这里的英特尔 8080 只是一个例子,因为它是我最简单的模拟代码),其中周期和时间保持准确的指令很重要,因为视频和声音必须根据这些确切的时间进行处理。
当处理这种级别的准确性时,性能成为一个问题,即使对于较旧的控制台也是如此(例如,看看 bSnes)。在处理具有长流水线的处理器时,是否有任何追索权或这只是一个事实?
最佳答案
相反,switch
语句很有可能被转换为jump tables ,这意味着它们可能会执行一些 if
(用于范围检查)和一次跳转。 if
不应导致分支预测出现问题,因为您不太可能有错误的操作码。跳转对流水线不是很友好,不过到头来,整个switch
语句,就这么一个..
我认为您无法将较长的操作码 switch
语句转换为任何其他可带来更好性能的形式。当然,前提是您的编译器足够聪明,可以将其转换为跳转表。如果没有,您可以手动执行此操作。
如有疑问,请实现其他方法并衡量绩效。
首先,请确保您不会混淆 branch prediction和 branch target prediction .
分支预测仅适用于分支语句。它决定分支条件是失败还是成功。它们与跳转语句无关。
另一方面,分支目标预测试图猜测跳跃的终点。
因此,您的陈述“分支预测器无法预测跳跃”应该是“分支目标预测器无法预测跳跃”。
在您的特定情况下,我认为您实际上无法避免这种情况。如果你有一个非常小的操作集,也许你可以想出一个涵盖所有操作的公式,就像在逻辑电路中所做的那样。然而,对于与 CPU 一样大的指令集,即使它是 RISC,该计算的成本也远高于单次跳转的代价。
关于c - 在 CPU 仿真中使用 switch case 时如何处理分支预测,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11668090/
是否有使用 switch 语句检查数字 0-9 的简单方法?我正在编写一个程序来检查某些字符和数字。就像检查“\0”、“F”或“f”一样,想知道是否还有一种方法可以以类似的方式检查 0-9。我知道我可
我有一些数据需要转换,为此我需要一个超过 50 个案例的切换条件,我需要 3 次相同的案例,但在第三次我需要 50 个案例和一些更多,我不想写两次相同的代码。也许有可能做这样的事情。 switch (
我遇到这种情况,我必须检查两个 GET 变量。在检查语句内的第一个 switch 语句后,必须在第一个 case 循环内的第二个 switch 语句中检查第二个变量。 我无法在这里发布确切的代码,但这
如何使用函数指针代替 switch 语句? 最佳答案 与 ars 发布的链接略有不同的方法:您可以将 switch 语句中的值用作函数指针数组中的数组索引。所以不要写 switch (i) {
我必须评估很多条件。就我而言,我必须做这样的事情: switch(id) { case 5: // switch some other cases here case
switch 按钮位于 switch 语句内,但仅在 switch 语句外部时才有效这是我的代码:
我试图在 switch 语句中有一个 case 跳转到不同的 switch 语句。 在实践中,我希望用户在文本框中键入“关闭页面”,并且在浏览器关闭页面之前,我希望询问用户是否确定。输入“yes”将关
(引用java)我试图确定哪个更好,编写更多代码并可能节省一些计算时间,或者编写更少代码但可能牺牲一些计算时间。这就是我好奇的地方这样做会更有效率吗: switch (availability) {
我正在尝试构建一个 Android 应用程序,该应用程序可以访问加速度计传感器,并在单击按钮时将加速度计值(由 <> 包围)输出到串行 USB。当我更新值并尝试在 onClick 命令中调用它时遇到问
如何使用 Dwoo 模板引擎实现 switch case 语法。 最佳答案 {if 3 == 5} never gonna happen {elseif 3 == 3} if you don'
我想知道一个大的 switch 语句和几个小的 switch 语句之间是否有性能差异。 包含数百个案例的大型 switch 语句。 switch(quest){ case 1:
用户在 2 个选择框中进行选择后,我尝试计算出报值(value)。 看起来 1 需要 2 个 switch 语句。这可能吗? (可能的值比下面的值多得多。为了清楚起见,我删除了它们) var wor
我在主 while 循环内有一个开关,它将运行我的游戏。我正在尝试打破我的开关,以便转到不同的案例。下面的例子更好地解释了这一点: int j = 0; While(1){ switch(j){ ca
我用 Java 创建了一个菜单,其中每个选项都有另一个菜单。我想知道是否可以从内部菜单退出回到主菜单。 编辑:添加了主菜单选项 System.out.println("Main Menu");
我有一个计算新分数的方法。下面的方法有效,但问题是代码本身看起来可以被显着清理。我只是不知道什么是最好的方法。我根据 filterString 和枚举 individualScoreState 分配
在 Lisp 中使用字符串切换语句。 (defun switch(value) (case value (("XY") (print "XY"))
我想做这样的事情: int i = 0; switch(difficulty) { case 1: i++; break; case 2: i--; break; defaul
在 Obj-c 中,我做了一个 switch 语句,我曾经使用 UIsplitviewcontroller 在我的 iPad 应用程序中四处移动现在我想快速地做同样的事情……我试了几个小时,现在我唯一
我想写一个结构如下的报告: \begin{document} \input[option=a]{class} \input[option=b]{class} \in
我正在认真阅读 ngSwitch 的 AngularJS API 引用。当我谈到那部分时的指令: place an expression on the on="..." attribute (or t
我是一名优秀的程序员,十分优秀!