我在编写 Microsoft 特定的 C++ 代码时被告知要编写 Sleep(1)在自旋锁定方面比 Sleep(0) 好得多,因为 Sleep(0) 将使用更多的 CPU 时间,而且,它只有在有另一个同等优先级线程等待运行。
但是,对于 C++11 线程库,没有太多关于 std::this_thread::yield() 效果的文档(至少我能够找到) vs. std::this_thread::sleep_for(std::chrono::milliseconds(1));第二个肯定更冗长,但它们对于自旋锁是否同样有效,或者它是否受到影响 Sleep(0) 与 Sleep(1)?
一个示例循环,其中 std::this_thread::yield() 或 std::this_thread::sleep_for( std::chrono::milliseconds(1) ) 可以接受:
void SpinLock( const bool& bSomeCondition )
{
// Wait for some condition to be satisfied
while( !bSomeCondition )
{
/*Either std::this_thread::yield() or
std::this_thread::sleep_for( std::chrono::milliseconds(1) )
is acceptable here.*/
}
// Do something!
}
最佳答案
这里的标准有些模糊,因为具体的实现很大程度上会受到底层操作系统调度能力的影响。
话虽如此,您可以放心地假设任何现代操作系统上的一些事情:
yield 将放弃当前的时间片并将线程重新插入调度队列。在线程再次执行之前到期的时间量通常完全取决于调度程序。请注意,标准将产量称为重新安排的机会。因此,如果需要,实现可以完全自由地立即从 yield 中返回。 yield 永远不会将线程标记为非事件状态,因此在 yield 上旋转的线程将始终在一个内核上产生 100% 的负载。如果没有其他线程准备好,您最多可能会丢失当前时间片的剩余部分,然后再重新安排。sleep_* 将至少在请求的时间内阻塞线程。一个实现可以将 sleep_for(0) 变成 yield。另一方面, sleep_for(1) 将使您的线程暂停。该线程不是返回调度队列,而是首先进入另一个 sleep 线程队列。只有在请求的时间量过去后,调度程序才会考虑将线程重新插入调度队列。小 sleep 产生的负载仍然会非常高。如果请求的休眠时间小于系统时间片,可以预期线程只会跳过一个时间片(即一个yield释放事件时间片,然后跳过一个),这仍然会导致cpu负载在一个核心上接近甚至等于 100%。关于哪个更适合自旋锁定的几句话。当对锁几乎没有争用时,自旋锁是一种选择的工具。如果在绝大多数情况下您希望锁可用,则自旋锁是一种便宜且有值(value)的解决方案。但是,一旦发生争用,自旋锁会让你付出代价。如果您担心 yield 或 sleep 是否是更好的解决方案,那么这里自旋锁是 错误的工作工具。您应该改用互斥锁。
对于自旋锁,您实际上必须等待锁定的情况应该被视为异常(exception)。因此,在这里直接屈服是完全没问题的——它清楚地表达了意图,浪费 CPU 时间一开始就不应该成为问题。
关于C++11 线程等待行为:std::this_thread::yield() 与 std::this_thread::sleep_for( std::chrono::milliseconds(1) ),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17325888/
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:
我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我
我明白了defa(&block)block.call(self)end和defa()yieldselfend导致相同的结果,如果我假设有这样一个blocka{}。我的问题是-因为我偶然发现了一些这样的代码,它是否有任何区别或者是否有任何优势(如果我不使用变量/引用block):defa(&block)yieldselfend这是一个我不理解&block用法的具体案例:defrule(code,name,&block)@rules=[]if@rules.nil?@rules 最佳答案 我能想到的唯一优点就是自省(introspecti
在Ruby(尤其是Rails)中,您经常需要检查某物是否存在,然后对其执行操作,例如:if@objects.any?puts"Wehavetheseobjects:"@objects.each{|o|puts"hello:#{o}"end这是最短的,一切都很好,但是如果你有@objects.some_association.something.hit_database.process而不是@objects呢?我将不得不在if表达式中重复两次,如果我不知道实现细节并且方法调用很昂贵怎么办?显而易见的选择是创建一个变量,然后测试它,然后处理它,但是你必须想出一个变量名(呃),它也会在内存中
我有一个数组数组,想将元素附加到子数组。+=做我想做的,但我想了解为什么push不做。我期望的行为(并与+=一起工作):b=Array.new(3,[])b[0]+=["apple"]b[1]+=["orange"]b[2]+=["frog"]b=>[["苹果"],["橙子"],["Frog"]]通过推送,我将推送的元素附加到每个子数组(为什么?):a=Array.new(3,[])a[0].push("apple")a[1].push("orange")a[2].push("frog")a=>[[“苹果”、“橙子”、“Frog”]、[“苹果”、“橙子”、“Frog”]、[“苹果”、“
有没有办法让Ruby能够做这样的事情?classPlane@moved=0@x=0defx+=(v)#thisiserror@x+=v@moved+=1enddefto_s"moved#{@moved}times,currentxis#{@x}"endendplane=Plane.newplane.x+=5plane.x+=10putsplane.to_s#moved2times,currentxis15 最佳答案 您不能在Ruby中覆盖复合赋值运算符。任务在内部处理。您应该覆盖+,而不是+=。plane.a+=b与plane.a=
在尝试实现应用auto_orient的过程之后!对于我的图片,我收到此错误:ArgumentError(noimagesinthisimagelist):app/uploaders/image_uploader.rb:36:in`fix_exif_rotation'app/controllers/posts_controller.rb:12:in`create'Carrierwave在没有进程的情况下工作正常,但在添加进程后尝试上传图像时抛出错误。流程如下:process:fix_exif_rotationdeffix_exif_rotationmanipulate!do|image|
出于某种原因,heroku尝试要求dm-sqlite-adapter,即使它应该在这里使用Postgres。请注意,这发生在我打开任何URL时-而不是在gitpush本身期间。我构建了一个默认的Facebook应用程序。gem文件:source:gemcuttergem"foreman"gem"sinatra"gem"mogli"gem"json"gem"httparty"gem"thin"gem"data_mapper"gem"heroku"group:productiondogem"pg"gem"dm-postgres-adapter"endgroup:development,:t
我是Ruby和RubyonRails世界的新手。我已经阅读了一些指南,但我在使用以下语法时遇到了一些麻烦。我认为在Ruby中使用:condition语法来定义具有某种访问器的类属性,例如:classSampleattr_accessor:conditionend隐式声明“条件”属性的getter和setter。当我查看一些Rails示例代码时,我发现以下示例我并不完全理解。例如:@post=Post.find(params[:id])为什么它使用这种语法访问id属性,而不是:@post=Post.find(params[id])或者,例如:@posts=Post.find(:all):