웹 프론트엔드 JS 튜토리얼 JS 비동기 원리 문제에 대한 심층적인 이해

JS 비동기 원리 문제에 대한 심층적인 이해

Apr 09, 2018 am 11:48 AM
javascript

우리는 종종 JS가 단일 스레드라고 말합니다. 예를 들어 Node.js 세미나에서 모든 사람들은 JS의 기능 중 하나가 JS를 더 간단하고 명확하게 만든다고 말했습니다. JS의 단일 스레드 메커니즘이라고 합니까? 단일 스레드일 때 이벤트 기반 비동기 메커니즘은 무엇이어야 합니까? 이 지식은 "JavaScript Definitive Guide"에 소개되지 않았으며, 외국 기사를 읽고 나서야 몇 가지 아이디어를 얻었습니다. 여기에서 여러분과 공유하겠습니다. 번역 과정에서 누군가가 이미 이 글을 번역한 것을 발견하여, 그 중 일부 문장을 빌려왔습니다. 기사 URL: 링크. 나중에 "JavaScript Advanced 프로그래밍"이 고급 타이머와 루프 타이머를 도입했다는 것을 알게 되었는데, 제가 번역한 원문보다 더 철저하게 소개되지는 않은 것 같았습니다. 제 글이 좋지 않다고 느끼신다면 외국어 원문을 확인하시면 됩니다. 언어

1 먼저 읽어보세요. 다음 두 가지 예 1 1.1. Simple Settimeout

setTimeout(function () { while (true) { } }, 1000);
setTimeout(function () { alert('end 2'); }, 2000);
setTimeout(function () { alert('end 1'); }, 100);
alert('end');
로그인 후 복사

"End', 'End 1'의 결과를 실행한 다음 브라우저가 가짜이지만 'End 2'가 팝업되지 않습니다. 위로. 즉, 첫 번째 settimeout이 무한 루프로 실행되므로 이론적으로 1초 후에 실행되는 두 번째 settimeout의 함수가 직접 차단됩니다. 이는 우리가 일반적으로 비동기 함수 멀티스레딩으로 이해하는 것과 다릅니다. 서로 간섭하지 않으려면 일관성이 없습니다.

부착된 타이머 사용법

-

-初始化一个简单的js的计时器,一段时间后,才触发并执行回调函数。 setTimeout 返回一个唯一id,可用这个id来取消这个计时器。
var id = setTimeout(fn,delay);
 
--类似于setTimeout,不一样的是,每隔一段时间,会持续调用回调fn,直到被取消
var id = setInterval(fn,delay);
 
--传入一个计时器的id,取消计时器。
clearInterval(id);
clearTimeout(id);
로그인 후 복사

1.2.Ajax 요청 콜백

그럼 xmlhttprequest를 통해 비동기 ajax 요청 호출을 테스트해 보겠습니다. 주요 코드는 다음과 같습니다.

var xmlReq = createXMLHTTP();//创建一个xmlhttprequest对象
function testAsynRequest() {
    var url = "/AsyncHandler.ashx?action=ajax";
    xmlReq.open("post", url, true);
    xmlReq.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
    xmlReq.onreadystatechange = function () {
        if (xmlReq.readyState == 4) {
            if (xmlReq.status == 200) {
                var jsonData = eval('(' + xmlReq.responseText + ')');
                alert(jsonData.message);
            }
            else if (xmlReq.status == 404) {
                alert("Requested URL is not found.");
            } else if (xmlReq.status == 403) {
                alert("Access denied.");
            } else {
                alert("status is " + xmlReq.status);
            }
        }
    };
    xmlReq.send(null);
}
testAsynRequest();//1秒后调用回调函数
 
while (true) {
}
로그인 후 복사

서버 측에서 간단한 출력을 달성합니다.

private void ProcessAjaxRequest(HttpContext context)
{
    string action = context.Request["ajax"];
    Thread.Sleep(1000);//等1秒
    string jsonObject = "{\"message\":\"" + action + "\"}";
    context.Response.Write(jsonObject);
}
로그인 후 복사

