aboutsummaryrefslogtreecommitdiff
path: root/files/zh-cn/web/api/background_tasks_api/index.html
blob: 4407043fc306e8a8adf475af2e56b0355d409ba2 (plain)
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
---
title: Background Tasks API
slug: Web/API/Background_Tasks_API
tags:
  - API
  - 后台任务API
  - 指南
  - 概述
translation_of: Web/API/Background_Tasks_API
---
<div>{{DefaultAPISidebar("Background Tasks")}}{{draft}}</div>

<p><strong>幕后任务协作调度 API</strong> (也叫幕后任务 API 或者简单称为 <code>requestIdleCallback()</code> API) 提供了由用户代理决定,在空闲时间自动执行队列任务的能力。</p>

<h2 id="概念和用法">概念和用法</h2>

<p>浏览器的主线程以其事件循环队列为中心。此代码渲染 {{domxref("Document")}} 上待更新展示的内容,执行页面待运行的 JavaScript 脚本,接收来自输入设备的事件,以及分发事件给需要接收事件的元素。此外,事件循环队列处理与操作系统的交互、浏览器自身用户界面的更新等等。这是一个非常繁忙的代码块,您的主要 JavaScript 代码可能会和这些代码一起也在这个线程中执行。当然,大多数(不是所有)能够更改 DOM 的代码都在主线程中运行,因为用户界面更改通常只对主线程可用。</p>

<p>因为事件处理和屏幕更新是用户关注性能最明显的两种方式。对于您的代码来说,防止在事件队列中出现卡顿是很重要的。在过去,除了编写尽可能高效的代码和将尽可能多的工作移交给 <a href="/en-US/docs/Web/API/Web_Workers_API">workers</a> 之外,没有其他可靠的方法可以做到这一点。 {{domxref("Window.requestIdleCallback()")}} 允许浏览器告诉您的代码可以安全使用多少时间而不会导致系统延迟,从而有助于确保浏览器的事件循环平稳运行。如果您保持在给定的范围内,您可以使用户体验更好。</p>

<h3 id="充分利用空闲回调">充分利用空闲回调</h3>

<p>因为 idle callbacks 旨在为代码提供一种与事件循环协作的方式,以确保系统充分利用其潜能,不会过度分配任务,从而导致延迟或其他性能问题,因此您应该考虑如何使用它。</p>

<ul>
 <li><strong>对非高优先级的任务使用空闲回调。</strong> 已经创建了多少回调,用户系统的繁忙程度,你的回调多久会执行一次(除非你指定了 <code>timeout</code>),这些都是未知的。不能保证每次事件循环(甚至每次屏幕更新)后都能执行空闲回调;如果事件循环用尽了所有可用时间,那你可就倒霉了(再说一遍,除非你用了 <code>timeout</code>)。 </li>
 <li><strong>空闲回调应尽可能不超支分配到的时间。</strong>尽管即使你超出了规定的时间上限,通常来说浏览器、代码、网页也能继续正常运行,这里的时间限制是用来保证系统能留有足够的时间去完成当前的事件循环然后进入下一个循环,而不会导致其他代码卡顿或动画效果延迟。目前,{{domxref("IdleDeadline.timeRemaining", "timeRemaining()")}} 有一个50 ms 的上限时间,但实际上你能用的时间比这个少,因为在复杂的页面中事件循环可能已经花费了其中的一部分,浏览器的扩展插件也需要处理时间,等等。</li>
 <li><strong>避免在空闲回调中改变 DOM。 </strong>空闲回调执行的时候,当前帧已经结束绘制了,所有布局的更新和计算也已经完成。如果你做的改变影响了布局, 你可能会强制停止浏览器并重新计算,而从另一方面来看,这是不必要的。 如果你的回调需要改变DOM,它应该使用{{domxref("Window.requestAnimationFrame()")}}来调度它。</li>
 <li><strong>避免运行时间无法预测的任务。</strong> 你的空闲回调必须避免做任何占用时间不可预测的事情。比如说,应该避免做任何会影响页面布局的事情。你也必须避免 执行{{domxref("Promise")}}<code>resolve</code><code>reject</code>,因为这会在你的回调函数返回后立即引用Promise对象对<code>resolve</code><code>reject</code>的处理程序。</li>
 <li><strong>在你需要的时候要用 timeout,但记得只在需要的时候才用。</strong> 使用 timeout可以保证你的代码按时执行,但是在剩余时间不足以强制执行你的代码的同时保证浏览器的性能表现的情况下,timeout就会造成延迟或者动画不流畅。</li>
