- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
在进行一些性能测试时,我遇到了一种我似乎无法解释的情况。
我编写了以下 C 代码:
void multi_arr(int32_t *x, int32_t *y, int32_t *res, int32_t len)
{
for (int32_t i = 0; i < len; ++i)
{
res[i] = x[i] * y[i];
}
}
我使用 gcc 将其与测试驱动程序一起编译为单个二进制文件。我还使用 gcc 将它自己编译成一个共享对象,我通过 p/invoke 从 C# 调用它。目的是测量从 C# 调用 native 代码的性能开销。
在 C 和 C# 中,我创建等长的随机值输入数组,然后测量运行 multi_arr
需要多长时间。在 C# 和 C 中,我都使用 POSIX clock_gettime() 调用进行计时。我将计时调用定位在对 multi_arr 的调用之前和之后,因此输入准备时间等不会影响结果。我运行 100 次迭代并报告平均时间和最小时间。
尽管 C 和 C# 执行完全相同的功能,但 C# 大约有 50% 的时间领先,通常是相当可观的。例如,对于 1,048,576 的 len,C# 的最小值为 768,400 ns,而 C 的最小值为 1,344,105。 C# 的平均值是 1,018,865,而 C 的平均值是 1,852,880。我在这张图中放入了一些不同的数字(注意对数刻度):
这些结果对我来说似乎非常错误,但工件在多个测试中是一致的。我已经检查了 asm 和 IL 以验证正确性。位数是一样的。我不知道什么会影响性能到这种程度。我放了一个最小的复制示例 here .
这些测试都在 Linux(KDE neon,基于 Ubuntu Xenial)上运行,dotnet-core 2.0.0 和 gcc 5.0.4。
有人见过这个吗?
最佳答案
正如您已经怀疑的那样,它取决于对齐方式。返回内存,以便编译器可以将其用于在存储或检索数据类型(例如 double 或整数)时不会导致不必要的错误的结构,但它不保证内存块如何适合缓存。
这会如何变化取决于您测试的硬件。假设您在这里谈论的是 x86_64,这意味着 Intel 或 AMD 处理器及其缓存相对于主内存访问的相对速度。
您可以通过使用各种对齐方式进行测试来模拟这一点。
这是我拼凑的示例程序。在我的 i7 上,我看到了很大的变化,但第一个最不对齐的访问确实比更对齐的版本慢。
#include <inttypes.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
void multi_arr(int32_t *x, int32_t *y, int32_t *res, int32_t len)
{
for (int32_t i = 0; i < len; ++i)
{
res[i] = x[i] * y[i];
}
}
uint64_t getnsec()
{
struct timespec n;
clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &n);
return (uint64_t) n.tv_sec * 1000000000 + n.tv_nsec;
}
#define CACHE_SIZE (16 * 1024 * 1024 / sizeof(int32_t))
int main()
{
int32_t *memory;
int32_t *unaligned;
int32_t *x;
int32_t *y;
int count;
uint64_t start, elapsed;
int32_t len = 1024 * 16;
int64_t aligned = 1;
memory = calloc(sizeof(int32_t), 4 * CACHE_SIZE);
// make unaligned as unaligned as possible, e.g. to 0b11111111111111100
unaligned = (int32_t *) (((intptr_t) memory + CACHE_SIZE) & ~(CACHE_SIZE - 1));
printf("memory starts at %p, aligned %p\n", memory, unaligned);
unaligned = (int32_t *) ((intptr_t) unaligned | (CACHE_SIZE - 1));
printf("memory starts at %p, unaligned %p\n", memory, unaligned);
for (aligned = 1; aligned < CACHE_SIZE; aligned <<= 1)
{
x = (int32_t *) (((intptr_t) unaligned + CACHE_SIZE) & ~(aligned - 1));
start = getnsec();
for (count = 1; count < 1000; count++)
{
multi_arr(x, x + len, x + len + len, len);
}
elapsed = getnsec() - start;
printf("memory starts at %p, aligned %08"PRIx64" to > cache = %p elapsed=%"PRIu64"\n", unaligned, aligned - 1, x, elapsed);
}
exit(0);
}
关于C#调用native代码比native调用native快,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46394497/
在这种情况下,我们在应用程序中同时使用react-native-gesture-handler Touchable和react-native Touchable。 (通过Touchables,我的意思
我有一个 MainFooter包含页脚和迷你播放器的组件,单击时动画显示为全 View 。我有一个问题,每当我们点击一个页脚选项卡时,播放器最大化然后卡在那里,没有响应。 此外,播放器内部的向下箭
我在 native react 之上使用 native 基础组件,我想知道如何在 UI 中使卡片呈圆形而不是矩形。有圆形的 Prop 吗? 最佳答案 好吧,实际上没有人能回答这个问题,但幸运的是我在
我在 native react 之上使用 native 基础组件,我想知道如何在 UI 中使卡片呈圆形而不是矩形。有圆形的 Prop 吗? 最佳答案 好吧,实际上没有人能回答这个问题,但幸运的是我在
我是 react-native 的新手,所以我认为“HTML”而不是“native”可能有点太多了,所以我的问题看起来很愚蠢。 我使用 react-native-router-flux 进行路由,并使
当我使用这个例子在我的应用程序上实现 Image-slider 时,我遇到了这个错误。 import React,{Component} from 'react' import {View,T
我正在为我们的产品使用“Native Base”组件,并且效果很好, 但我有一点被卡住了,它是关于将 Items 放入 Nativebase Picker 的问题。我的代码是这样的 渲染方法代码 -
正如文档中所建议的,我将一些长的数据获取代码移动到 native 模块中以释放 JS 线程,但我观察到这仍然阻塞了 UI。为什么会这样,我能做些什么来避免这种情况? 从 JS 调用 native 模块
我正在使用一个名为 react-native-svg 的框架在 React Native View 中绘制 SVG 元素。 我的目标是,当我点击 View 时(我在全局 View 上使用 PanRes
在 IOS 中发现错误 Native Module cannot be null 我不使用 react-native-push-notification 最佳答案 这通常发生在您未能将第三个库链接到您
当应用程序关闭时,我可以获得由 Linking.getInitialURL() 点击的深层链接网址。 .当应用程序处于后台状态时,则不会安装任何内容。所以,我什至无法通过 Linking.addEve
1) 说原生库是什么意思?什么样的图书馆?那些将用作 gradle 依赖项? 2)如何链接这些?我在使用 link 或 rnpm 时遇到了麻烦。 最佳答案 链接 native 库意味着您要将已经实现的
我需要帮助来构建我的 react 原生项目。我尝试过react-native run-android,但出现以下错误: react-native : The term 'react-native' i
我需要帮助来构建我的 react 原生项目。我尝试过react-native run-android,但出现以下错误: react-native : The term 'react-native' i
我是 React-Native 的新手,到目前为止我很喜欢它。我正在尝试创建一个屏幕(用于跨平台应用程序),右上角有一个菜单图标,单击时,我想打开一个菜单,希望使用 react-native-menu
RN doco 和其他示例展示了如何从 native iOS View Controller 启动 React-Native View ,但反之则不然。谁能解释一下我该怎么做? 最佳答案 我能够弄清楚
对于 react-native - WebStorm 用户: 我正在使用 Jet Brains IDE WebStorm 开始一个带有 React Native 的项目。 在项目 => node_mo
在升级过去的 react-native 0.60 之后......我被警告我应该取消链接所有手动链接的第 3 方库(因为 RN 现在通过自动链接处理它)。 但是,当我运行 react-native u
你可以使用像 https://github.com/tolu360/react-native-google-places 这样的库吗?在世博项目中?我假设任何 npm 库都可以添加,但是像这个 goo
我主要喜欢 React Native。自 0.22 以来一直在使用它。目前为 0.35。 但是为什么链接原生库就像抽奖一样呢?我很少在第一次拍摄时让它发挥作用,而破裂的东西通常是完全不同的东西。 每个
我是一名优秀的程序员,十分优秀!