- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
简单的多线程 c++11 程序,其中所有线程在紧密循环中锁定相同的互斥量。
当它使用8个线程时(作为逻辑cpu的数量)可以达到500万锁/秒
但是只添加一个额外的线程 - 性能会下降到 200,000/秒!
编辑:
在 g++4.8.2 (ubuntu x64) 下:即使有 100 个线程,性能也不会下降!(性能是两倍多,但那是另一回事了) - 所以这似乎确实是一个特定于 VC++ 互斥实现的问题
我用下面的代码(Windows 7 x64)复制了它:
#include <chrono>
#include <thread>
#include <memory>
#include <mutex>
#include <atomic>
#include <sstream>
#include <iostream>
using namespace std::chrono;
void thread_loop(std::mutex* mutex, std::atomic_uint64_t* counter)
{
while (true)
{
std::unique_lock<std::mutex> ul(*mutex);
counter->operator++();
}
}
int _tmain(int argc, _TCHAR* argv[])
{
int threads = 9;
std::mutex mutex;
std::atomic_uint64_t counter = 0;
std::cout << "Starting " << threads << " threads.." << std::endl;
for (int i = 0; i < threads; ++i)
new std::thread(&thread_loop, &mutex, &counter);
std::cout << "Started " << threads << " threads.." << std::endl;
while (1)
{
counter = 0;
std::this_thread::sleep_for(seconds(1));
std::cout << "Counter = " << counter.load() << std::endl;
}
}
VS 2013 分析器告诉我,大部分时间 (95.7%) 都浪费在一个紧密的循环中(rtlocks.cpp 中的第 697 行):
while (IsBlocked() & & spinWait._SpinOnce())
{
//_YieldProcessor is called inside _SpinOnce
}
可能是什么原因?如何改进?
操作系统:windows 7 x64
CPU:i7 3770 4 核(x2 超线程)
最佳答案
如果有 8 个线程,您的代码会旋转,但无需 CPU 在线程失去其时间片之前挂起线程即可获得锁。
随着您添加越来越多的线程,争用级别会增加,因此线程无法在其时间片内获取锁的机会也会增加。当发生这种情况时,线程被挂起并且另一个线程发生上下文切换,CPU 将检查该线程是否可以被唤醒。
所有这些切换、挂起和唤醒都需要从用户模式转换到内核模式,这是一个昂贵的操作,因此性能会受到显着影响。
为了改进,要么减少争用锁的线程数量,要么增加可用内核的数量。在您的示例中,您使用的是 std::atomic
编号,因此您无需锁定即可对其调用 ++
,因为它已经是线程安全的.
关于c++ - VC++ : Performance drop x20 when more threads than cpus but not under g++,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21255311/
当应用程序启动时,您将从 ViewController(中间的那个)开始。从那里我在右边添加了 VC。当用户点击此 VC 上的按钮时,我希望该 View 消失(我目前只需将其框架更改为 900 即可实
我在 Xcode 7 中使用简单的主/详细信息分割 View 模板,并希望用户从具有主要类别的高级 Master VC 开始,然后选择一个并深入到各个项目 - 仍然在左侧- 然后当他们从第二个主 VC
我有一个名为“firstVC”的 View Controller 和另一个名为“secondVC”的 View Controller 。第二个 VC 将以模态方式呈现在第一个 VC 上,当用户按下第二
我被困在一个问题上,我有一个案例 我有一个父 VC 它有2个容器[childVC A, childVC B] API 正在调用 childVC (UIViewController) A 一旦 chil
我从呈现的 VC 导航到 VC(让我们调用 VC A)。在 VC A 中,我有一个容器 View ,它有一个 VC(让我们调用 VC B)。从 VC B 我提出了另一个 VC(我们称之为 VC C)。
我有一个主 VC(称为 VC A),它有一个子 VC(VC B)。当我点击 VC B 上的按钮时,我将其作为子 VC 关闭,但是一旦完成,我想实例化另一个 VC(VC C)。我通过在 VC B 上创建
我们正在寻求改变用户退出我们应用程序的方式。为了做到这一点,我们要关闭当前 VC 下面的所有 VC,并将另一个 VC 作为根 VC。现在我们正在这样做,我相信这不会从内存中消除下面的任何 VC。 le
我仅使用 segue 在我的应用程序中导航。 我只是想知道,如果我从 1st-VC 导航到 V2,再到 V3,然后再到 4th-VC,如何直接从 4th-VC 导航回 1st VC,而不返回到 V3、
最新的 Xcode/Swift/iOS。 我有一个主 VC(称为 StartVC),其中包含一个子 VC(称为 TopBarVC)via 和嵌入式 segue。子 VC 包含一个按钮,按下该按钮时,会
我试图将从 map 中获得的地址传递到我的其他 View Controller 中并保持相同的设置(即开关仍然打开) mapVC 名为 MapViewController,tableVC 名为 Add
我有一个嵌入在 navigationcontroller 中的 viewcontroller,它将另一个 viewcontroller 推送到堆栈上。这个推送的 View Controller 有一个
昨天更新后,我在 iPad 上遇到了这个问题:如果我加载第二个 View Controller ,它看起来比第一个 View Controller 小(见下图)。有趣的是,自从我在 iPad 上更新了
我的项目有这样的设计: VC1 ---firstSegue----> VC2 ---secondSegue---> VC3 通过使用协议(protocol)/委托(delegate)方法,我可以毫无问
图像不是从一个 VC 到另一个 VC。 问题是将它显示到主 UIViewController 中的 ImageView 。我已将所有内容正确连接到 Storyboard中。 点击here查看我的 St
如何在vc++中调用vc++dll。 在.h文件中 typedef int (*LPVAR)(char * ptr_f, char *CC); HINSTANCE hDLL; 在.cpp 文件中 hD
当我从 VC1 转到 VC2 时,如果 VC2 被解雇,我可以通过在 VC2 中设置协议(protocol)并让 VC1 符合它,轻松地将数据传回 VC1。我想做类似的事情,但有以下区别 当我从 VC
我有一个 VC(A),它有一个容器 View 并根据 segementControl 值更改它的 VC(B-C),我在 A ViewController 中发送请求并获得响应,我想确保 B 和C Vi
所以我在我的应用程序中实现了通用链接,当按下按钮时,我打开了我的第二个应用程序: UIApplication.shared.open(aUrl!) 我也接到了电话: func application(
我想在 ios 中模拟联系人列表。当点击 + 按钮添加联系人时,会出现一个新的 View Controller ,为您提供文本字段以输入联系人姓名和用于保存该联系人的其他信息。一旦你点击完成按钮,下一
我有两个 ViewControllers - Slider 和 Foo。我将 Slider 添加为 Foo subview 并将其框架设置在 Foo VC 中,但是 Slider 的框架没有改变 ->
我是一名优秀的程序员,十分优秀!