- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
有一天,我遇到了一项任务,需要在没有条件运算符和循环(以及位运算)的情况下确定两个数字中的最大值。经过深思熟虑,我做出了这个决定:
long long mmax(long long a, long long b) {
return (a+b+(a-b)*((2*(a-b)+1)%2))/2;
}
只是为了好玩,我决定检查一下哪个函数更快,因此我在包含 10^7 对从 1 到 10^17 的随机整数的 3 个数据样本上测试了这两个函数大约 100 次。我很惊讶,因为对于从 1 到 10^5 的整数,我的函数的每次调用速度至少加快了 0.092 秒,而对于从 10^5 到 10^17 的整数,每次调用的速度至少加快了 0.044 秒。平均而言,我的函数在处理从 1 到 10^5 的整数时速度快了 0.1 秒,在处理从 10^5 到 10^17 的整数时速度快了 0.06 秒。所以,我不是优化方面的专家,因此我想问这个函数是否真的比 std:max 更快?
这是我的测试代码:
#include <chrono>
#include <iostream>
#include <fstream>
#include <algorithm>
using namespace std;
long long mmax(long long a, long long b) {
return (a+b+(a-b)*((2*(a-b)+1)%2))/2;
}
int main(){
auto started = std::chrono::high_resolution_clock::now();
ifstream in("bilt.txt");
long long a, b;
while(in >> a >> b) {
mmax(a,b);
}
auto done = std::chrono::high_resolution_clock::now();
std::cout << std::chrono::duration_cast<std::chrono::milliseconds>(done-started).count() << endl;
}
最佳答案
您的基准测试不是对max
进行基准测试。它完全受到输入操作所需时间的限制。输入操作花费的时间比 std::max
或您的 mmax
实现长很多倍。
此外,mmax
或 std::max
调用将被任何优化编译器优化掉,因为它们的结果从未被使用,并且它们没有任何其他结果副作用。参见例如here on godbolt 。因此,您可能根本不会对它们进行基准测试。
假设您的说法属实:
您的函数对于某些参数具有未定义的行为,而 std::max
则没有,例如如果a+b
导致溢出,则它具有未定义的行为。因此,比较速度确实不公平,因为您的实现并不总是有效。
Here is a quick-bench具有更好的(尽管未经严格验证)基准。
std::max
正如您在图中所看到的,您的实现比简单的实现或 std::max
更差,两者在性能上是相同的。
然而,与您的相反,天真的和标准库实现实际上适用于所有可能的输入值(我将 vector 中的测试用例值限制在适合您的实现的范围内。)
<小时/>在此基准测试的先前版本中,我犯了一个错误(正确执行基准测试很困难!),这使得幼稚的实现看起来比 std::max
和 OP 的实现要糟糕得多,事实证明,这是谷歌基准测试的 DoNotOptimize
工作原理的产物(至少在 Clang 上,也许这是谷歌基准测试中的一个错误,或者可能是我使用错误)。如果有人发现其他缺陷,请告诉我!
基准代码:
#include<random>
#include<cmath>
#include<utility>
#include<algorithm>
const auto N = 10'000;
auto values = []{
std::vector<std::pair<long long, long long>> v;
std::default_random_engine rng{std::random_device{}()};
std::uniform_int_distribution<long long> dist{-100'000'000, 100'000'000};
for(int i = 0; i<N; i++)
v.emplace_back(dist(rng), dist(rng));
return v;
}();
void std_impl(benchmark::State& state) {
for (auto _ : state) {
for(auto& x : values) {
auto result = std::max(x.first, x.second);
benchmark::DoNotOptimize(result);
}
}
}
void naive_impl(benchmark::State& state) {
for (auto _ : state) {
for(auto& x : values) {
auto result = x.first > x.second ? x.first : x.second;
benchmark::DoNotOptimize(result);
}
}
}
long long mmax(long long a, long long b) {
return (a+b+(a-b)*((2*(a-b)+1)%2))/2;
}
void op_impl(benchmark::State& state) {
for (auto _ : state) {
for(auto& x : values) {
auto result = mmax(x.first, x.second);
benchmark::DoNotOptimize(result);
}
}
}
void only_iter(benchmark::State& state) {
for (auto _ : state) {
for(auto& x : values) {
auto result = x.first;
benchmark::DoNotOptimize(result);
}
}
}
BENCHMARK(std_impl);
BENCHMARK(naive_impl);
BENCHMARK(op_impl);
BENCHMARK(only_iter);
关于c++ - 在我看来,我发现了一个比 std :max 运行得更快的函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59457889/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!