이론적으로 ajax가 비동기 요청을 하고 비동기 콜백 함수가 별도의 스레드에 있는 경우 콜백 함수는 다른 스레드에 의해 "차단"되어서는 안 되며 원활하게 실행되어야 합니다. 즉, 1초 후에 콜백 실행이 팝업됩니다. 'ajax'를 실행했지만 실제 상황은 아닙니다. 브라우저가 다시 무한 루프로 인해 애니메이션을 일시 중지했기 때문입니다.

위의 두 가지 예를 바탕으로 요약하면 다음과 같습니다.

①    JavaScript引擎是单线程运行的,浏览器无论在什么时候都只且只有一个线程在运行JavaScript程序.
②    JavaScript引擎用单线程运行也是有意义的,单线程不必理会线程同步这些复杂的问题,问题得到简化。
로그인 후 복사

2.JavaScript 엔진

하지만 JS에서 내부적으로 구현하는 방법은 다음에 논의하겠습니다.

타이머의 내부 작동을 이해하기 전에 트리거링과 실행이 동일한 개념이 아니라는 점을 분명히 해야 합니다. 타이머의 콜백 함수는 지정된 지연 시간 후에 반드시 트리거되지만 반드시 즉시 실행되는 것은 아니며, 기다려야합니다. 모든 JavaScript 코드는 스레드에서 실행되며 마우스 클릭, 타이머 등의 이벤트는 JS 단일 스레드가 유휴 상태일 때만 실행됩니다.

JS의 스레드, 이벤트 루프 및 작업 대기열 소개

JS는 단일 스레드이지만 비동기 작업을 수행할 수 있는 이유는 주로 이벤트 루프(Event Loop)와 작업 대기열(Task Queue)이 있기 때문입니다. ) JS에서. JS 비동기 원리 문제에 대한 심층적인 이해

이벤트 루프: JS는 while(true)과 유사한 루프를 생성하며 루프 본문이 실행될 때마다 프로세스를 Tick이라고 합니다. 각 Tick의 프로세스는 보류 중인 이벤트가 있는지 확인하는 것입니다. 있는 경우 관련 이벤트와 콜백 함수를 꺼내서 메인 스레드에서 실행하기 위해 실행 스택에 넣습니다. 보류 중인 이벤트는 작업 대기열에 저장됩니다. 즉, 각 Tick은 작업 대기열에서 실행해야 하는 작업이 있는지 확인합니다.

작업 대기열: 비동기 작업은 관련 콜백을 작업 대기열에 추가합니다. 다양한 비동기 작업은 서로 다른 시간에 작업 대기열에 추가됩니다. 예를 들어 onclick, setTimeout 및 ajax는 위 그림의 세 가지 webAPI를 포함하는 브라우저 커널의 웹 코어에 의해 실행됩니다. . DOM 바인딩, 네트워크 및 타이머 모듈입니다.

  onclick은 브라우저 커널의 DOM 바인딩 모듈에 의해 처리됩니다. 이벤트가 트리거되면 콜백 함수가 작업 대기열에 즉시 추가됩니다.

  setTimeout은 브라우저 커널의 타이머 모듈에 의해 지연됩니다. 시간이 되면 콜백 함수가 작업 대기열에 추가됩니다.

  Ajax는 브라우저 커널의 네트워크 모듈에 의해 처리됩니다. 네트워크 요청이 완료되고 반환된 후 콜백이 작업 대기열에 추가됩니다.

主线程:JS 只有一个线程,称之为主线程。而事件循环是主线程中执行栈里的代码执行完毕之后,才开始执行的。所以,主线程中要执行的代码时间过长,会阻塞事件循环的执行,也就会阻塞异步操作的执行。只有当主线程中执行栈为空的时候(即同步代码执行完后),才会进行事件循环来观察要执行的事件回调,当事件循环检测到任务队列中有事件就取出相关回调放入执行栈中由主线程执行。

Update:

《你不知道的 JavaScript》一书中,重新讲解了 ES6 新增的任务队列,和上面的任务队列略有不同,上面的任务队列书中称为事件队列。

上面提到的任务(事件)队列是在事件循环中的,事件循环每一次 tick 便执行上面所述的任务(事件)队列中的一个任务。而任务(事件)队列是只能往尾部添加任务。

而 ES6 中新增的任务队列是在事件循环之上的,事件循环每次 tick 后会查看 ES6 的任务队列中是否有任务要执行,也就是 ES6 的任务队列比事件循环中的任务(事件)队列优先级更高。

