-
Notifications
You must be signed in to change notification settings - Fork 55
/
ThreadMonitor.java
530 lines (476 loc) · 19.3 KB
/
ThreadMonitor.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
/*
* Copyright (C) 2017. Aesean
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.aesean.activitystack.utils;
import android.os.Build;
import android.os.Debug;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Looper;
import android.text.TextUtils;
import android.util.Log;
import android.util.Printer;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
import java.util.regex.Pattern;
import androidx.annotation.NonNull;
/**
* ThreadMonitor
* 原理大致就是:通过{@link Looper#setMessageLogging(Printer)}方法,监测Handler打印消息之间间隔的时间,
* 来监控获取主线程执行持续时间,然后通过一个HandlerThread子线程来打印主线程堆栈信息。
* 由于大部分耗时操作都是通过子线程处理的,基本可以忽略当前监控服务对于主线程性能的影响。
* {@link #receiveDispatchingMessage()}方法会给WatchHandler发送一个开启监听的消息。
* WatchHandler会每隔{@link #mDumpStackDelayMillis}dump一次堆栈数据,
* 此时如果{@link #receiveFinishedMessage()}检测到没有超时,会通知WatchHandler取消继续dump
* (这里会有较小概率发生线程安全问题,明明正常finishMessage,WatchHandler还是打印了堆栈,但是不会无限打印)。
* 如果{@link #receiveFinishedMessage()}没有收到消息,或者收到超时消息,则忽略消息,WatchHandler
* 会自己处理超时打印(这里之所以这么做主要是避免线程同步)。
*
* @author xl
* @version V1.3
* @since 16/8/15
*/
@SuppressWarnings({"unused", "WeakerAccess"})
public class ThreadMonitor {
private static final String TAG = "ThreadMonitor";
private static final String HANDLER_THREAD_TAG = "watch_handler_thread";
/**
* 用于分割字符串,LogCat对打印的字符串长度有限制
*/
private static final String PART_SEPARATOR = "3664113077962208511";
/**
* 卡顿,单位毫秒,这个参数因为子线程也要用,为了避免需要线程同步,所以就static final了,自定义请直接修改这个值.
*/
private final long mBlockDelayMillis;
/**
* 最大处理次数,-1会无限打印,只有收到finish消息的时候才会终止打印。
*/
private final int mMaxPostTimes;
/**
* Dump堆栈数据时间间隔,单位毫秒
*/
private final long mDumpStackDelayMillis;
private final long mStartDumpStackDelayMillis;
private final long mSyncDelay;
private final boolean mPrintInDebuggerConnected;
private final IPrinter mPrinter;
/**
* 堆栈内容相同的情况下是否打印
*/
private final boolean mPrintSameStack;
private boolean mRunning = false;
private LooperProxy mTargetLooperProxy;
/**
* 起一个子线程,用来打印主线程的堆栈信息.因为是要监控主线程是否有卡顿的,所以主线程现在是无法打印堆栈的,
* 所以需要起一个子线程来打印主线程的堆栈信息.
*/
private HandlerThread mWatchThread;
private Handler mWatchHandler;
private PrintStaceInfoRunnable mPrintStaceInfoRunnable;
private long mReceiveDispatchingMessageTime;
private String mFilter;
private Printer mMessagePrinter;
private ThreadMonitor(Builder builder) {
mTargetLooperProxy = builder.looperProxy;
mBlockDelayMillis = builder.blockDelayMillis;
mMaxPostTimes = builder.maxPostTimes;
mDumpStackDelayMillis = builder.dumpStackDelayMillis;
mStartDumpStackDelayMillis = builder.startDumpStackDelayMillis;
mSyncDelay = builder.syncDelay;
mPrintInDebuggerConnected = builder.printInDebuggerConnected;
mPrintSameStack = builder.printSameStack;
mPrinter = builder.printer;
mFilter = builder.filter;
}
private ThreadMonitor(LooperProxy looperProxy, long blockDelayMillis, int maxPostTimes
, long dumpStackDelayMillis, long startDumpStackDelayMillis, long syncDelay
, boolean printInDebuggerConnected, boolean printSameStack, IPrinter printer) {
mTargetLooperProxy = looperProxy;
mBlockDelayMillis = blockDelayMillis;
mMaxPostTimes = maxPostTimes;
mDumpStackDelayMillis = dumpStackDelayMillis;
mStartDumpStackDelayMillis = startDumpStackDelayMillis;
mSyncDelay = syncDelay;
mPrintInDebuggerConnected = printInDebuggerConnected;
mPrintSameStack = printSameStack;
mPrinter = printer;
}
/**
* dump堆栈数据,会添加时间标签
*
* @param dumpThread 需要处理的线程
* @return 堆栈数据
*/
public static String dumpStackWithTimeHead(Thread dumpThread) {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss.SSS", Locale.getDefault());
String head = "\n" + dateFormat.format(new Date(System.currentTimeMillis())) + "时堆栈状态\n";
String dumpStack = dumpStack(dumpThread);
return head + dumpStack;
}
/**
* dump堆栈数据
*
* @param dumpThread 需要处理的线程
* @return 堆栈数据
*/
public static String dumpStack(Thread dumpThread) {
StackTraceElement[] stackTraceElements = dumpThread.getStackTrace();
// 注意这里仅仅是打印当前堆栈信息而已,实际代码不一定就是卡这里了.
// 比如此次Handler一共要处理三个方法
// method0(); 需要100ms
// method1(); 需要200ms
// method2(); 需要300ms
// 其实最佳方案是这三个方法全部打印,但从代码层面很难知道是这三个方法时候打印
// 只能每隔一段时间(比如:100ms)dump一次主线程堆栈信息,但是因为线程同步问题,可能第一个method0dump不到
String temp = "";
for (StackTraceElement stackTraceElement : stackTraceElements) {
String stack = stackTraceElement.toString() + "\n";
temp += stack;
}
if (TextUtils.isEmpty(temp)) {
temp = "null";
}
return temp;
}
public static ThreadMonitor getInstance() {
return InstanceHolder.sInstance;
}
private Printer createMessagePrinter() {
return new Printer() {
/**
* 纪录当前Printer回调的状态,注意这里初始状态必须是true.
*/
private boolean mPrinterStart = true;
@Override
public void println(String s) {
if (!mPrintInDebuggerConnected && Debug.isDebuggerConnected()) {
return;
}
// 默认不修正,如果实际出现错乱,可以调用下面方法进行修正。
// checkMessage(s);
// 这里因为Looper.loop方法内会在Handler开始和结束调用这个方法,所以这里也对应两个状态,start和finish
if (mPrinterStart) {
receiveDispatchingMessage();
} else {
receiveFinishedMessage();
}
mPrinterStart = !mPrinterStart;
}
/**
* 校验,这里默认收到消息是一次start,下一次finish,如果实际出现错乱,
* 可以根据{@link Looper#loop()}中logging.println打印的Message格式,进行一次修正。
*/
private void checkMessage(String s) {
if (s.startsWith(">>>>> Dispatching to")) {
mPrinterStart = true;
}
if (s.startsWith("<<<<< Finished to")) {
mPrinterStart = false;
}
}
};
}
private void receiveDispatchingMessage() {
mReceiveDispatchingMessageTime = System.currentTimeMillis();
// 如果有出现没有正确收到finishMessage的情况,这里可以每次start强制再处理一次。
// if (mPrintStaceInfoRunnable != null) {
// mPrintStaceInfoRunnable.cancel();
// mWatchHandler.removeCallbacks(mPrintStaceInfoRunnable);
// }
// 注意当前类所有代码,除了这个方法里的代码,其他全部是在主线程执行.
mPrintStaceInfoRunnable = new PrintStaceInfoRunnable();
mWatchHandler.postDelayed(mPrintStaceInfoRunnable, mStartDumpStackDelayMillis);
}
private void receiveFinishedMessage() {
long end = System.currentTimeMillis();
long delay = end - mReceiveDispatchingMessageTime;
if (delay >= mBlockDelayMillis) {
String msg = "(" + mPrintStaceInfoRunnable.getTag() + ") 检测到超时,App执行本次Handler消息消耗了:" + delay + "ms\n";
if (mPrinter != null) {
if (!mPrinter.print(msg)) {
Log.w(TAG, msg);
}
} else {
Log.w(TAG, msg);
}
}
// 这里是主线程,设置cancel后,没有做线程同步,子线程同步数据可能会有延迟。
mPrintStaceInfoRunnable.cancel();
mWatchHandler.removeCallbacks(mPrintStaceInfoRunnable);
mPrintStaceInfoRunnable = null;
}
public boolean isRunning() {
return mRunning;
}
public void install() {
if (mWatchThread != null || mWatchHandler != null) {
throw new RuntimeException("请勿重复install。如果需要释放资源,请调用release方法。");
}
mWatchThread = new HandlerThread(HANDLER_THREAD_TAG);
mWatchThread.start();
mWatchHandler = new Handler(mWatchThread.getLooper());
mMessagePrinter = createMessagePrinter();
mTargetLooperProxy.addMessageLogging(mMessagePrinter);
mRunning = true;
}
public void release() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
mWatchThread.quitSafely();
} else {
mWatchThread.quit();
}
mWatchThread = null;
mWatchHandler = null;
mTargetLooperProxy.removeMessageLogging(mMessagePrinter);
mTargetLooperProxy = null;
mRunning = false;
}
public LooperProxy getTargetLooperProxy() {
return mTargetLooperProxy;
}
public interface IPrinter {
boolean print(String msg);
}
public static class Builder {
private long blockDelayMillis = 400;
private int maxPostTimes = -1;
private long dumpStackDelayMillis = 100;
private long startDumpStackDelayMillis = 20;
private long syncDelay = 0;
private boolean printInDebuggerConnected = true;
private boolean printSameStack = false;
private IPrinter printer = null;
private LooperProxy looperProxy;
private String filter;
/**
* 卡顿,单位毫秒,这个参数因为子线程也要用,为了避免需要线程同步,所以就static final了,自定义请直接修改这个值.
*
* @param blockDelayMillis 卡顿间隔,单位毫秒
*/
public Builder setBlockDelayMillis(long blockDelayMillis) {
this.blockDelayMillis = blockDelayMillis;
return this;
}
/**
* 最大处理次数,-1会无限打印,只有收到finish消息的时候才会终止打印。
*
* @param maxPostTimes 次数,-1无限打印
*/
public Builder setMaxPostTimes(int maxPostTimes) {
this.maxPostTimes = maxPostTimes;
return this;
}
/**
* Dump堆栈数据时间间隔,单位毫秒
*
* @param dumpStackDelayMillis Dump堆栈数据时间间隔,单位毫秒
*/
public Builder setDumpStackDelayMillis(long dumpStackDelayMillis) {
this.dumpStackDelayMillis = dumpStackDelayMillis;
return this;
}
/**
* 第一次开始Dump堆栈延迟
*
* @param startDumpStackDelayMillis 延迟,单位毫秒
*/
public Builder setStartDumpStackDelayMillis(long startDumpStackDelayMillis) {
this.startDumpStackDelayMillis = startDumpStackDelayMillis;
return this;
}
/**
* 跨线程同步延迟
*
* @param syncDelay 同步延迟,单位毫秒
*/
public Builder setSyncDelay(long syncDelay) {
this.syncDelay = syncDelay;
return this;
}
/**
* 是否在debug状态继续打印。注意,因为debug时候代码的执行会跟着调试节奏走,
* 所以可能会出现一个方法执行时间很长的情况,如果不希望在debug时候打印就设置false
*
* @param printInDebuggerConnected true,debug状态继续打印,false不打印
*/
public Builder setPrintInDebuggerConnected(boolean printInDebuggerConnected) {
this.printInDebuggerConnected = printInDebuggerConnected;
return this;
}
/**
* 是否打印相同堆栈数据。
*
* @param printSameStack true,堆栈数据相同继续打印,false,只打印不同的堆栈数据
*/
public Builder setPrintSameStack(boolean printSameStack) {
this.printSameStack = printSameStack;
return this;
}
/**
* 设置需要监控的线程
* {@link #setLooperProxy(LooperProxy)}
*
* @param looper 需要监控线程的Looper
*/
@Deprecated
public Builder setLooper(Looper looper) {
this.looperProxy = new LooperProxy(looper);
return this;
}
public Builder setLooperProxy(LooperProxy looperProxy) {
this.looperProxy = looperProxy;
return this;
}
public Builder setFilter(String filter) {
this.filter = filter;
return this;
}
/**
* 构建BlockUtils对象
*
* @return ThreadMonitor
*/
public ThreadMonitor builder() {
if (looperProxy == null) {
throw new NullPointerException("looperProxy can't be null. Please call setLooperProxy(). ");
}
return new ThreadMonitor(this);
}
}
private static class InstanceHolder {
private static final ThreadMonitor sInstance = new Builder().setLooper(Looper.getMainLooper()).builder();
}
private class PrintStaceInfoRunnable implements Runnable {
private String mTag = String.valueOf(this.hashCode());
private long mStartMillis = System.currentTimeMillis();
private int mPostTimes = 0;
private String mLastDumpStack;
private int mTimeoutTimes = 0;
private boolean mCancel = false;
private StringBuilder mStackInfo = new StringBuilder();
private Thread mDumpThread = mTargetLooperProxy.getThread();
private boolean isTimeOut() {
return System.currentTimeMillis() - mStartMillis > mBlockDelayMillis + mSyncDelay;
}
private String getTag() {
return mTag;
}
private void recordTimes() {
mPostTimes++;
if (isTimeOut()) {
mTimeoutTimes++;
}
}
private void cancel() {
mCancel = true;
}
private boolean isFirstTimeout() {
return mTimeoutTimes == 1;
}
private String newTimeHead() {
return "\n(" + getTag() + ") " + (System.currentTimeMillis() - mStartMillis) + "ms时堆栈状态\n";
}
private boolean equalsLastDump(String stack) {
//noinspection StringEquality
return (mLastDumpStack == stack) || (mLastDumpStack != null && mLastDumpStack.equals(stack));
}
private String getFilter() {
return mFilter;
}
private String dumpStack() {
String stack = ThreadMonitor.dumpStack(getDumpThread()) + "\n";
String[] split = stack.split("\n");
StringBuilder result = new StringBuilder();
String filter = getFilter();
if (filter != null && !filter.equals("")) {
for (int i = 0; i < split.length; i++) {
if (Pattern.matches(filter, split[i])) {
result.append(split[i]).append("\n");
}
}
} else {
for (int i = 0; i < split.length; i++) {
result.append(split[i]).append("\n");
}
}
return result.toString();
}
@Override
public void run() {
if (mCancel) {
return;
}
recordTimes();
//noinspection ConstantConditions
if (mPostTimes < mMaxPostTimes || mMaxPostTimes == -1) {
mWatchHandler.postDelayed(this, mDumpStackDelayMillis);
}
String timeHead = newTimeHead();
final String dumpStack = dumpStack();
final boolean equalsLastDump = equalsLastDump(dumpStack);
if (isTimeOut()) {
String stack = "";
// 超时打印
if (isFirstTimeout()) {
// 取出前面的数据
stack = mStackInfo.toString();
mStackInfo.delete(0, mStackInfo.length());
// 前面数据也可能为空
if (!TextUtils.isEmpty(stack)) {
stack += PART_SEPARATOR;
}
stack += timeHead + dumpStack;
} else {
if (mPrintSameStack || !equalsLastDump) {
stack += timeHead + dumpStack;
}
}
// 非空打印
if (!TextUtils.isEmpty(stack)) {
printStackTraceInfo(stack);
}
} else {
if (mPrintSameStack || !equalsLastDump) {
mStackInfo.append(PART_SEPARATOR).append(timeHead).append(dumpStack);
}
}
mLastDumpStack = dumpStack;
}
@NonNull
private Thread getDumpThread() {
return mDumpThread;
}
private void printStackTraceInfo(String info) {
String[] split = info.split(PART_SEPARATOR);
if (mPrinter == null) {
for (String s : split) {
if (!TextUtils.isEmpty(s)) {
Log.d(TAG, s + "\n");
}
}
} else {
for (String s : split) {
if (!TextUtils.isEmpty(s)) {
String msg = s + "\n";
if (!mPrinter.print(msg)) {
Log.d(TAG, msg);
}
}
}
}
}
}
}