</ul>

<h3 id="回退到_setTimeout">回退到 setTimeout</h3>

<p>因为后台任务API还是相当新的,而你的代码可能需要在那些不仍不支持此API的浏览器上运行。你可以把 {{domxref("WindowTimers.setTimeout()", "setTimeout()")}} 用作回调选项来做这样的事。这个并不是 {{Glossary("polyfill")}} ,因为它在功能上并不相同; <code>setTimeout()</code> 并不会让你利用空闲时段,而是使你的代码在情况允许时执行你的代码,以使我们可以尽可能地避免造成用户体验性能表现延迟的后果。</p>

<pre class="brush: js">window.requestIdleCallback = window.requestIdleCallback || function(handler) {
  let startTime = Date.now();

  return setTimeout(function() {
    handler({
      didTimeout: false,
      timeRemaining: function() {
        return Math.max(0, 50.0 - (Date.now() - startTime));
      }
    });
  }, 1);
}</pre>

<p>如果 {{domxref("Window.requestIdleCallback", "window.requestIdleCallback")}} 是undefined, 我们在这里把它创建出来。这个函数首先会记录我们调用具体实现的时间。我们将用它计算填充程序{{domxref("IdleDeadline.timeRemaining()", "timeRemaining()")}}返回的值 。</p>

<p>接着,我们调用 {{domxref("WindowTimers.setTimeout", "setTimeout()")}},并给它传一个函数,在这个函数里,我们传给<code>requestIdleCallback()</code>的具体实现的回调会得以执行。这个回调会接收一个和{{domxref("IdleDeadline")}}相符合的object,此object的 {{domxref("IdleDeadline.didTimeout", "didTimeout")}}被设定为false,并拥有一个{{domxref("IdleDeadline.timeRemaining", "timeRemaining()")}} 方法,用来给回调函数50毫秒的开始时间。每次调用<code>timeRemaining()</code>,它都会从开始的50毫秒中减去已逝去的时间,来确定还剩余的时间。</p>

<p>结果是,虽然我们的填充程序不会像真正的<code>requestIdleCallback()</code>将自己限制在当前事件循环传递中的空闲时间内,但它至少将每次传递的运行时间限制为不超过50毫秒。</p>

<p>我们{{domxref("Window.cancelIdleCallback", "cancelIdleCallback()")}}的具体实现要简单的多。</p>

<pre class="brush: js">window.cancelIdleCallback = window.cancelIdleCallback || function(id) {
  clearTimeout(id);
}</pre>

<p>如果<code>cancelIdleCallback()</code>没有定义,它将创建一个来简单地把指定回调ID传递给{{domxref("WindowTimers.clearTimeout", "clearTimeout()")}}</p>

<p>现在,尽管效率不高,你的代码也可以在不支持后台任务API的浏览器上运行了。</p>

<h2 id="接口">接口</h2>

<p>后台任务API只添加了一个新的接口:</p>

<dl>
 <dt>{{domxref("IdleDeadline")}}</dt>
 <dd>这个类型的对象接口空闲回调以提供空闲时段的预估持续时长,以及回调是否因为定时时段过期使其正在运行当中。</dd>
</dl>

<p>这个API给 {{domxref("Window")}} 接口增加了新的 {{domxref("window.requestIdleCallback", "requestIdleCallback()")}}{{domxref("window.cancelIdleCallback", "cancelIdleCallback()")}} 方法。</p>

<h2 id="Example" name="Example">示例</h2>

<p>在这个示例中,我们将了解我们怎么用{{domxref("window.requestIdleCallback", "requestIdleCallback()")}}来在浏览器空闲时运行高耗时、低优先级的任务。此外,这个示例会演示如何使用{{domxref("window.requestAnimationFrame", "requestAnimationFrame()")}}安排文档内容的更新。</p>

<p>在下面,你只会看到示例的HTML和JavaScript。CSS没有展示出来,因为它对理解此功能并不关键。</p>