如 Promise 就使用了 ES6 的任务队列特性。 

3. JavaScript引擎线程和其它侦听线程

在浏览器中,JavaScript引擎是基于事件驱动的,这里的事件可看作是浏览器派给它的各种任务,这些任务可能源自当前执行的代码块,如调用setTimeout(),也可能来自浏览器内核,如onload()、onclick()、onmouseover()、setTimeOut()、setInterval()、Ajax等。如果从代码的角度来看,所谓的任务实体就是各种回调函数,由于“单线程”的原因,这些任务会进行排队,一个接着一个等待着被引擎处理。

JS 비동기 원리 문제에 대한 심층적인 이해

上图中,定时器和事件都按时触发了,这表明JavaScript引擎的线程和计时器触发线程、事件触发线程是三个单独的线程,即使JavaScript引擎的线程被阻塞,其它两个触发线程都在运行。

  浏览器内核实现允许多个线程异步执行,这些线程在内核制控下相互配合以保持同步。假如某一浏览器内核的实现至少有三个常驻线程: JavaScript引擎线程,事件触发线程,Http请求线程,下面通过一个图来阐明单线程的JavaScript引擎与另外那些线程是怎样互动通信的。虽然每个浏览器内核实现细节不同,但这其中的调用原理都是大同小异。

线程间通信:JavaScript引擎执行当前的代码块,其它诸如setTimeout给JS引擎添加一个任务,也可来自浏览器内核的其它线程,如界面元素鼠标点击事件,定时触发器时间到达通知,异步请求状态变更通知等.从代码角度看来任务实体就是各种回调函数,JavaScript引擎一直等待着任务队列中任务的到来.由于单线程关系,这些任务得进行排队,一个接着一个被引擎处理.

GUI渲染也是在引擎线程中执行的,脚本中执行对界面进行更新操作,如添加结点,删除结点或改变结点的外观等更新并不会立即体现出来,这些操作将保存在一个队列中,待JavaScript引擎空闲时才有机会渲染出来。来看例子(这块内容还有待验证,个人觉得当Dom渲染时,才可阻止渲染)

<p id="test">test</p>
<script type="text/javascript" language="javascript">
var i=0;
while(1) {
    document.getElementById("test").innerHTML+=i++ + "<br />";
}
</script>
로그인 후 복사

  这段代码的本意是从0开始顺序显示数字,它们将一个接一个出现,现在我们来仔细研究一下代码,while(1)创建了一个无休止的循环,但是对于单线程的JavaScript引擎而言,在实际情况中就会造成浏览器暂停响应并处于假死状态。

  alert()会停止JS引擎的执行,直到按确认键,在JS调试的时候,查看当前实时页面的内容。

4. setTimeout和 setInterval

回到文章开头,我们来看下setTimeout和setsetInterval的区别。

setTimeout(function(){
    /* Some long block of code ... */
    setTimout(arguments.callee,10);
},10);
 
setInterval(function(){
    /* Some long block of code ... */
},10);
로그인 후 복사

  这两个程序段第一眼看上去是一样的,但并不是这样。setTimeout代码至少每隔10ms以上才执行一次;然而setInterval固定每隔10ms将尝试执行,不管它的回调函数的执行状态。

我们来总结下:

l JavaScript引擎只有一个线程,强制异步事件排队等待执行。
l setTimeout和setInterval在异步执行时,有着根本性不同。
l 如果一个计时器被阻塞执行,它将会延迟,直到下一个可执行点(这可能比期望的时间更长)
l setInterval的回调可能被不停的执行,中间没间隔(如果回调执行的时间超过预定等待的值)
로그인 후 복사

《JavaScript高级程序设计》中,针对setInterval说法如下:

当使用setInterval()时,仅当没有该定时器的任何其他代码实例时,才将定时器代码添加到队列中。还要注意两问题:

①    某些间隔会被跳过(抛弃);
② 多个定时器的代码执行之间的间隔可能会比预期小。此时可采取 setTimeout和setsetInterval的区别 的例子方法。
로그인 후 복사

5.       Ajax异步

