목차
Events
이벤트 루프
이벤트 루프 차단
이벤트 루프 예시
스택 및 메시지 큐
Job queue and promise
process.nextTick()
setImmediate()
setInterval()
웹 프론트엔드 JS 튜토리얼 nodejs의 이벤트 및 이벤트 루프 이해

nodejs의 이벤트 및 이벤트 루프 이해

Dec 08, 2020 pm 05:40 PM
node.js 이벤트 이벤트 청취 이벤트 중심 비동기 프로그래밍

nodejs의 이벤트 및 이벤트 루프 이해

관련 추천: "nodejs Tutorial"

JavaScript에 익숙한 친구들은 마우스 움직임, 마우스 클릭, 키보드 입력 등의 이벤트를 사용해 본 적이 있을 것입니다. 우리는 해당 처리를 트리거하기 위해 자바스크립트에서 이러한 이벤트를 수신합니다.

nodejs에도 이벤트가 있고, 특화된 처리를 위한 특별 이벤트 모듈도 있습니다.

동시 이벤트와 이벤트 루프도 nodejs에서 비동기 IO를 구축하는 데 매우 중요한 개념입니다.

오늘은 이에 대해 자세히 알아 보겠습니다.

Events

nodejs는 이벤트 전용 모듈인 lib/events.js를 제공합니다.

nodejs를 사용하여 웹 서버를 구축하는 방법에 대해 이야기했던 때를 기억하시나요?

const server = http.createServer((req, res) => {
  res.statusCode = 200
  res.setHeader('Content-Type', 'text/plain')
  res.end('welcome to www.flydean.com\n')
})
로그인 후 복사

여기서 각 요청은 요청 이벤트를 트리거합니다.

nodejs의 핵심 API는 비동기식 이벤트 중심 아키텍처를 기반으로 하기 때문에 nodejs에는 많은 이벤트가 있습니다.

예: net.Server는 새 연결이 있을 때마다 이벤트를 트리거하고, fs.ReadStream은 파일이 열릴 때 이벤트를 트리거하며, stream은 데이터를 읽을 수 있을 때 이벤트를 트리거합니다.

nodejs 이벤트를 구축하는 방법을 살펴보겠습니다.

const EventEmitter = require('events')
const eventEmitter = new EventEmitter()
로그인 후 복사

이벤트에는 일반적으로 on과 Emit이라는 두 가지 메서드가 사용됩니다.

on은 이벤트를 수신하는 데 사용되고, Emit은 이벤트를 트리거하는 데 사용됩니다.

eventEmitter.on('fire', () => {
  console.log('开火')
})

eventEmitter.emit('fire')
로그인 후 복사

emit은 매개변수를 사용할 수도 있습니다.

eventEmitter.on('fire', who => {
  console.log(`开火 ${who}`)
})

eventEmitter.emit('fire', '美帝')
로그인 후 복사

두 매개변수를 다시 살펴보겠습니다.

eventEmitter.on('fire', (who, when) => {
  console.log(`开火 ${who} ${when}`)
})

eventEmitter.emit('fire', '川建国','now')
로그인 후 복사

기본적으로 EventEmitter는 등록 순서에 따라 모든 리스너를 동기적으로 호출합니다. 이는 이벤트의 올바른 순서를 보장하고 경쟁 조건 및 논리 오류를 방지하는 데 도움이 됩니다.

비동기 실행이 필요한 경우 setImmediate() 또는 process.nextTick()을 사용하여 비동기 실행 모드로 전환할 수 있습니다.

eventEmitter.on('fire', (who, when) => {
    setImmediate(() => {
      console.log(`开火 ${who} ${when}`);
  });
})

eventEmitter.emit('fire', '川建国','now')
로그인 후 복사

또한 이벤트는 여러 가지 다른 메서드도 지원합니다.

once(): 단일 리스너 추가

removeListener() / off(): 이벤트에서 이벤트 리스너 제거

removeAllListeners(): 모든 이벤트 리스너 제거

이벤트 루프

우리는 nodejs 코드가 단일 스레드 환경에서 실행되고 한 번에 한 가지 작업만 처리한다는 것을 알고 있습니다.

이 처리 방법은 멀티 스레드 환경에서 데이터 동기화 문제를 방지하고 처리 효율성을 크게 향상시킵니다.