<h3 id="HTML_内容">HTML 内容</h3>

<p>为了了解我们的目标,看一下HTML。这里创建了一个盒子 (ID <code>"Container"</code>)来显示操作进度(因为毕竟我们没法知道解码“ <span class="tlid-translation translation" lang="zh-CN"><span title="">量子丝极谱发射</span></span> ”会用多长时间),还创建了一个次要的盒子 (ID <code>"logBox"</code>)来展示文本输出。</p>

<pre>&lt;p&gt;
  演示使用 &lt;a href="https://developer.mozilla.org/en-US/docs/Web/API/Background_Tasks_API"&gt;
   协作调度幕后任务 &lt;/a&gt; 使用 &lt;code&gt;requestIdleCallback()&lt;/code&gt;
  方法.
&lt;/p&gt;

&lt;div class="container"&gt;
  &lt;div class="label"&gt;解码<span class="tlid-translation translation" lang="zh-CN"><span title="">量子丝极谱发射</span></span>中...&lt;/div&gt;
  &lt;progress id="progress" value="0"&gt;&lt;/progress&gt;
  &lt;div class="button" id="startButton"&gt;
    开始
  &lt;/div&gt;
  &lt;div class="label counter"&gt;
    任务 &lt;span id="currentTaskNumber"&gt;0&lt;/span&gt; / &lt;span id="totalTaskCount"&gt;0&lt;/span&gt;
  &lt;/div&gt;
&lt;/div&gt;

&lt;div class="logBox"&gt;
  &lt;div class="logHeader"&gt;
    记录
  &lt;/div&gt;
  &lt;div id="log"&gt;
  &lt;/div&gt;
&lt;/div&gt;</pre>

<p>这个进度框用一个 {{HTMLElement("progress")}} 元素展示进度,随着它标签部分的变化,会呈现进度的数字信息。此外,这还有一个开始按钮(id为'startButton'),用户可以使用它开始数据处理。</p>

<div class="hidden">
<h3 id="CSS_content">CSS content</h3>

<pre class="brush: css">body {
  font-family: "Open Sans", "Lucida Grande", "Arial", sans-serif;
  font-size: 16px;
}

.logBox {
  margin-top: 16px;
  width: 400px;
  height:500px;
  border-radius: 6px;
  border: 1px solid black;
  box-shadow: 4px 4px 2px black;
}

.logHeader {
  margin: 0;
  padding: 0 6px 4px;
  height: 22px;
  background-color: lightblue;
  border-bottom: 1px solid black;
  border-radius: 6px 6px 0 0;
}

#log {
  font: 12px "Courier", monospace;
  padding: 6px;
  overflow: auto;
  overflow-y: scroll;
  width: 388px;
  height: 460px;
}

.container {
  width: 400px;
  padding: 6px;
  border-radius: 6px;
  border: 1px solid black;
  box-shadow: 4px 4px 2px black;
  display: block;
  overflow: auto;
}

.label {
  display: inline-block;
}

.counter {
  text-align: right;
  padding-top: 4px;
  float: right;
}

.button {
  padding-top: 2px;
  padding-bottom: 4px;
  width: 100px;
  display: inline-block;
  float: left;
  border: 1px solid black;
  cursor: pointer;
  text-align: center;
  margin-top: 0;
  color: white;
  background-color: darkgreen;
}

#progress {
  width: 100%;
  padding-top: 6px;
}</pre>
</div>

<h3 id="JavaScript_内容">JavaScript 内容</h3>

<p>现在,已经定义了文档结构,再构造出JavaS代码就可以运行了。目标:可以<span class="tlid-translation translation" lang="zh-CN"><span title="">向队列添加调用函数的请求,并具有一个空闲回调,空闲回调会在系统空闲且空闲时间足够长以取得进展时运行。</span></span></p>

<h4 id="变量声明">变量声明</h4>

<pre class="brush: js">let taskList = [];
let totalTaskCount = 0;
let currentTaskNumber = 0;
let taskHandle = null;
</pre>

<p>这些变量用于管理等待执行的任务列表和任务队列和其执行的状态信息:</p>

