Несмотря на то, что вы инкапсулируете фактический вызов функции, вероятно, будет лучше, если вы заставите вашу функцию meter
возвращать значение, возвращаемое функцией, которую она измеряет, чтобы сделать возможным цепочку вызовов - но все же с возможностьюпроверьте, сколько времени занимал каждый отдельный звонок.Это делает теоретически возможным запуск решения RVO / copy и, следовательно, не так сильно замедляет код.Пример:
#include <chrono>
#include <iostream>
#include <thread> // for debug sleeps only
using namespace std::chrono;
template<typename D, typename F, typename... P>
auto meter(D& dur, F func, P&&... params) {
auto start = high_resolution_clock::now();
auto retval = func(std::forward<P>(params)...);
// put duration in the duration reference
dur = duration_cast<D>(high_resolution_clock::now() - start);
// and return func()'s return value
return retval;
}
namespace m {
double add(double a, double b) {
std::this_thread::sleep_for(milliseconds(10));
return a + b;
}
double sub(double a, double b) {
std::this_thread::sleep_for(milliseconds(11));
return a - b;
}
double mul(double a, double b) {
std::this_thread::sleep_for(milliseconds(12));
return a * b;
}
double div(double a, double b) {
std::this_thread::sleep_for(milliseconds(13));
return a / b;
}
} // namespace m
int main() {
milliseconds Add, Sub, Mul, Div;
// chaining calls for this calculation:
// (1000 / (100 * (4.3 - (1.1+2.2))))
auto result = meter(Div, m::div,
1000.0, meter(Mul, m::mul,
100.0, meter(Sub, m::sub,
4.3, meter(Add, m::add,
1.1, 2.2)
)
)
);
std::cout << "Add: " << Add.count() << " ms.\n";
std::cout << "Sub: " << Sub.count() << " ms.\n";
std::cout << "Mul: " << Mul.count() << " ms.\n";
std::cout << "Div: " << Div.count() << " ms.\n";
std::cout << result << "\n";
}
Вероятный вывод:
Add: 10 ms.
Sub: 11 ms.
Mul: 12 ms.
Div: 13 ms.
10