많은 학우들과 친구들이 혼란스러워합니다. JavaScript는 단일 스레드에서 실행된다고 하는데 XMLHttpRequest는 연결 후 정말 비동기인가요? 실제로 요청은 비동기식이지만 이 요청은 브라우저에서 새 스레드를 열도록 요청합니다(위 그림 참조). 요청 상태가 변경되고 이전에 콜백이 설정된 경우 비동기 스레드가 생성됩니다. 상태 변경 이벤트를 JavaScript 엔진에 넣습니다. 작업이 처리될 때 JavaScript 엔진은 항상 단일 스레드에서 콜백 함수를 실행합니다. 단일 스레드.

팁: 특히 많은 수의 비동기 이벤트가 (지속적으로) 발생할 때 JavaScript 엔진의 작동을 이해하는 것이 매우 중요하며 이는 프로그램 코드의 효율성을 향상시킬 수 있습니다.

위 내용은 JS 비동기 원리 문제에 대한 심층적인 이해의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

노흡의 기능 및 원리 분석 노흡의 기능 및 원리 분석 Mar 25, 2024 pm 03:24 PM

nohup의 역할과 원리 분석 nohup은 유닉스 및 유닉스 계열 운영체제에서 사용자가 현재 세션을 종료하거나 터미널 창을 닫아도 백그라운드에서 명령을 실행하는 데 일반적으로 사용되는 명령입니다. 아직도 계속 처형되고 있다. 이번 글에서는 nohup 명령의 기능과 원리를 자세히 분석해보겠습니다. 1. nohup의 역할: 백그라운드에서 명령 실행: nohup 명령을 통해 사용자가 터미널 세션을 종료해도 영향을 받지 않고 장기 실행 명령이 백그라운드에서 계속 실행되도록 할 수 있습니다. 이건 실행해야 해

Struts 프레임워크의 원칙과 실무에 대한 심층적인 토론 Struts 프레임워크의 원칙과 실무에 대한 심층적인 토론 Feb 18, 2024 pm 06:10 PM

Struts 프레임워크의 원리 분석 및 실제 탐색 JavaWeb 개발에서 일반적으로 사용되는 MVC 프레임워크인 Struts 프레임워크는 우수한 디자인 패턴과 확장성을 가지며 엔터프라이즈 수준 애플리케이션 개발에 널리 사용됩니다. 이 기사에서는 Struts 프레임워크의 원리를 분석하고 실제 코드 예제를 통해 이를 탐색하여 독자가 프레임워크를 더 잘 이해하고 적용할 수 있도록 돕습니다. 1. Struts 프레임워크의 원리 분석 1. MVC 아키텍처 Struts 프레임워크는 MVC(Model-View-Con)를 기반으로 합니다.

MyBatis의 배치 삽입 구현 원리에 대한 심층적인 이해 MyBatis의 배치 삽입 구현 원리에 대한 심층적인 이해 Feb 21, 2024 pm 04:42 PM

MyBatis는 다양한 Java 프로젝트에서 널리 사용되는 인기 있는 Java 지속성 계층 프레임워크입니다. 그중 일괄 삽입은 데이터베이스 작업의 성능을 효과적으로 향상시킬 수 있는 일반적인 작업입니다. 이번 글에서는 MyBatis에서 일괄 Insert의 구현 원리를 심층적으로 살펴보고 구체적인 코드 예제를 통해 자세히 분석해 보겠습니다. MyBatis의 일괄 삽입 MyBatis에서 일괄 삽입 작업은 일반적으로 동적 SQL을 사용하여 구현됩니다. 삽입된 여러 값을 포함하는 S를 구성하여

Linux RPM 도구의 기능과 원리에 대한 심층적인 논의 Linux RPM 도구의 기능과 원리에 대한 심층적인 논의 Feb 23, 2024 pm 03:00 PM

Linux 시스템의 RPM(RedHatPackageManager) 도구는 시스템 소프트웨어 패키지를 설치, 업그레이드, 제거 및 관리하기 위한 강력한 도구입니다. RedHatLinux 시스템에서 일반적으로 사용되는 패키지 관리 도구이며 다른 많은 Linux 배포판에서도 사용됩니다. RPM 도구의 역할은 시스템 관리자와 사용자가 시스템의 소프트웨어 패키지를 쉽게 관리할 수 있도록 하는 데 매우 중요합니다. RPM을 통해 사용자는 쉽게 새로운 소프트웨어 패키지를 설치하고 기존 소프트웨어를 업그레이드할 수 있습니다.