<ul>
 <li><code>taskList</code> 是一个对象的 {{jsxref("Array")}} , 每个对象代表一个待运行的任务。</li>
 <li><code>totalTaskCount</code> 是一个已被添加到队列的任务数量计数器,只会增大,不会减小。我们用它计算总工作量进度的百分比值。</li>
 <li><code>currentTaskNumber</code> 用于追踪到现在为止已处理了多少任务。</li>
 <li><code>taskHandle</code> 是对当前处理中任务的一个引用。</li>
</ul>

<pre class="brush: js">let totalTaskCountElem = document.getElementById("totalTaskCount");
let currentTaskNumberElem = document.getElementById("currentTaskNumber");
let progressBarElem = document.getElementById("progress");
let startButtonElem = document.getElementById("startButton");
let logElem = document.getElementById("log");
</pre>

<p>接下来我们有了引用要交互DOM元素的变量。这些元素是:</p>

<ul>
 <li><code>totalTaskCountElem</code>  {{HTMLElement("span")}} 用于插入我们在进度框显示状态中创建的任务总数。</li>
 <li><code>currentTaskNumberElem</code> 是我们用来呈现到当前为止处理过的任务数的元素。</li>
 <li><code>progressBarElem</code>  {{HTMLElement("progress")}} ,我们用它来呈现到当前为止处理过任务的百分比。</li>
 <li><code>startButtonElem</code> 是开始按钮。</li>
 <li><code>logElem</code> 我们在 {{HTMLElement("div")}} 中显示记录过的文本信息。</li>
</ul>

<pre class="brush: js">let logFragment = null;
let statusRefreshScheduled = false;
</pre>

<p>最后,我们为其他项目设置一对变量:</p>

<ul>
 <li><code>logFragment</code> 当渲染下一帧,我们的记录方法都会生成一个 {{domxref("DocumentFragment")}} 来创建添加到记录的内容,并保存到<code>logFragment</code>{{domxref("DocumentFragment")}}</li>
 <li><code>statusRefreshScheduled </code>我们用它来追踪我们是否已经为即将到来的帧安排了状态显示框的更新,所以我们每一帧只执行一次。</li>
</ul>

<div class="hidden">
<p>The shim to function even if idle callbacks aren't supported. Already discussed above, so it's hidden here to save space in the article.</p>

<pre class="brush: js">window.requestIdleCallback = window.requestIdleCallback || function(handler) {
  let startTime = Date.now();

  return setTimeout(function() {
    handler({
      didTimeout: false,
      timeRemaining: function() {
        return Math.max(0, 50.0 - (Date.now() - startTime));
      }
    });
  }, 1);
};

window.cancelIdleCallback = window.cancelIdleCallback || function(id) {
  clearTimeout(id);
};
</pre>
</div>

<h4 id="管理任务队列">管理任务队列</h4>

<p>接下来,让我们来了解我们管理需要执行的任务的方式。为此,我们将创建一个先进先出(FIFO)的任务队列,在空闲回调期间,如果时间允许,我们将执行这个队列。</p>

<h5 id="排队任务">排队任务</h5>

<p>首先,我们需要一个函数把任务排成队列,以便将来执行。这个函数<code>enqueueTask()</code>,就像这个:</p>

<pre class="brush: js">function enqueueTask(taskHandler, taskData) {
  taskList.push({
    handler: taskHandler,
    data: taskData
  });

  totalTaskCount++;

  if (!taskHandle) {
    taskHandle = requestIdleCallback(runTaskQueue, { timeout: 1000 });
  }

  scheduleStatusRefresh();
}
</pre>

<p><code>enqueueTask()</code>接受两个参数作为参数</p>

<ul>
 <li><code>taskHandler</code> 一个函数,被调用来处理任务。</li>
 <li><code>taskData</code> is 一个对象(object),被当作输入参数传递给<code>taskHandler</code>,以允许任务接收自定义数据。</li>
</ul>

<p>为了把任务排成队列,我们把一个对象(object)<a href="/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/push">push</a><code>taskList</code>数组;此对象包含<code>taskHandler</code><code>taskData</code>的值(命名分别是<code>handler</code><code>data</code>),然后体现我们队列里任务总数的<code>totalTaskCount</code>增加(我们不会在从队列中移除任务时减少<code>totalTaskCount</code>)。</p>

