JavaScript의 마이크로 초 타이밍
마이크로 초 해상도의 JavaScript에 타이밍 기능이 있습니까?
저는 Chrome 용 timer.js 를 알고 있으며 Firefox, Safari, Opera, Epiphany, Konqueror 등과 같은 다른 친숙한 브라우저에 대한 솔루션이 있기를 바랍니다. IE 지원에는 관심이 없지만 IE를 포함한 답변 환영합니다.
(JS에서 밀리 초 타이밍의 정확도가 낮다는 점을 감안할 때, 나는 이것에 숨을 참을 수 없습니다!)
업데이트 : timer.js는 마이크로 초 해상도를 광고하지만 단순히 밀리 초 판독 값에 1,000을 곱합니다. 테스트 및 코드 검사를 통해 확인되었습니다. 실망한. : [
: 마크 Rejhon의 대답에 언급 된 바와 같이, 스크립트로 서브 밀리 초 고해상도 타이밍 데이터를 노출 최신 브라우저에서 사용할 수있는 API가 W3C 고해상도 타이머는 일명, window.performance.now()
.
now()
Date.getTime()
두 가지 중요한 측면에서 기존보다 낫습니다 .
now()
페이지 탐색이 시작된 이후의 밀리 초 수를 나타내는 밀리 초 미만 해상도의 double입니다. 분수 단위로 마이크로 초 수를 반환합니다 (예 : 1000.123 값은 1 초 및 123 마이크로 초).now()
단조롭게 증가합니다. 이것은 중요하다Date.getTime()
할 가능성이 후속 호출에 뒤로 앞으로 또는 이동. 특히 OS의 시스템 시간이 업데이트되면 (예 : 원자 시계 동기화)Date.getTime()
업데이트됩니다.now()
항상 단조롭게 증가하므로 OS의 시스템 시간에 영향을받지 않습니다. 항상 벽시계 시간입니다 (벽시계가 원 자성이 아니라고 가정하면 ...).
now()
에서 사용되는 거의 모든 장소가 될 수있다 new Date.getTime()
, + new Date
하고 Date.now()
있습니다. 예외는 점이다 Date
과 now()
같은 시간, 혼합하지 마십시오 Date
기반으로 유닉스 시대 (1970 년 이후 밀리 초 단위) 동안, now()
(그것보다 훨씬 작은 될 것입니다 귀하의 페이지 탐색이 시작된 이후의 밀리 초 수입니다 Date
).
now()
Chrome 안정, Firefox 15 이상 및 IE10에서 지원됩니다. 사용 가능한 여러 polyfill도 있습니다.
이제 자바 스크립트에서 마이크로 초를 측정하는 새로운 방법이 있습니다. http://gent.ilcore.com/2012/06/better-timer-for-javascript.html
그러나 과거에는 밀리 초 타이머에서 JavaScript에서 0.1 밀리 초 정밀도를 얻는 조잡한 방법을 발견했습니다. 불가능한? 아니. 계속 읽으세요:
자체 검사 된 타이머 정확도가 필요한 몇 가지 고정밀 실험을 수행하고 있으며 특정 시스템의 특정 브라우저에서 0.1 밀리 초의 정밀도를 안정적으로 얻을 수 있다는 것을 발견했습니다.
빠른 시스템의 최신 GPU 가속 웹 브라우저 (예 : 여러 코어가 유휴 상태 인 i7 쿼드 코어, 브라우저 창만 있음)에서 타이머가 밀리 초 단위로 정확하다고 믿을 수 있습니다. 실제로 유휴 i7 시스템에서는 매우 정확 해져서 1,000 회 이상의 시도에서 정확히 동일한 밀리 초를 안정적으로 얻을 수있었습니다. 추가 웹 페이지를로드하는 등의 작업을 수행하려고 할 때만 밀리 초의 정확도가 저하됩니다 (그리고 전후 시간 확인을 통해 저하 된 정확도를 성공적으로 파악할 수 있습니다. 처리 시간이 갑자기 1 밀리 초 이상으로 늘어났습니다. 이는 CPU 변동으로 인해 너무 부정적인 영향을받은 결과를 무효화하는 데 도움이됩니다.)
i7 쿼드 코어 시스템 (브라우저 창이 유일한 창일 때)의 일부 GPU 가속 브라우저에서 너무 정확 해 졌기 때문에 정확도가 드디어 이제는 JavaScript에서 0.1ms 정밀 타이머에 액세스 할 수 있기를 바랐습니다. 고정밀이 필요한 특정 유형의 틈새 애플리케이션과 애플리케이션이 정확도 편차를 자체 검증 할 수있는 특정 유형의 틈새 애플리케이션에 대해 이러한 타이머 정밀도를 가치있게 만드는 일부 고급 브라우징 시스템에 있습니다.
분명히 여러 패스를 수행하는 경우 여러 패스 (예 : 10 패스)를 실행 한 다음 10으로 나누어 0.1 밀리 초 정밀도를 얻을 수 있습니다. 그것은 더 나은 정밀도를 얻는 일반적인 방법입니다. 여러 번 통과하고 총 시간을 통과 횟수로 나눕니다.
그러나 ... 비정상적으로 독특한 상황으로 인해 특정 테스트의 단일 벤치 마크 통과 만 수행 할 수 있다면 다음을 수행하여 0.1 (때로는 0.01ms) 정밀도를 얻을 수 있다는 것을 알았습니다.
초기화 / 보정 :
- 타이머가 다음 밀리 초까지 증가 할 때까지 기다리려면 사용 중 루프를 실행합니다 (타이머를 다음 밀리 초 간격의 시작 부분에 정렬).이 사용중인 루프는 1 밀리 초 미만으로 지속됩니다.
- 타이머가 증가 할 때까지 기다리는 동안 카운터를 증가시키기 위해 다른 바쁜 루프를 실행합니다. 카운터는 1 밀리 초 동안 발생한 카운터 증가 횟수를 알려줍니다. 이 바쁜 루프는 1 밀리 초 동안 지속됩니다.
- 숫자가 매우 안정적이 될 때까지 (로드 시간, JIT 컴파일러 등) 위의 과정을 반복합니다. 4. 참고: 숫자의 안정성은 유휴 시스템에서 달성 가능한 정밀도를 제공합니다. 정밀도를 자체 확인해야하는 경우 분산을 계산할 수 있습니다. 차이는 일부 브라우저에서는 더 크고 다른 브라우저에서는 더 작습니다. 빠른 시스템에서는 더 크고 느린 시스템에서는 더 느립니다. 일관성도 다양합니다. 어떤 브라우저가 다른 브라우저보다 일관성 / 정확한지 알 수 있습니다. 느린 시스템과 바쁜 시스템은 초기화 단계 사이에 더 큰 차이로 이어집니다. 이렇게하면 브라우저가 0.1ms 또는 0.01ms 측정을 허용하기에 충분한 정밀도를 제공하지 않는 경우 경고 메시지를 표시 할 수 있습니다. 타이머 왜곡이 문제가 될 수 있지만 일부 시스템의 일부 정수 밀리 초 타이머는 매우 정확하게 증가하므로 (점에서 거의) 신뢰할 수있는 매우 일관된 보정 값을 얻을 수 있습니다.
- 최종 카운터 값 (또는 마지막 몇 번의 보정 통과 평균)을 저장합니다.
1 회 통과를 밀리 초 미만의 정밀도로 벤치마킹 :
- 타이머가 다음 밀리 초로 증가 할 때까지 기다리려면 사용중인 루프를 실행합니다 (타이머를 다음 밀리 초 간격의 시작 부분에 맞 춥니 다). 이 바쁜 루프는 1 밀리 초 미만으로 지속됩니다.
- 시간을 정확하게 벤치마킹하려는 작업을 실행하십시오.
- 타이머를 확인하십시오. 이것은 정수 밀리 초를 제공합니다.
- 타이머가 증가 할 때까지 기다리는 동안 카운터를 증가시키기 위해 마지막 사용중 루프를 실행합니다. 이 바쁜 루프는 1 밀리 초 미만으로 지속됩니다.
- 이 카운터 값을 초기화에서 원래 카운터 값으로 나눕니다.
- 이제 밀리 초의 소수 부분을 얻었습니다 !!!!!!!!
경고 : 바쁜 루프는 웹 브라우저에서 권장되지 않지만 다행스럽게도 이러한 바쁜 루프는 각각 1 밀리 초 미만으로 실행되며 몇 번만 실행됩니다.
Variables such as JIT compilation and CPU fluctuations add massive inaccuracies, but if you run several initialization passes, you'll have full dynamic recompilation, and eventually the counter settles to something very accurate. Make sure that all busy loops is exactly the same function for all cases, so that differences in busy loops do not lead to differences. Make sure all lines of code are executed several times before you begin to trust the results, to allow JIT compilers to have already stabilized to a full dynamic recompilation (dynarec).
In fact, I witnessed precision approaching microseconds on certain systems, but I wouldn't trust it yet. But the 0.1 millisecond precision appears to work quite reliably, on an idle quad-core system where I'm the only browser page. I came to a scientific test case where I could only do one-off passes (due to unique variables occuring), and needed to precisely time each pass, rather than averaging multiple repeat pass, so that's why I did this.
I did several pre-passes and dummy passes (also to settle the dynarec), to verify reliability of 0.1ms precision (stayed solid for several seconds), then kept my hands off the keyboard/mouse, while the benchmark occured, then did several post-passes to verify reliability of 0.1ms precision (stayed solid again). This also verifies that things such as power state changes, or other stuff, didn't occur between the before-and-after, interfering with results. Repeat the pre-test and post-test between every single benchmark pass. Upon this, I was quite virtually certain the results in between were accurate. There is no guarantee, of course, but it goes to show that accurate <0.1ms precision is possible in some cases in a web browser.
This method is only useful in very, very niche cases. Even so, it literally won't be 100% infinitely guaranteeable, you can gain quite very trustworthy accuracy, and even scientific accuracy when combined with several layers of internal and external verifications.
The answer is "no", in general. If you're using JavaScript in some server-side environment (that is, not in a browser), then all bets are off and you can try to do anything you want.
edit — this answer is old; the standards have progressed and newer facilities are available as solutions to the problem of accurate time. Even so, it should be remembered that outside the domain of a true real-time operating system, ordinary non-privileged code has limited control over its access to compute resources. Measuring performance is not the same (necessarily) as predicting performance.
Here is an example showing my high-resolution timer for node.js:
function startTimer() {
const time = process.hrtime();
return time;
}
function endTimer(time) {
function roundTo(decimalPlaces, numberToRound) {
return +(Math.round(numberToRound + `e+${decimalPlaces}`) + `e-${decimalPlaces}`);
}
const diff = process.hrtime(time);
const NS_PER_SEC = 1e9;
const result = (diff[0] * NS_PER_SEC + diff[1]); // Result in Nanoseconds
const elapsed = result * 0.0000010;
return roundTo(6, elapsed); // Result in milliseconds
}
Usage:
const start = startTimer();
console.log('test');
console.log(`Time since start: ${endTimer(start)} ms`);
Normally, you might be able to use:
console.time('Time since start');
console.log('test');
console.timeEnd('Time since start');
If you are timing sections of code that involve looping, you cannot gain access to the value of console.timeEnd()
in order to add your timer results together. You can, but it get gets nasty because you have to inject the value of your iterating variable, such as i
, and set a condition to detect if the loop is done.
Here is an example because it can be useful:
const num = 10;
console.time(`Time til ${num}`);
for (let i = 0; i < num; i++) {
console.log('test');
if ((i+1) === num) { console.timeEnd(`Time til ${num}`); }
console.log('...additional steps');
}
Cite: https://nodejs.org/api/process.html#process_process_hrtime_time
참고URL : https://stackoverflow.com/questions/6233927/microsecond-timing-in-javascript
'code' 카테고리의 다른 글
Makefile에서 괄호 $ ()와 중괄호 $ {} 구문의 차이점은 무엇입니까? (0) | 2020.09.12 |
---|---|
입력 유형 버튼-레이블 대 값 (0) | 2020.09.12 |
레이블 뒤에 초기화 된 변수가있는 경우 "레이블은 명령문의 일부만 될 수 있고 선언은 명령문이 아닙니다"라는 메시지가 표시되는 이유는 무엇입니까? (0) | 2020.09.12 |
LNK2019 오류를 해결하는 방법 : 해결되지 않은 외부 기호-기능? (0) | 2020.09.12 |
PHP를 사용하여 SQL Server에서 문자열을 이스케이프하는 방법은 무엇입니까? (0) | 2020.09.12 |