- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我测试了一下select的性能,发现结果不是好的。 go版本是1.7.3
package main
import (
"fmt"
"log"
"os"
"runtime/pprof"
"time"
)
var serverDone = make(chan struct{})
var serverDone1 = make(chan struct{})
var serverDone2 = make(chan struct{})
var serverDone3 = make(chan struct{})
var serverDone4 = make(chan struct{})
var serverDone5 = make(chan struct{})
func main() {
f, err := os.Create("cpu.pprof")
if err != nil {
log.Fatal(err)
}
pprof.StartCPUProfile(f)
defer pprof.StopCPUProfile()
for i := 0; i < 1000; i++ {
go messageLoop()
}
<-time.After(10 * time.Second)
close(serverDone)
fmt.Println("finished")
}
func messageLoop() {
var ticker = time.NewTicker(100 * time.Millisecond)
defer ticker.Stop()
var counter = 0
for {
select {
case <-serverDone:
return
case <-serverDone1:
return
// case <-serverDone2:
// return
// case <-serverDone3:
// return
// case <-serverDone4:
// return
// case <-serverDone5:
// return
case <-ticker.C:
counter += 1
}
}
}
当运行上面的代码时,你会发现每次添加一个serverDone case时,CPU都会上升(在我的书中,大约是5%)。
去掉所有serverDone的情况下,CPU占用率在5%左右,不太好。
如果我将全局锁定对象(如 serverDone)转到本地,性能会更好,但仍然不够好。
谁知道我的情况有什么问题,或者 select 语句的正确用法是什么?
最佳答案
简答: channel 使用互斥锁。更多的 channel 意味着更多的futex
系统调用
这是程序上的 strace。
有 7 个 select 语句等待 7 个channels
% time seconds usecs/call calls errors syscall
------ ----------- ----------- --------- --------- ----------------
98.20 0.424434 13 33665 6061 futex
1.09 0.004731 10 466 sched_yield
0.47 0.002038 30 67 select
0.11 0.000484 4 114 rt_sigaction
0.05 0.000203 5 41 8 rt_sigreturn
0.03 0.000128 9 15 mmap
0.02 0.000081 27 3 clone
0.01 0.000052 7 8 rt_sigprocmask
0.01 0.000032 32 1 openat
0.00 0.000011 4 3 setitimer
0.00 0.000009 5 2 sigaltstack
0.00 0.000008 8 1 munmap
0.00 0.000006 6 1 execve
0.00 0.000006 6 1 sched_getaffinity
0.00 0.000004 4 1 arch_prctl
0.00 0.000004 4 1 gettid
0.00 0.000000 0 2 2 restart_syscall
------ ----------- ----------- --------- --------- ----------------
100.00 0.432231 34392 6071 total
包含 3 个 select 语句的代码等待 3 个channels
% time seconds usecs/call calls errors syscall
------ ----------- ----------- --------- --------- ----------------
90.47 0.118614 11 10384 1333 futex
6.64 0.008704 11 791 sched_yield
2.06 0.002706 23 120 select
0.39 0.000512 4 114 rt_sigaction
0.14 0.000181 8 22 2 rt_sigreturn
0.10 0.000131 9 15 mmap
0.05 0.000060 60 1 openat
0.04 0.000057 19 3 setitimer
0.04 0.000051 17 3 clone
0.03 0.000045 6 8 rt_sigprocmask
0.01 0.000009 9 1 execve
0.01 0.000009 5 2 sigaltstack
0.01 0.000009 9 1 sched_getaffinity
0.01 0.000008 8 1 munmap
0.01 0.000007 7 1 arch_prctl
0.00 0.000005 5 1 gettid
------ ----------- ----------- --------- --------- ----------------
100.00 0.131108 11468 1335 total
这里很明显,futex
调用的数量与 channel 的数量成正比,而 futex 系统调用是这种性能的原因。
这是对此的解释
您可以在以下文件中找到 channel 实现 src/runtime/chan.go .
这是 hchan
channel 的结构
type hchan struct {
qcount uint // total data in the queue
dataqsiz uint // size of the circular queue
buf unsafe.Pointer // points to an array of dataqsiz elements
elemsize uint16
closed uint32
elemtype *_type // element type
sendx uint // send index
recvx uint // receive index
recvq waitq // list of recv waiters
sendq waitq // list of send waiters
lock mutex
}
在 runtime2.go 中定义了一个 Lock 嵌入式结构,它根据操作系统用作互斥锁 (futex
) 或信号量。
因此随着 channel 数量的增加,会有更多的futex 系统调用
调用,这会影响性能
您可以在以下位置阅读更多相关信息:futex(2) , Channels in steroids
关于go - golang select语句在for循环中的性能,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42059800/
在 的 React 组件中菜单,我需要设置selected反射(reflect)应用程序状态的选项的属性。 在 render() , optionState从状态所有者传递给 SortMenu 组件
我是初级 Ruby-mysql 程序员,我想知道如何使我的(存储过程)查询结果更快.. 这是我的存储过程我正在使用 SQL_CACHE.. 但我不确定.. 缓存使我的过程更快.. : ( DROP
我一直在 Python 中进行套接字编程,以使用 select.select(rfile, wfile, xlist[, timeout]) 处理由已连接的客户端套接字列表发出的请求,并且我还想用 J
我试图通过用空格填充文本来创建下拉列表中的列效果,如下例所示: [Aux1+1] [*] [Aux1+1] [@Tn=PP] [Main] [*] [Main A
我为 MySQL 编写了以下查询: SELECT subquery.t1_column1, subquery.t2_id, MAX(subquery.val) FROM ( S
为什么我们要用 select 标签来编写.attr('selected','selected') 例如: $('#countryList option').filter(function () {
Lokalizacja: Gdańsk Rzeszów Wrocław 不知道发生了什么,但在那种情况下没有选择的选项,我必须从列表中选择一些东西。当我从选
我的表单中有两个选择字段。第一个是单选,另一个是多选。现在我想做的是根据单选中所选的选项,使用给定的数据选择多选中的选项。为此,我在单选更改时触发 ajax 请求: $.ajax({ type
我在 Firefox 5 中发现了一个奇怪的错误(我现在无法访问 4)。但是,我认为它可能在 Firefox 4 中工作,因为我刚买了一台新电脑,而且我不记得以前见过这个错误。 我有几个选择框。所选值
此 SQL 有何不同: 第一个: select * from table_1 a join table_2 b on a.id = b.acc_id 第二个: select * f
预选 的最佳做法是什么?在 ? 根据不同的网站,两者都有效。但是哪个更好呢?最兼容? Foo Bar 最佳答案 如果您正在编写 XHTML,则 selected="selected" 是必需的。 如
我使用 Angular JS 创建了一个多选选择框:下面是相同的代码: JS: $scope.foobars = [{ 'foobar_id': 'foobar01', 'name':
我在 jqGrid 中有几列 edittype="select"。如何读取特定行中当前选定值的选项值? 例如:当我提供以下选项时,如何获得 FedEx 等的“FE” editoption: { val
这是我更大问题的精简查询,但要点是我试图内部联接到一个选择,其中选择受到外部选择的限制。那可能吗?我在内部选择上收到有关多部分标识符 S.Item 和 S.SerialNum 的错误。 要点是这样的,
如果chat.chat_type IS NULL,我想选择user.*,但如果chat.chat_type = 1 我想选择组。* SELECT CASE WHEN ch
我正在编写一个小脚本来测试表单在提交之前是否已被更改。所以我可以使用普通输入(文本、文本区域等): if(element.defaultValue != element.value) { al
我正在尝试为 Prototype 编写一个插件,用户在其中单击下拉菜单并将其替换为多选元素。我快完成了。在用户选择他们想要显示的内容并将表单提交到同一页面之前,一切都很好。我正在使用 PHP 来使用
你如何在 MongoDB 中进行嵌套选择,类似于 SELECT id FROM table1 WHERE id IN (SELECT id FROM table2) 最佳答案 MongoDB 尚不具备
我有以下用于选择下拉列表的代码: {{unit.Text}} UnitOfMeasurements 数组中的每一项看起来像这样: Selected: false Text: "lb" Va
我正在尝试使用[选定]和[ngValue]来设置表单中包含对象的选择标记的默认值。但出于某种原因,它们似乎无法相提并论。。示例代码:。这段代码最终只显示空白作为缺省值。如果删除[ngValue],它就
我是一名优秀的程序员,十分优秀!