이벤트 루프라고 불리는 것은 프로그램 주기에서 프로세서가 이 주기의 이벤트를 처리한 후 다음 이벤트 주기에 들어가 다음 이벤트 주기의 이벤트를 처리한다는 것을 의미합니다.

이벤트 루프 차단

이벤트 처리 중에 이벤트 처리가 차단되면 다른 이벤트 실행에도 영향을 미치게 되므로 JS에서는 거의 모든 IO가 Non-Blocking인 것을 알 수 있습니다. 이것이 JavaScript에 콜백이 너무 많은 이유이기도 합니다.

이벤트 루프 예시

간단한 이벤트 루프 예시를 살펴보겠습니다:

const action2 = () => console.log('action2')

const action3 = () => console.log('action3')

const action1 = () => {
    console.log('action1')
    action2()
    action3()
}

action1()
로그인 후 복사

위 코드 출력:

action1
action2
action3
로그인 후 복사

스택 및 메시지 큐

우리는 함수 간의 호출이 스택을 통해 구현된다는 것을 알고 있습니다. 호출 시퀀스도 스택을 통해 구현됩니다.

그러나 함수의 모든 메서드가 스택에 푸시되는 것은 아니며 일부 메서드는 메시지 대기열에 배치됩니다.

다른 예를 들어보겠습니다.

const action2 = () => console.log('action2')

const action3 = () => console.log('action3')

const action1 = () => {
    console.log('action1')
    setTimeout(action2, 0)
    action3()
}

action1()
로그인 후 복사

위 코드를 실행한 결과:

action1
action3
action2
로그인 후 복사

결과가 다릅니다. 이는 settimeout이 타이머를 트리거하면 콜백 함수가 스택에 배치되는 대신 처리될 메시지 대기열에 배치되기 때문입니다.

이벤트 루프는 스택에 데이터가 없는 경우에만 메시지 대기열의 이벤트를 소비하도록 전환합니다.

위 예시에서 setTimeout의 타임아웃 시간은 0이지만, action3이 실행될 때까지 기다려야 실행될 수 있습니다.

setTimeout의 시간 제한은 현재 스레드에서 기다리지 않으며 브라우저나 다른 JS 실행 환경에 의해 호출됩니다.

Job queue and promise

ES6의 Promise에는 작업 대기열 개념이 도입되었습니다. 작업 대기열을 사용하면 비동기 함수의 결과를 호출 스택 끝에 배치하는 대신 최대한 빨리 실행합니다.

예:

const action2 = () => console.log('action2')

const action3 = () => console.log('action3')

const action1 = () => {
    console.log('action1')
    setTimeout(action2, 0)
    new Promise((resolve, reject) =>
        resolve('应该在action3之后、action2之前')
    ).then(resolve => console.log(resolve))
    action3()
}

action1()
로그인 후 복사

출력 결과:

action1
action3
应该在action3之后、action2之前
action2
로그인 후 복사

이는 현재 함수가 끝나기 전에 해결된 Promise가 현재 함수 직후에 실행되기 때문입니다.

즉, 스택이 먼저 실행되고 그 다음 작업 큐가 실행되고 마지막으로 메시지 큐가 실행됩니다.

process.nextTick()

먼저 틱이라는 정의를 알려드리겠습니다. 틱은 이벤트 주기를 의미합니다. Process.nextTick()은 다음 이벤트 루프 틱이 시작되기 전에 이 함수를 호출하는 것을 의미합니다.

process.nextTick(() => {
  console.log('i am the next tick');
})
로그인 후 복사

따라서 nextTick은 메시지 대기열의 setTimeout보다 빨라야 합니다.

setImmediate()

nodejs는 코드를 최대한 빨리 실행할 수 있는 setImmediate 메서드를 제공합니다.

setImmediate(() => {
  console.log('I am immediate!');
})
로그인 후 복사

setImmediate의 함수는 이벤트 루프의 다음 반복에서 실행됩니다.

setImmediate() 및 setTimeout(() => {}, 0)의 기능은 기본적으로 유사합니다. 이벤트 루프의 다음 반복에서 모두 실행됩니다.

setInterval()

특정 콜백 함수를 정기적으로 실행하려면 setInterval을 사용해야 합니다.