<p>接下来,我们来检查我们是否已经创建了一个空闲回调;如果<code>taskHandle</code>是0,那我们得知还没有空闲回调,所以我们调用{{domxref("Window.requestIdleCallback", "requestIdleCallback()")}}去创建一个。它被配置为调用一个叫<code>runTaskQueue()</code>的函数(我们随后会对其研究),它的<code>timeout</code>为1秒,因此,即使没有任何实际可用的空闲时间,它也至少会每秒运行一次。</p>

<h5 id="执行任务">执行任务</h5>

<p>我们的空闲回调处理方法,<code>runTaskQueue()</code>,将在浏览器确定有足够的可用空闲时间让我们做一些我们的工作时,或者1秒的<code>timeout</code>到期时被调用。这个方法的作用是执行队列中的任务。</p>

<pre class="brush: js">function runTaskQueue(deadline) {
  while ((deadline.timeRemaining() &gt; 0 || deadline.didTimeout) &amp;&amp; taskList.length) {
    let task = taskList.shift();
    currentTaskNumber++;

    task.handler(task.data);
    scheduleStatusRefresh();
  }

  if (taskList.length) {
    taskHandle = requestIdleCallback(runTaskQueue, { timeout: 1000} );
  } else {
    taskHandle = 0;
  }
}
</pre>

<p><code>runTaskQueue()</code>的核心是一个循环,只要有剩余时间(通过检查{{domxref("deadline.timeRemaining", "IdleDeadline.timeRemaining")}}来确认它大于0),或者已经达到了timeout期限({{domxref("IdleDeadline.didTimeout", "deadline.didTimeout")}}值为真),且任务列表中有任务就会一直持续。</p>

<p>对队列中每个我们有时间执行的任务,我们做以下操作:</p>

<ol>
 <li>我们 <a href="/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/shift">把任务对象(object)从队列中移除</a></li>
 <li>我们让<code>currentTaskNumber</code>增加来追踪我们已执行的任务数量。</li>
 <li>我们调用任务处理方法,<code>task.handler</code>,并任务的数据对象(<code>task.data</code>)传入其中。</li>
 <li>我们调用一个方法,<code>scheduleStatusRefresh()</code>,去处理调度一个屏幕更新来体现我们进度的变化。</li>
</ol>

<p>当时间耗尽,如果列表里还有任务,我们再次调用{{domxref("Window.requestIdleCallback", "requestIdleCallback()")}}使我们可以在下次有可用空闲时间时继续运行这些任务。如果队列是空的,我们将把<code>taskHandle</code>设置为0来表示我们没有回调日程了。这样,下一次<code>enqueueTask()</code>被调用时,我们就知道要请求一个回调了。</p>

<h4 id="更新状态显示">更新状态显示</h4>

<p>我们想要能够做的一件事是根据记录输出和进度信息来更新文档。然后在空闲回调中改变DOM是不安全的。作为替代,我们使用 {{domxref("Window.requestAnimationFrame", "requestAnimationFrame()")}} 来让浏览器在可以安全地更新显示时通知我们。</p>

<h5 id="安排显示的更新">安排显示的更新</h5>

<p>调用<code>scheduleStatusRefresh()</code>函数来安排DOM的改变。</p>

<pre class="brush: js">function scheduleStatusRefresh() {
    if (!statusRefreshScheduled) {
      requestAnimationFrame(updateDisplay);
      statusRefreshScheduled = true;
  }
}
</pre>

<p>这是一个简单的函数。它检查<code>statusRefreshScheduled</code>的值来得知我们是否已经安排了一个显示更新。如果值为<code>false</code>,我们调用{{domxref("Window.requestAnimationFrame", "requestAnimationFrame()")}}来安排一个更新,也就是提供一个<code>updateDisplay()</code>函数以被调用去处理那个工作。</p>

<h5 id="更新显示">更新显示</h5>

<p><code>updateDisplay()</code>函数负责绘制进度框的内容和记录。当DOM的状况安全,我们可以在下次渲染过程中申请改变时,浏览器会调用它。</p>

