猿问

为节拍器在浏览器中准确计时声音

我想创建一个虚拟节拍器并在每一步播放节拍器声音。setTimeout()/等常用方法setInterval()不够准确。我怎样才能让它在不延迟或丢失节拍的情况下工作?



潇潇雨雨
浏览 88回答 1
1回答

缥缈止盈

这是一个每 500 毫秒发出一次哔声的函数。function beep() {&nbsp; &nbsp; var snd = new Audio("data:audio/wav;base64,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qJrx4KOQjahZxWbcZgztj2c49nKmkId44S71j0c8eV9yDK6uPRzx5X18eDvjvQ6yKo9ZSS6l//8elePK/Lf//IInrOF/FvDoADYAGBMGb7FtErm5MXMlmPAJQVgWta7Zx2go+8xJ0UiCb8LHHdftWyLJE0QIAIsI+UbXu67dZMjmgDGCGl1H+vpF4NSDckSIkk7Vd+sxEhBQMRU8j/12UIRhzSaUdQ+rQU5kGeFxm+hb1oh6pWWmv3uvmReDl0UnvtapVaIzo1jZbf/pD6ElLqSX+rUmOQNpJFa/r+sa4e/pBlAABoAAAAA3CUgShLdGIxsY7AUABPRrgCABdDuQ5GC7DqPQCgbbJUAoRSUj+NIEig0YfyWUho1VBBBA//uQZB4ABZx5zfMakeAAAAmwAAAAF5F3P0w9GtAAACfAAAAAwLhMDmAYWMgVEG1U0FIGCBgXBXAtfMH10000EEEEEECUBYln03TTTdNBDZopopYvrTTdNa325mImNg3TTPV9q3pmY0xoO6bv3r00y+IDGid/9aaaZTGMuj9mpu9Mpio1dXrr5HERTZSmqU36A3CumzN/9Robv/Xx4v9ijkSRSNLQhAWumap82WRSBUqXStV/YcS+XVLnSS+WLDroqArFkMEsAS+eWmrUzrO0oEmE40RlMZ5+ODIkAyKAGUwZ3mVKmcamcJnMW26MRPgUw6j+LkhyHGVGYjSUUKNpuJUQoOIAyDvEyG8S5yfK6dhZc0Tx1KI/gviKL6qvvFs1+bWtaz58uUNnryq6kt5RzOCkPWlVqVX2a/EEBUdU1KrXLf40GoiiFXK///qpoiDXrOgqDR38JB0bw7SoL+ZB9o1RCkQjQ2CBYZKd/+VJxZRRZlqSkKiws0WFxUyCwsKiMy7hUVFhIaCrNQsKkTIsLivwKKigsj8XYlwt/WKi2N4d//uQRCSAAjURNIHpMZBGYiaQPSYyAAABLAAAAAAAACWAAAAApUF/Mg+0aohSIRobBAsMlO//Kk4soosy1JSFRYWaLC4qZBYWFRGZdwqKiwkNBVmoWFSJkWFxX4FFRQWR+LsS4W/rFRb/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////VEFHAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAU291bmRib3kuZGUAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAMjAwNGh0dHA6Ly93d3cuc291bmRib3kuZGUAAAAAAAAAACU=");&nbsp;&nbsp;&nbsp; &nbsp; snd.play();}setInterval(beep, 500)然而,这不是很精确。回调可能会延迟触发(取决于很多,这可以从几毫秒到超过 1000 毫秒)。要解决这个问题,您可以使用AudioContext.&nbsp;我们将使用一个简单Oscillator的节拍器声音。您可以一次安排多个音符,也可以setInterval用来运行一个安排下一个应该播放的音符的功能。function noteDurationToMs (bpm, dur, type) {&nbsp; &nbsp; return 60000 * 4 * dur * type / bpm}function scheduleNote(ac, time, dur) {&nbsp; &nbsp; var osc = ac.createOscillator();&nbsp; &nbsp; osc.connect( ac.destination );&nbsp; &nbsp; osc.start(time);&nbsp; &nbsp; osc.stop(time + dur);}const ac = new AudioContext();let lastNote= ac.currentTime;const step = noteDurationToMs(120, 1 / 4, 1) / 1000;const lookAhead = step / 2;let id, timer = ()=> {&nbsp; const diff = ac.currentTime - lastNote;&nbsp; if (diff >= lookAhead) {&nbsp; &nbsp;const nextNote = lastNote + step;&nbsp; &nbsp;scheduleNote(ac, nextNote, 0.025)&nbsp; &nbsp;lastNote = nextNote;&nbsp; }}a.addEventListener('click', () => {&nbsp; &nbsp; ac.resume();&nbsp; &nbsp; id = setInterval(timer, 15)})b.addEventListener('click', () => clearInterval(id))<button id="a">Start</button><button id="b">Stop</button>注意:由于某种原因AudioContext,Stack Snippets 似乎没有工作。这是第一个代码片段的jsfiddle。这是第二个片段的一个。function noteDurationToMs (bpm, dur, type) {&nbsp; &nbsp; return 60000 * 4 * dur * type / bpm}function scheduleNote(ac, time, dur) {&nbsp; &nbsp; var osc = ac.createOscillator();&nbsp; &nbsp; osc.connect( ac.destination );&nbsp; &nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; osc.start(time);&nbsp; &nbsp; osc.stop(time + dur);}const ac = new AudioContext();let to, lastNote = 0;const Metronome = () => {&nbsp; const [bpm, setBpm] = React.useState(100);&nbsp; const [dur, setDur] = React.useState(1);&nbsp;&nbsp;&nbsp; const [type, setType] = React.useState(1);&nbsp; &nbsp; const [run, setRun] = React.useState(false);&nbsp; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; const handleChangeBPM = (e) => {&nbsp; &nbsp; setBpm(e.target.value)&nbsp; }&nbsp;&nbsp;&nbsp; const handleChangeDur = (e) => {&nbsp; &nbsp; setDur(e.target.value)&nbsp; }&nbsp;&nbsp;&nbsp; const handleChangeType = (e) => {&nbsp; &nbsp; setType(e.target.value)&nbsp; }&nbsp;&nbsp;&nbsp; const step = noteDurationToMs(bpm, dur, type) / 1000;&nbsp; &nbsp; const lookAhead = step / 2;&nbsp;&nbsp;&nbsp; const timer = () => {&nbsp; &nbsp; const diff = ac.currentTime - lastNote;&nbsp; &nbsp; if (diff >= lookAhead) {&nbsp; &nbsp; &nbsp;const nextNote = lastNote + step;&nbsp; &nbsp; &nbsp;scheduleNote(ac, nextNote, 0.025)&nbsp; &nbsp; &nbsp;lastNote = nextNote;&nbsp; &nbsp; }&nbsp; }&nbsp;&nbsp;&nbsp; const start = () => {&nbsp; &nbsp; ac.resume()&nbsp; &nbsp; setRun(true);&nbsp; }&nbsp;&nbsp;&nbsp; const stop = () => {&nbsp; &nbsp; clearInterval(to);&nbsp; &nbsp; setRun(false);&nbsp; }&nbsp;&nbsp;&nbsp; const toggle = () => {&nbsp; &nbsp; run?stop():start();&nbsp; }&nbsp; &nbsp; React.useEffect(() => {&nbsp; &nbsp; if (run) {&nbsp; &nbsp; &nbsp; &nbsp; clearInterval(to);&nbsp; &nbsp; &nbsp; to = setInterval(timer, step / 4);&nbsp; &nbsp; }&nbsp; });&nbsp;&nbsp;&nbsp; &nbsp; return <div>&nbsp; &nbsp; <label for="bpm">BPM:</label>&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; &nbsp; <input id="bpm" type="number" onChange={handleChangeBPM} value={bpm} />&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; <label for="dur">Duration:</label>&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; <select id="dur" onChange={handleChangeDur} value={dur}>&nbsp; &nbsp; &nbsp; <option value={1}>Whole</option>&nbsp; &nbsp; &nbsp; <option value={1 / 2}>Half</option>&nbsp; &nbsp; &nbsp; <option value={1 / 4}>Quarter</option>&nbsp; &nbsp; &nbsp; <option value={1 / 8}>Eigth </option>&nbsp; &nbsp; &nbsp; <option value={1 / 16}>Sixteenth</option>&nbsp; &nbsp; &nbsp; <option value={1 / 32}>Thirtysecond</option>&nbsp; &nbsp; </select>&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; <label for="typ">Type:</label>&nbsp; &nbsp; <select id="typ" onChange={handleChangeType} value={type}>&nbsp; &nbsp; &nbsp; <option value={1}>Regular</option>&nbsp; &nbsp; &nbsp; <option value={3 / 2}>Dotted</option>&nbsp; &nbsp; &nbsp; <option value={2 / 3}>Triplet</option>&nbsp; &nbsp; </select>&nbsp; &nbsp; <button onClick={toggle}>{!run?'Start':'Stop'}</button>&nbsp; &nbsp; <br />&nbsp; &nbsp; <label for="ms">MS:</label>&nbsp; &nbsp; <output id="ms">{noteDurationToMs(bpm, dur, type)}</output>&nbsp; &nbsp; </div>}ReactDOM.render(<Metronome />, document.querySelector("#app"))<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.4/umd/react.production.min.js"></script><script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.4/umd/react-dom.production.min.js"></script><div id="app"></div>要包含您自己的编辑: 您还可以使用AudioBufferSourceNode和调用.start(),when参数来准确地播放音频,但您将无法使用它运行函数。AudioBufferSourceNode.start([when]);
随时随地看视频慕课网APP

相关分类

JavaScript
我要回答