setInterval(() => {
  console.log('每隔2秒执行一次');
}, 2000)
로그인 후 복사

要清除上面的定时任务,可以使用clearInterval:

const id = setInterval(() => {
  console.log('每隔2秒执行一次');
}, 2000)

clearInterval(id)
로그인 후 복사

注意,setInterval是每隔n毫秒启动一个函数,不管该函数是否执行完毕。

如果一个函数执行时间太长,就会导致下一个函数同时执行的情况,怎么解决这个问题呢?

我们可以考虑在回调函数内部再次调用setTimeout,这样形成递归的setTimeout调用:

const myFunction = () => {
  console.log('做完后,隔2s再次执行!');

  setTimeout(myFunction, 2000)
}

setTimeout(myFunction, 2000)
로그인 후 복사

更多编程相关知识,请访问:编程视频!!

위 내용은 nodejs의 이벤트 및 이벤트 루프 이해의 상세 내용입니다. 자세한 내용은 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를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

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

SublimeText3 중국어 버전

SublimeText3 중국어 버전

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

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

C++ 함수로 비동기 프로그래밍을 구현하는 방법은 무엇입니까? C++ 함수로 비동기 프로그래밍을 구현하는 방법은 무엇입니까? Apr 27, 2024 pm 09:09 PM

요약: C++의 비동기 프로그래밍을 사용하면 시간이 많이 걸리는 작업을 기다리지 않고 멀티태스킹이 가능합니다. 함수 포인터를 사용하여 함수에 대한 포인터를 만듭니다. 콜백 함수는 비동기 작업이 완료되면 호출됩니다. Boost::asio와 같은 라이브러리는 비동기 프로그래밍 지원을 제공합니다. 실제 사례에서는 함수 포인터와 Boost::asio를 사용하여 비동기 네트워크 요청을 구현하는 방법을 보여줍니다.

동시 프로그래밍에서 C++ 함수의 이벤트 중심 메커니즘은 무엇입니까? 동시 프로그래밍에서 C++ 함수의 이벤트 중심 메커니즘은 무엇입니까? Apr 26, 2024 pm 02:15 PM

동시 프로그래밍의 이벤트 중심 메커니즘은 이벤트가 발생할 때 콜백 함수를 실행하여 외부 이벤트에 응답합니다. C++에서는 이벤트 기반 메커니즘을 함수 포인터로 구현할 수 있습니다. 함수 포인터는 이벤트가 발생할 때 실행될 콜백 함수를 등록할 수 있습니다. 람다 표현식은 이벤트 콜백을 구현하여 익명 함수 객체를 생성할 수도 있습니다. 실제 사례에서는 함수 포인터를 사용하여 GUI 버튼 클릭 이벤트를 구현하고, 콜백 함수를 호출하고 이벤트가 발생할 때 메시지를 인쇄합니다.

Java 프레임워크 비동기 프로그래밍의 일반적인 문제 및 솔루션 Java 프레임워크 비동기 프로그래밍의 일반적인 문제 및 솔루션 Jun 04, 2024 pm 05:09 PM

Java 프레임워크 비동기 프로그래밍의 3가지 일반적인 문제와 해결 방법: 콜백 지옥: Promise 또는 CompletableFuture를 사용하여 보다 직관적인 스타일로 콜백을 관리합니다. 리소스 경합: 동기화 기본 요소(예: 잠금)를 사용하여 공유 리소스를 보호하고 스레드로부터 안전한 컬렉션(예: ConcurrentHashMap) 사용을 고려하세요. 처리되지 않은 예외: 작업에서 예외를 명시적으로 처리하고 예외 처리 프레임워크(예: CompletableFuture.Exceptionally())를 사용하여 예외를 처리합니다.

golang 프레임워크는 동시성과 비동기 프로그래밍을 어떻게 처리합니까? golang 프레임워크는 동시성과 비동기 프로그래밍을 어떻게 처리합니까? Jun 02, 2024 pm 07:49 PM