<pre class="brush: js">function updateDisplay() {
  let scrolledToEnd = logElem.scrollHeight - logElem.clientHeight &lt;= logElem.scrollTop + 1;

  if (totalTaskCount) {
    if (progressBarElem.max != totalTaskCount) {
      totalTaskCountElem.textContent = totalTaskCount;
      progressBarElem.max = totalTaskCount;
    }

    if (progressBarElem.value != currentTaskNumber) {
      currentTaskNumberElem.textContent = currentTaskNumber;
      progressBarElem.value = currentTaskNumber;
    }
  }

  if (logFragment) {
    logElem.appendChild(logFragment);
    logFragment = null;
  }

  if (scrolledToEnd) {
      logElem.scrollTop = logElem.scrollHeight - logElem.clientHeight;
  }

  statusRefreshScheduled = false;
}</pre>

<p>首先,在记录被滚动到底的时候<code>scrolledToEnd</code>会被设置为<code>true</code>,否则被设置为<code>false</code>。我们用它来决定,我们是否必须要更新滚动位置来确保我们在给记录添加内容的动作结束后,记录停留在末尾。</p>

<p>接下来,如果有任务进入队列中,我们更新进度和状态信息。</p>

<ol>
 <li>如果进度条当前的最大值不同于队列中当前的任务总数(<code>totalTaskCount</code>),我们就要更新任务总数(<code>totalTaskCountElem</code>)的显示内容和进度条的最大值,以使它的比例正确。</li>
 <li>我们对已运行的任务数做同样的操作;如果<code>progressBarElem.value</code>不同于当前正被处理的任务数(<code>currentTaskNumber</code>),我们就要更新当前运行的程序数量值和进度条当前值的显示。</li>
</ol>

<p>然后,如果有文本等待被添加到记录中(也就是说,<code>logFragment</code>不为<code>null</code>),我们使用{{domxref("Node.appendChild", "Element.appendChild()")}}将它添加到记录元素中,并将<code>logFragment</code>设置为以避免重复操作。</p>

<p>如果我们操作开始的时候记录被滚动到末尾,我们要确保它一直处理末尾的位置。然后我们将<code>statusRefreshScheduled</code>设置为<code>false</code>,以表明我们已经处理过更新,可以安全地请求新的更新了。</p>

<h4 id="向记录添加文本">向记录添加文本</h4>

<p><code>log()</code>函数可以向记录中添加指定的文本。因为我们不知道调用<code>log()</code>的时候是否可以立即安全地联系DOM,我们将缓存记录文本一直到可以安全更新。在上面,在<code>updateDisplay</code><code>()</code>的代码中,你可以找到更新动画帧时,实际添加记录的代码。</p>

<pre class="brush: js">function log(text) {
  if (!logFragment) {
      logFragment = document.createDocumentFragment();
  }

  let el = document.createElement("div");
  el.innerHTML = text;
  logFragment.appendChild(el);
}</pre>

<p><span class="tlid-translation translation" lang="zh-CN"><span title="">首先,如果当前不存在一个名为</span></span><code>logFragment</code><span class="tlid-translation translation" lang="zh-CN"><span title=""></span></span> {{domxref("DocumentFragment")}} <span class="tlid-translation translation" lang="zh-CN"><span title="">对象。</span> <span title="">该元素是伪DOM,我们可以在其中插入元素,而无需立即更改主DOM本身。</span></span></p>

<p>然后我们创建一个新的元素, <span class="tlid-translation translation" lang="zh-CN"><span title="">并将其内容设置为与输入文本匹配</span></span>。接下来我们向<code>logFragment</code>中的伪DOM末尾添加一个新的元素。<code>logFragment</code>将会累积记录条目直到下次因DOM改变而调用<code>updateDisplay()</code>的时候。</p>

<h3 id="运行任务">运行任务</h3>

<p>现在,我们的任务管理和显示维护代码已经完成了,我们实际上可以开始设定完成工作的代码了</p>

<h4 id="任务处理器">任务处理器</h4>

<p><code>logTaskHandler()</code>,将是我们用来作为任务处理器的函数,也是用作任务对象<code>handler</code>属性的值。它是一个简单的为每个任务向记录输出大量内容的函数。 <span class="tlid-translation translation" lang="zh-CN"><span title="">在您自己的应用程序中,您可以将此代码替换为您希望在空闲时间执行的任何任务</span></span> 。只要记住任何DOM变化都需要通过 {{domxref("Window.requestAnimationFrame", "requestAnimationFrame()")}} 处理。</p>