MyBatis 페이징 플러그인의 원리에 대한 자세한 설명 MyBatis 페이징 플러그인의 원리에 대한 자세한 설명 Feb 22, 2024 pm 03:42 PM

MyBatis는 XML과 주석을 기반으로 하는 뛰어난 지속성 레이어 프레임워크입니다. 또한 간단하고 사용하기 쉬운 플러그인 메커니즘도 제공합니다. 그 중 페이징 플러그인은 가장 많이 사용되는 플러그인 중 하나입니다. 이 기사에서는 MyBatis 페이징 플러그인의 원리를 자세히 살펴보고 특정 코드 예제를 통해 설명합니다. 1. 페이징 플러그인 원리 MyBatis 자체는 기본 페이징 기능을 제공하지 않지만 플러그인을 사용하여 페이징 쿼리를 구현할 수 있습니다. 페이징 플러그인의 원리는 주로 MyBatis를 가로채는 것입니다.

Linux chage 명령의 기능 및 작동 원리에 대한 심층 분석 Linux chage 명령의 기능 및 작동 원리에 대한 심층 분석 Feb 24, 2024 pm 03:48 PM

Linux 시스템의 chage 명령은 사용자 계정의 비밀번호 만료일을 수정하는 데 사용되는 명령이며, 계정의 사용 가능한 가장 긴 날짜와 가장 짧은 날짜를 수정하는 데에도 사용할 수 있습니다. 이 명령은 사용자 계정 보안 관리에 매우 중요한 역할을 하며 사용자 비밀번호의 사용 기간을 효과적으로 제어하고 시스템 보안을 강화할 수 있습니다. chage 명령 사용 방법: chage 명령의 기본 구문은 다음과 같습니다: chage [옵션] 사용자 이름 예를 들어, 사용자 "testuser"의 비밀번호 만료 날짜를 수정하려면 다음 명령을 사용할 수 있습니다.

Golang에서 상속 방법을 구현하는 기본 원칙과 방법 Golang에서 상속 방법을 구현하는 기본 원칙과 방법 Jan 20, 2024 am 09:11 AM

Golang 상속 방법의 기본 원리 및 구현 방법 Golang에서 상속은 객체 지향 프로그래밍의 중요한 기능 중 하나입니다. 상속을 통해 부모 클래스의 속성과 메서드를 사용하여 코드 재사용 및 확장성을 달성할 수 있습니다. 이 글에서는 Golang 상속 메소드의 기본 원리와 구현 방법을 소개하고 구체적인 코드 예제를 제공합니다. 상속 방법의 기본 원칙 Golang에서는 상속이 임베딩 구조로 구현됩니다. 구조가 다른 구조에 포함되면 포함된 구조가 포함됩니다.

Python asyncio 고급 가이드: 초보자부터 전문가까지 Python asyncio 고급 가이드: 초보자부터 전문가까지 Mar 04, 2024 am 09:43 AM

동시 및 비동기 프로그래밍 동시 프로그래밍은 동시에 실행되는 여러 작업을 처리하며, 비동기 프로그래밍은 작업이 스레드를 차단하지 않는 일종의 동시 프로그래밍입니다. asyncio는 프로그램이 메인 스레드를 차단하지 않고 I/O 작업을 수행할 수 있도록 하는 Python의 비동기 프로그래밍용 라이브러리입니다. 이벤트 루프 asyncio의 핵심은 I/O 이벤트를 모니터링하고 해당 작업을 예약하는 이벤트 루프입니다. 코루틴이 준비되면 이벤트 루프는 I/O 작업을 기다릴 때까지 이를 실행합니다. 그런 다음 코루틴을 일시 중지하고 다른 코루틴을 계속 실행합니다. 코루틴 코루틴은 실행을 일시 중지하고 다시 시작할 수 있는 함수입니다. asyncdef 키워드는 코루틴을 만드는 데 사용됩니다. 코루틴은 I/O 작업이 완료될 때까지 기다리기 위해 wait 키워드를 사용합니다. 다음과 같은 asyncio의 기본 사항

See all articles