Go 프레임워크는 Go의 동시성 및 비동기 기능을 사용하여 동시 및 비동기 작업을 효율적으로 처리하기 위한 메커니즘을 제공합니다. 1. 동시성은 Goroutine을 통해 달성되어 동시에 여러 작업을 실행할 수 있습니다. 2. 비동기 프로그래밍은 채널을 통해 구현됩니다. 작업을 차단하지 않고 실행할 수 있습니다. 3. HTTP 요청 동시 처리, 데이터베이스 데이터의 비동기 획득 등과 같은 실제 시나리오에 적합합니다.

jQuery에서 선택 요소의 변경 이벤트 바인딩을 구현하는 방법 jQuery에서 선택 요소의 변경 이벤트 바인딩을 구현하는 방법 Feb 23, 2024 pm 01:12 PM

jQuery는 DOM 조작, 이벤트 처리, 애니메이션 효과 등을 단순화하는 데 사용할 수 있는 인기 있는 JavaScript 라이브러리입니다. 웹 개발에서 우리는 선택 요소에 대한 이벤트 바인딩을 변경해야 하는 상황에 자주 직면합니다. 이 기사에서는 jQuery를 사용하여 선택 요소 변경 이벤트를 바인딩하는 방법을 소개하고 특정 코드 예제를 제공합니다. 먼저 라벨을 사용하여 옵션이 포함된 드롭다운 메뉴를 만들어야 합니다.

PHP 비동기 프로그래밍의 장점과 단점은 무엇입니까? PHP 비동기 프로그래밍의 장점과 단점은 무엇입니까? May 06, 2024 pm 10:00 PM

PHP에서 비동기 프로그래밍의 장점에는 더 높은 처리량, 더 낮은 대기 시간, 더 나은 리소스 활용도 및 확장성이 포함됩니다. 단점으로는 복잡성, 디버깅의 어려움, 제한된 라이브러리 지원 등이 있습니다. 실제 사례에서 ReactPHP는 WebSocket 연결을 처리하는 데 사용되어 비동기 프로그래밍의 실제 적용을 보여줍니다.

Python 비동기 프로그래밍: 비동기 코드에서 효율적인 동시성을 달성하는 방법 Python 비동기 프로그래밍: 비동기 코드에서 효율적인 동시성을 달성하는 방법 Feb 26, 2024 am 10:00 AM

1. 비동기 프로그래밍을 사용하는 이유는 무엇입니까? 기존 프로그래밍에서는 차단 I/O를 사용합니다. 즉, 프로그램은 작업을 계속하기 전에 작업이 완료될 때까지 기다립니다. 이는 단일 작업에 적합할 수 있지만 많은 수의 작업을 처리할 때 프로그램 속도가 느려질 수 있습니다. 비동기 프로그래밍은 기존 차단 I/O의 한계를 깨고 비차단 I/O를 사용합니다. 즉, 프로그램은 작업이 완료될 때까지 기다리지 않고 실행을 위해 여러 스레드나 이벤트 루프에 작업을 배포할 수 있습니다. 이를 통해 프로그램은 여러 작업을 동시에 처리할 수 있어 프로그램의 성능과 효율성이 향상됩니다. 2. Python 비동기 프로그래밍의 기본 Python 비동기 프로그래밍의 기본은 코루틴과 이벤트 루프입니다. 코루틴은 함수가 일시 중지와 재개 사이를 전환할 수 있도록 하는 함수입니다. 이벤트 루프는 일정 관리를 담당합니다.

PHP를 사용하여 이벤트 기반 애플리케이션을 구축하는 방법 PHP를 사용하여 이벤트 기반 애플리케이션을 구축하는 방법 May 04, 2024 pm 02:24 PM

PHP에서 이벤트 기반 애플리케이션을 구축하는 방법에는 EventSourceAPI를 사용하여 이벤트 소스를 생성하고 EventSource 객체를 사용하여 클라이언트 측에서 이벤트를 수신하는 방법이 포함됩니다. SSE(Server Sent Events)를 사용하여 이벤트를 보내고 XMLHttpRequest 객체를 사용하여 클라이언트 측에서 이벤트를 수신합니다. 실제적인 예는 EventSource를 사용하여 전자 상거래 웹 사이트에서 실시간으로 재고 수를 업데이트하는 것입니다. 이는 재고를 무작위로 변경하고 업데이트를 보내는 방식으로 서버 측에서 이루어지며, 클라이언트는 EventSource를 통해 재고 업데이트를 수신하고 이를 표시합니다. 실시간.

See all articles