- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我需要运行 DAXPY 的计时线性代数内核。天真地,我想尝试这样的事情:
fill(X,operand_size);
fill(Y,operand_size);
double seconds = timer();
daxpy(alpha,X,Y,operand_size);
seconds = timer() - seconds;
如果需要,完整的代码链接在最后。
问题是,填充操作数 x 和 y 的内存访问将导致它们被放置在处理器缓存中。因此,随后在 DAXPY 调用中访问内存比在生产运行中实际访问要快得多。
我比较了两种解决这个问题的方法。第一种方法是通过 clflush 刷新各级缓存中的操作数。操作说明。第二种方法是读取一个非常大的数组,直到操作数条目“自然地”从缓存中逐出。我测试了两者,这是使用大小为 2048 的操作数的单个 DAXPY 调用的运行时:
Without flush: time=2840 ns
With clflush: time=4090 ns
With copy flush: time=5919 ns
这是几秒钟后的另一次运行:
Without flush: time=2717 ns
With clflush: time=4121 ns
With copy flush: time=4796 ns
正如预期的那样,刷新会增加运行时间。但是,我不明白复制刷新如何导致 DAXPY 例程的运行时间大大延长。 clflush 指令应该从所有缓存中逐出操作数,因此 clflush 的时间应该是任何其他缓存刷新过程的执行时间的上限。不仅如此,刷新时间(对于两种方法)也有很大的反弹(数百纳秒与未刷新情况下的不到 10 纳秒)。有谁知道为什么手动刷新在运行时会有如此大的差异?
附录
包含所有计时例程和刷新例程的完整代码位于此处(194 行):
这是我的 gcc 版本。代码是使用 -O3 选项编译的。 (我知道,它很旧;我必须构建的一些软件与较新的 gcc 不兼容)
使用内置规范。
Target: i686-apple-darwin10
Configured with: /var/tmp/gcc/gcc-5646.1~2/src/configure --disable-checking --enable-werror --prefix=/usr --mandir=/share/man --enable-languages=c,objc,c++,obj-c++ --program-transform-name=/^[cg][^.-]*$/s/$/-4.2/ --with-slibdir=/usr/lib --build=i686-apple-darwin10 --with-gxx-include-dir=/include/c++/4.2.1 --program-prefix=i686-apple-darwin10- --host=x86_64-apple-darwin10 --target=i686-apple-darwin10
Thread model: posix
gcc version 4.2.1 (Apple Inc. build 5646) (dot 1)
我在 Mac OS X 10.6 上使用 Intel Xeon 5650 处理器。
最佳答案
CPU 实际为正常读取所做的(部分)伪代码(在您的基准测试中会经常发生)可能是:
if( cache line is not in cache ) {
if(cache is full) {
find cache line to evict
if( cache line to evict is in "modified" state ) {
write cache line to evict to slower RAM
}
set cache line to "free"
}
fetch the cache line from slower RAM
}
temp = extract whatever we're reading from the cache line
如果您使用 CLFLUSH
刷新缓存,则 if(cache is full)
将为 false,因为 CLFUSH
将缓存留空。
如果您使用复制来刷新缓存,那么 if(cache is full)
分支将为真,并且 if( cache line is modified )
也将为一半时间为真(一半缓存将包含您在复制期间读取的数据,另一半将包含您在复制期间写入的数据)。这意味着有一半的时间您最终会执行写入缓存行以逐出较慢的 RAM
。
执行写入缓存行以逐出到 RAM
将消耗 RAM 芯片带宽并影响从较慢的 RAM 中获取缓存行
的性能。
关于c++ - 缓存刷新例程之间的时间不一致,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15712491/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!