Skip to content
项目
群组
代码片段
帮助
正在加载...
帮助
为 GitLab 提交贡献
登录/注册
切换导航
H
h2database
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
分枝图
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
计划
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
分枝图
统计图
创建新议题
作业
提交
议题看板
打开侧边栏
Administrator
h2database
Commits
43cad5c9
提交
43cad5c9
authored
14 年前
作者:
Thomas Mueller
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
The built-in Profiler now also lists ticks per package.
上级
c2b3b4c2
隐藏空白字符变更
内嵌
并排
正在显示
1 个修改的文件
包含
77 行增加
和
38 行删除
+77
-38
Profiler.java
h2/src/main/org/h2/util/Profiler.java
+77
-38
没有找到文件。
h2/src/main/org/h2/util/Profiler.java
浏览文件 @
43cad5c9
...
...
@@ -17,6 +17,7 @@ import org.h2.engine.Constants;
* A simple CPU profiling tool similar to java -Xrunhprof.
*/
public
class
Profiler
implements
Runnable
{
private
static
Instrumentation
instrumentation
;
private
static
final
int
MAX_ELEMENTS
=
1000
;
...
...
@@ -25,15 +26,21 @@ public class Profiler implements Runnable {
public
boolean
paused
;
private
String
[]
ignoreLines
=
StringUtils
.
arraySplit
(
""
,
','
,
true
);
private
String
[]
ignorePackages
=
StringUtils
.
arraySplit
(
"java,"
+
"sun,"
+
"com.sun.,"
,
','
,
true
);
private
String
[]
ignoreThreads
=
StringUtils
.
arraySplit
(
"java.lang.Object.wait,"
+
"java.lang.Thread.dumpThreads,"
+
"java.lang.Thread.getThreads,"
+
"java.net.PlainSocketImpl.socketAccept,"
+
"java.net.SocketInputStream.socketRead0,"
+
"java.net.SocketOutputStream.socketWrite0,"
+
"java.lang.UNIXProcess.waitForProcessExit,"
+
"java.lang.Object.wait,"
+
"java.lang.Thread.sleep,"
+
"java.lang.UNIXProcess.waitForProcessExit,"
+
"java.net.PlainSocketImpl.accept,"
+
"java.net.PlainSocketImpl.socketAccept,"
+
"java.net.SocketInputStream.socketRead,"
+
"java.net.SocketOutputStream.socketWrite,"
+
"sun.awt.windows.WToolkit.eventLoop,"
+
"sun.misc.Unsafe.park,"
+
"dalvik.system.VMStack.getThreadStackTrace,"
+
...
...
@@ -41,6 +48,7 @@ public class Profiler implements Runnable {
,
','
,
true
);
private
volatile
boolean
stop
;
private
HashMap
<
String
,
Integer
>
counts
=
new
HashMap
<
String
,
Integer
>();
private
HashMap
<
String
,
Integer
>
packages
=
new
HashMap
<
String
,
Integer
>();
private
int
minCount
=
1
;
private
int
total
;
private
Thread
thread
;
...
...
@@ -120,62 +128,74 @@ public class Profiler implements Runnable {
continue
;
}
StackTraceElement
[]
dump
=
entry
.
getValue
();
if
(
dump
.
length
==
0
)
{
if
(
dump
==
null
||
dump
.
length
==
0
)
{
continue
;
}
boolean
ignoreThis
=
false
;
for
(
String
ig
:
ignoreThreads
)
{
if
(
ig
.
length
()
>
0
&&
dump
[
0
].
toString
().
startsWith
(
ig
))
{
ignoreThis
=
true
;
break
;
}
}
if
(
ignoreThis
)
{
if
(
startsWithAny
(
dump
[
0
].
toString
(),
ignoreThreads
))
{
continue
;
}
StringBuilder
buff
=
new
StringBuilder
();
// simple recursive calls are ignored
String
last
=
null
;
boolean
packageCounts
=
false
;
for
(
int
j
=
0
,
i
=
0
;
i
<
dump
.
length
&&
j
<
depth
;
i
++)
{
String
el
=
dump
[
i
].
toString
();
ignoreThis
=
false
;
for
(
String
ig
:
ignoreLines
)
{
if
(
ig
.
length
()
>
0
&&
el
.
startsWith
(
ig
))
{
ignoreThis
=
true
;
break
;
}
}
if
(!
ignoreThis
&&
!
el
.
equals
(
last
))
{
if
(!
el
.
equals
(
last
)
&&
!
startsWithAny
(
el
,
ignoreLines
))
{
last
=
el
;
buff
.
append
(
"at "
).
append
(
el
).
append
(
SysProperties
.
LINE_SEPARATOR
);
if
(!
packageCounts
&&
!
startsWithAny
(
el
,
ignorePackages
))
{
packageCounts
=
true
;
int
index
=
0
;
for
(;
index
<
el
.
length
();
index
++)
{
char
c
=
el
.
charAt
(
index
);
if
(
Character
.
isUpperCase
(
c
)
||
c
==
'('
)
{
break
;
}
}
if
(
index
>
0
&&
el
.
charAt
(
index
-
1
)
==
'.'
)
{
index
--;
}
String
packageName
=
el
.
substring
(
0
,
index
);
increment
(
packages
,
packageName
,
0
);
}
j
++;
}
}
if
(
buff
.
length
()
>
0
)
{
increment
(
buff
.
toString
());
minCount
=
increment
(
counts
,
buff
.
toString
().
trim
(),
minCount
);
total
++;
}
}
}
private
void
increment
(
String
trace
)
{
total
++;
Integer
oldCount
=
counts
.
get
(
trace
);
private
boolean
startsWithAny
(
String
s
,
String
[]
prefixes
)
{
for
(
String
p
:
prefixes
)
{
if
(
p
.
length
()
>
0
&&
s
.
startsWith
(
p
))
{
return
true
;
}
}
return
false
;
}
private
static
int
increment
(
HashMap
<
String
,
Integer
>
map
,
String
trace
,
int
minCount
)
{
Integer
oldCount
=
map
.
get
(
trace
);
if
(
oldCount
==
null
)
{
counts
.
put
(
trace
,
1
);
map
.
put
(
trace
,
1
);
}
else
{
counts
.
put
(
trace
,
oldCount
+
1
);
map
.
put
(
trace
,
oldCount
+
1
);
}
if
(
counts
.
size
()
>
MAX_ELEMENTS
)
{
for
(
Iterator
<
Map
.
Entry
<
String
,
Integer
>>
ei
=
counts
.
entrySet
().
iterator
();
ei
.
hasNext
();)
{
while
(
map
.
size
()
>
MAX_ELEMENTS
)
{
for
(
Iterator
<
Map
.
Entry
<
String
,
Integer
>>
ei
=
map
.
entrySet
().
iterator
();
ei
.
hasNext
();)
{
Map
.
Entry
<
String
,
Integer
>
e
=
ei
.
next
();
if
(
e
.
getValue
()
<=
minCount
)
{
ei
.
remove
();
}
}
if
(
counts
.
size
()
>
MAX_ELEMENTS
)
{
if
(
map
.
size
()
>
MAX_ELEMENTS
)
{
minCount
++;
}
}
return
minCount
;
}
/**
...
...
@@ -190,12 +210,20 @@ public class Profiler implements Runnable {
buff
.
append
(
"Profiler: top "
).
append
(
count
).
append
(
" stack trace(s) of "
).
append
(
time
).
append
(
" ms [build-"
).
append
(
Constants
.
BUILD_ID
).
append
(
"]:"
).
append
(
SysProperties
.
LINE_SEPARATOR
);
if
(
counts
.
size
()
==
0
)
{
buff
.
append
(
"(none)"
);
buff
.
append
(
"(none)"
)
.
append
(
SysProperties
.
LINE_SEPARATOR
)
;
}
appendTop
(
buff
,
counts
,
count
,
total
,
false
);
buff
.
append
(
"packages:"
).
append
(
SysProperties
.
LINE_SEPARATOR
);
appendTop
(
buff
,
packages
,
count
,
total
,
true
);
buff
.
append
(
'.'
);
return
buff
.
toString
();
}
private
static
void
appendTop
(
StringBuilder
buff
,
HashMap
<
String
,
Integer
>
map
,
int
count
,
int
total
,
boolean
table
)
{
for
(
int
x
=
0
,
min
=
0
;;)
{
int
highest
=
0
;
Map
.
Entry
<
String
,
Integer
>
best
=
null
;
for
(
Map
.
Entry
<
String
,
Integer
>
el
:
counts
.
entrySet
())
{
for
(
Map
.
Entry
<
String
,
Integer
>
el
:
map
.
entrySet
())
{
if
(
el
.
getValue
()
>
highest
)
{
best
=
el
;
highest
=
el
.
getValue
();
...
...
@@ -204,18 +232,29 @@ public class Profiler implements Runnable {
if
(
best
==
null
)
{
break
;
}
counts
.
remove
(
best
.
getKey
());
map
.
remove
(
best
.
getKey
());
if
(++
x
>=
count
)
{
if
(
best
.
getValue
()
<
min
)
{
break
;
}
min
=
best
.
getValue
();
}
buff
.
append
(
best
.
getValue
()).
append
(
'/'
).
append
(
total
).
append
(
SysProperties
.
LINE_SEPARATOR
).
append
(
best
.
getKey
());
int
c
=
best
.
getValue
();
int
percent
=
100
*
c
/
Math
.
max
(
total
,
1
);
if
(
table
)
{
if
(
percent
>
1
)
{
buff
.
append
(
percent
).
append
(
"%: "
).
append
(
best
.
getKey
()).
append
(
SysProperties
.
LINE_SEPARATOR
);
}
}
else
{
buff
.
append
(
c
).
append
(
'/'
).
append
(
total
).
append
(
" ("
).
append
(
percent
).
append
(
"%):"
).
append
(
SysProperties
.
LINE_SEPARATOR
).
append
(
best
.
getKey
()).
append
(
SysProperties
.
LINE_SEPARATOR
);
}
}
buff
.
append
(
'.'
);
return
buff
.
toString
();
}
}
This diff is collapsed.
Click to expand it.
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论