<pre class="brush: js">function logTaskHandler(data) {
  log("&lt;strong&gt;Running task #" + currentTaskNumber + "&lt;/strong&gt;");

  for (i=0; i&lt;data.count; i+=1) {
    log((i+1).toString() + ". " + data.text);
  }
}
</pre>

<h4 id="主程序">主程序</h4>

<p>当用户点击“开始”按钮,会触发所有操作,也会导致调用<code>decodeTechnoStuff()</code>函数。</p>

<div class="hidden">
<p>The <code><a href="/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/random?document_saved=true#Getting_a_random_integer_between_two_values_inclusive">getRandomIntInclusive()</a></code> method comes from the examples for {{jsxref("Math.random()")}}; we'll just link to it below but it needs to be included here for the example to work.</p>

<pre class="brush: js">function getRandomIntInclusive(min, max) {
  min = Math.ceil(min);
  max = Math.floor(max);
  return Math.floor(Math.random() * (max - min + 1)) + min;
}
</pre>
</div>

<pre class="brush: js">function decodeTechnoStuff() {
  totalTaskCount = 0;
  currentTaskNumber = 0;
  updateDisplay();

  let n = getRandomIntInclusive(100, 200);

  for (i=0; i&lt;n; i++) {
    let taskData = {
      count: getRandomIntInclusive(75, 150),
      text: "This text is from task number " + (i+1).toString() + " of " + n
    };

    enqueueTask(logTaskHandler, taskData);
  }
}

document.getElementById("startButton").addEventListener("click", decodeTechnoStuff, false);</pre>

<p><code>decodeTechnoStuff()</code>开始执行时会将任务总数(到现在为止添加到队列中的任务数)清零,并随后调用<code>updateDisplay()</code>以重置显示为“没有任何事发生”的状态。</p>

<p>这个示例去创建一个随机数量(100到200之间)的任务。为此,我们使用{{jsxref("Math.random()")}}文档中作为示例提供的<a href="/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/random?document_saved=true#Getting_a_random_integer_between_two_values_inclusive"><code>getRandomIntInclusive()</code></a>以获得要创建的任务数。</p>

<p>随后我们开始一个循环以创建实际的任务。对于每个任务,我们创建一个对象,<code>taskData</code>,其中包含两个属性:</p>

<ul>
 <li><code>count</code> 是要从任务输出到记录中的字符串数量。</li>
 <li><code>text</code> <span class="tlid-translation translation" lang="zh-CN"><span title="">是要输出到日志的文本(由</span></span><code>count</code><span class="tlid-translation translation" lang="zh-CN"><span title="">指定的次数)。</span></span></li>
</ul>

<p>我们调用<code>enqueueTask()</code>来将每个任务排入队列,将<code>logTaskHandler</code><code>()</code>传入作为处理函数,将<code>taskData</code>传入,待处理函数调用时传入其中。</p>

<dl>
</dl>

<h3 id="结果">结果</h3>

<p>下面就是以上代码实际功能结果。试一下,在你的浏览器开发者工具中使用它,并把它融入自己的代码中体验一下。</p>

<p>{{ EmbedLiveSample('Example', 600, 700) }}</p>

<h2 id="规范">规范</h2>

<table class="standard-table">
 <tbody>
  <tr>
   <th scope="col">Specification</th>
   <th scope="col">Status</th>
   <th scope="col">Comment</th>
  </tr>
  <tr>
   <td>{{SpecName("Background Tasks")}}</td>
   <td>{{Spec2("Background Tasks")}}</td>
   <td></td>
  </tr>
 </tbody>
</table>

<h2 id="浏览器兼容性">浏览器兼容性</h2>

<div>


<p>{{Compat("api.Window.requestIdleCallback")}}</p>
</div>

<h2 id="相关链接">相关链接</h2>

<ul>
 <li>{{domxref("Window.requestIdleCallback()")}}</li>
 <li>{{domxref("Window.cancelIdleCallback()")}}</li>
 <li>{{domxref("IdleDeadline")}}</li>
</ul>