연속적으로 호출해야 하는 여러 함수 fn1, fn2 및 fn3이 있다고 가정해 보겠습니다. 물론 가장 간단한 방법은 다음과 같습니다.
fn1(); fn2(); fn3();
그러나 때로는 이러한 함수가 런타임 중에 하나씩 추가되고 다음과 같은 경우 호출되지 않습니다. 현재는 어떤 함수가 있는지는 모르겠지만, 함수를 추가할 때 그 안에 함수를 하나씩 꺼내서 사용하면 됩니다. 순서대로 호출하세요:
var stack = []; // 执行其他操作,定义fn1 stack.push(fn1); // 执行其他操作,定义fn2、fn3 stack.push(fn2, fn3); // 调用的时候 stack.forEach(function(fn) { fn() });
이러한 함수에는 이름이 없더라도 상관 없으며 익명 함수를 직접 전달할 수 있습니다. 테스트해 보겠습니다.
var stack = []; function fn1() { console.log('第一个调用'); } stack.push(fn1); function fn2() { console.log('第二个调用'); } stack.push(fn2, function() { console.log('第三个调用') }); stack.forEach(function(fn) { fn() }); // 按顺序输出'第一个调用'、'第二个调用'、'第三个调用'
이 구현은 지금까지 잘 작동했지만 비동기 함수 호출이라는 한 가지 상황을 무시했습니다. 비동기는 JavaScript에서 피할 수 없는 주제입니다. 여기서는 JavaScript의 비동기와 관련된 다양한 용어와 개념을 논의하지 않을 것입니다. 독자들이 직접 확인해 보시기 바랍니다(예: 유명한 논평). 다음 코드가 1, 3, 2를 출력한다는 것을 알고 계시다면 계속해서 읽어주세요:
console.log(1); setTimeout(function() { console.log(2); }, 0); console.log(3);
스택 큐에 유사한 비동기 함수인 함수가 있으면 구현이 엉망이 될 것입니다. :
var stack = []; function fn1() { console.log('第一个调用') }; stack.push(fn1); function fn2() { setTimeout(function fn2Timeout() { console.log('第二个调用'); }, 0); } stack.push(fn2, function() { console.log('第三个调用') }); stack.forEach(function(fn) { fn() }); // 输出'第一个调用'、'第三个调用'、'第二个调用'
문제는 분명합니다. fn2는 실제로 순서대로 호출되지만 setTimeout의 fn2Timeout() { console.log('Second call') } 함수는 즉시 실행되지 않습니다(시간 초과가 다음으로 설정된 경우에도 마찬가지). 0) ; fn2가 호출된 후 즉시 반환하고 fn3을 실행합니다. fn3이 실행되면 실제로는 fn2Timeout의 차례입니다.
어떻게 해결하나요? 여기서 핵심은 fn2Timeout입니다. fn3을 호출하기 전에 실제로 실행될 때까지 기다려야 합니다. 이상적으로는 다음과 같습니다.
function fn2() { setTimeout(function() { fn2Timeout(); fn3(); }, 0); }
그러나 그렇게 하는 것은 원래 fn2Timeout을 제거하고 교체하는 것과 같습니다. 하나의 New 함수를 사용하여 원래 fn2Timeout 및 fn3을 삽입합니다. 원래 기능을 동적으로 변경하는 이 방법에는 Monkey Patch라는 특별한 용어가 있습니다. 우리 프로그래머들의 진언에 따르면: "확실히 이루어질 수 있다"이지만, 작성하기가 약간 어색하고 직접 참여하기가 쉽습니다. 더 좋은 방법이 있나요?
한 걸음 물러나 fn3을 실행하기 전에 fn2Timeout이 완전히 실행될 때까지 기다리지 않고 대신 fn2Timeout 함수 본문의 마지막 줄에서 호출합니다.
function fn2() { setTimeout(function fn2Timeout() { console.log('第二个调用'); fn3(); // 注{1} }, 0); }
이것이 더 좋아 보이지만, fn2 정의 fn3이 없습니다. 이 fn3은 어디서 나온 것입니까?
또 다른 문제가 있습니다. fn2에서 fn3을 호출해야 하기 때문에 stack.forEach를 통해 fn3을 호출할 수 없습니다. 그렇지 않으면 fn3이 두 번 호출됩니다.
fn3을 fn2에 쓸 수 없습니다. 대신 fn2Timeout이 끝날 때 스택에서 fn2의 다음 함수를 찾은 후 다음을 호출하면 됩니다.
function fn2() { setTimeout(function fn2Timeout() { console.log('第二个调用'); next(); }, 0); }
이 다음 함수는 스택에서 다음 함수를 찾아 실행하는 일을 담당합니다. 이제 next를 구현해 봅시다.
var index = 0; function next() { var fn = stack[index]; index = index + 1; // 其实也可以用shift 把fn 拿出来 if (typeof fn === 'function') fn(); }
next는 stack[index]를 사용하여 함수를 스택에 가져옵니다. next가 호출될 때마다 인덱스는 1씩 증가하여 다음 항목을 꺼내는 목적을 달성합니다. 기능.
Next는 다음과 같이 사용됩니다.
var stack = []; // 定义index 和next function fn1() { console.log('第一个调用'); next(); // stack 中每一个函数都必须调用`next` }; stack.push(fn1); function fn2() { setTimeout(function fn2Timeout() { console.log('第二个调用'); next(); // 调用`next` }, 0); } stack.push(fn2, function() { console.log('第三个调用'); next(); // 最后一个可以不调用,调用也没用。 }); next(); // 调用next,最终按顺序输出'第一个调用'、'第二个调用'、'第三个调用'。
이제 stack.forEach 라인이 삭제되었으므로 Next를 직접 호출하여 실행할 스택에서 첫 번째 함수 fn1을 찾고 fn1에서 next를 호출합니다. 알아내기 다음 함수 fn2가 실행된 후 fn2에서 다음 함수가 호출되는 식입니다.
모든 함수는 다음을 호출해야 합니다. 특정 함수에 작성되지 않은 경우 프로그램은 계속할 메커니즘 없이 함수 실행 후 바로 종료됩니다.
이 기능 대기열 구현을 이해한 후에는 다음 인터뷰 질문을 해결할 수 있어야 합니다.
// 实现一个LazyMan,可以按照以下方式调用: LazyMan(“Hank”) /* 输出: Hi! This is Hank! */ LazyMan(“Hank”).sleep(10).eat(“dinner”)输出 /* 输出: Hi! This is Hank! // 等待10秒.. Wake up after 10 Eat dinner~ */ LazyMan(“Hank”).eat(“dinner”).eat(“supper”) /* 输出: Hi This is Hank! Eat dinner~ Eat supper~ */ LazyMan(“Hank”).sleepFirst(5).eat(“supper”) /* 等待5秒,输出 Wake up after 5 Hi This is Hank! Eat supper */ // 以此类推。
Node.js의 유명한 연결 프레임워크는 이러한 방식으로 미들웨어 대기열을 구현합니다.
주의하면 다음은 당분간 함수의 끝 부분에만 배치할 수 있다는 것을 알 수 있습니다. 중간에 배치하면 원래 문제가 계속 발생합니다.
function fn() { console.log(1); next(); console.log(2); // next()如果调用了异步函数,console.log(2)就会先执行 }
redux와 koa는 서로 다른 구현을 통해 사용될 수 있습니다. Next는 후속 기능을 실행한 후 다시 돌아가서 다음 코드를 실행합니다. 시간 나면 다시 쓰세요.
위 내용은 JavaScript 함수 비동기 실행 구현 코드에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!