我有一个类似于此的线程类:
class thr {
void run() {
for (;;) {
// block on a queue
// do some processing
++loops_;
}
}
void get_metrics(int& qps) {
std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
double delta = std::chrono::duration<double>(now - last_metrics_time_).count();
qps = std::round(loops_ / delta);
loops_ = 0;
last_metrics_time_ = now;
}
static std::atomic<int> loops_;
static std::chrono::steady_clock::time_point last_metrics_time_;
};
std::atomic<int> thr::loops_ { 0 };
std::chrono::steady_clock::time_point thr::last_metrics_time_ {
std::chrono::steady_clock::now() // initial value: when the program starts
};
这种情况有很多例子。还有另一个线程不时调用get_metrics()。
我想阻止run()
能够访问last_metrics_time_
,因为它不是原子的(只有一个度量收集器线程,所以没有问题)。
将last_metrics_time_
置于get_metrics
本地的静态变量似乎不对,因为它会在首次调用get_metrics
时初始化,而不是在程序启动时初始化。这将导致delta
的值接近于零(因为now
将非常接近last_metrics_time_
)并且第一次调用返回了一个巨大的值。
答案 0 :(得分:10)
没有
但您可以将此多责任类拆分为两个单一责任类:一个运行循环,更新loops_
,并提供对它的只读访问,另一个计算指标并记录时间。然后每个函数只能访问它所需的内容。