search
HomeWeb Front-endJS TutorialIntroduction to the principles and implementation techniques of asynchronous javascript_javascript techniques

Due to work needs, I need to write a script on the web page to submit data to the system in batches through the web page. So I thought of the Greasemonkey plug-in, so I started writing it and found that the problem was solved smoothly. But when summarizing and organizing the script, I habitually asked myself a question: Can it be simpler?
My answer is of course “yes”.

First review my requirements for batch submission of data: I have a batch of user data to be inserted into the system, but because the system library table structure is not determinant, it cannot be converted into SQL statements for insertion. There are nearly 200 pieces of data to be inserted. Even if it is entered into the system manually, it will probably take a day. As a programmer, of course I would not do such a stupid thing, I must use a program to solve it. This programming process took me 1 day. Compared with manual entry, my extra income is from this blog post, which is absolutely cost-effective!

It didn’t take any time to choose the programming platform. I directly chose to write my own script based on Greasemonkey. The browser was of course Firefox. The working process of the script:
Pre-store the data to be inserted in the script
Simulate a mouse click to open the input window on the page
Enter the data into the input window, and simulate clicking the "Submit" button to submit the data submitted to the system.
Loop in sequence until all data is processed.

The technical difficulty here is:
To open the input window, you need to wait for an irregular period of time, depending on the network conditions.
Submit data to the background, and you need to wait for the processing to be completed before you can cycle to the next data.
If I were a novice, of course I would directly write an application logic similar to this:

Copy code The code is as follows:

for(var i = 0; i { 3: clickButtonForInputWindow();
waitInputWindow();
enterInputData(dataArray[i] );
clickSubmitButton();
waitInputWindowClose();
}

In fact, all browsers will fall into a white screen after a few minutes and prompt "No response" and was forcibly terminated. The reason is that when the browser calls JavaScript, the main interface stops responding because the CPU is handed over to js for execution and there is no time to process interface messages.

In order to meet the requirement of "no locking", we can modify the script as follows:
Copy code The code is as follows:

for(var i = 0; i {
setTimeout(clickButtonForInputWindow);

setTimeout(waitInputWindowClose );
}

In fact, setTimeout and setInterval are the only asynchronous operations that browsers can support. How to use these two functions more elegantly to implement asynchronous operations? The current simple answer is Lao Zhao's Wind.js. Although I have not used this function library, the $await call alone meets my consistent requirement for simplicity. But for a single file script like mine, downloading an external js library from the Internet is obviously not as fast and enjoyable as copying a piece of code that supports asynchronous operations.

So I decided to find another way and make an asynchronous function library that does not require compilation and is easier to use and can be copied and pasted.

Before talking about asynchronous, let’s recall the several structural types of synchronous operations:

Sequence: It is the order in which statements are executed
Judgment: It is the judgment statement
Loop: strict It should be jump (goto), but most modern languages ​​have canceled goto. A loop should actually be a composite structure, a combination of if and goto.
The difficulty of asynchronous operations lies in two places:

Asynchronous judgment: The judgment in asynchronous situations is basically to detect that the conditions are satisfied, and then perform certain actions.
Asynchronous sequence: After each step in the sequence, control is returned and the next step is waited for in the next time slice. The difficulty is maintaining order. Especially when there is an asynchronous loop between two sequential actions.
Asynchronous loop: After each loop, control is returned to the browser, and this loop continues until the end of the run.
The simplest implementation is of course an asynchronous loop. My implementation code is as follows:
Copy code The code is as follows:

function asyncWhile(fn, interval)
{
if( fn == null || (typeof(fn) != "string" && typeof(fn) != "function") )
return;
var wrapper = function()
{
if( (typeof(fn) == "function" ? fn() : eval(fn) ) !== false )
setTimeout(wrapper, interval == null? 1: interval);
}
wrapper();
}

The core content is: if the return value of the fn function is not false, continue with the next setTimeout registration call.

In fact, the "wait and execute" logic is basically an asynchronous loop problem. An example of how to implement this situation is as follows:
Copy code The code is as follows:

asyncWhile(function (){
if( xxxCondition == false )
return true; // Indicates that the loop continues
else
doSomeThing();
return false; // Indicates that there is no need to continue the loop
});

For non-waiting and execution logic, a simple setTimeout is enough.
Asynchronous is easy, but the difficulty is to implement the order in asynchronous. The earliest reason was that I wanted to implement 3 steps, but the second part was an asynchronous loop of more than 100 times. In other words, the 3-step operation I want to implement is actually 103 sequential asynchronous operations. In order to find a way to implement responsive waiting in the browser, I searched my brain and only found an implementation in Firefox, and I had to apply for privileged calls.
Finally, I came up with a simple method, which is to introduce the concept of "Execution Chain". All registration functions of the same execution chain are sequential, and there is no relationship between different execution chains. In addition, it does not provide concepts such as mutual exclusion (mutex). If you want to synchronize, check it yourself in the code.
In the same execution chain, an execution token is saved. Only when the token matches the function serial number can execution be allowed, thus ensuring the order of asynchronous execution.
Copy code The code is as follows:

function asyncSeq(funcArray, chainName, abortWhenError)
{
if( typeof(funcArray) == "function" )
return asyncSeq([funcArray], chainName, abortWhenError);

if( funcArray == null || funcArray.length == 0 )
return;

if( chainName == null ) chainName = "__default_seq_chain__";
var tInfos = asyncSeq.chainInfos = asyncSeq.chainInfos || {};
var tInfo = tInfos[chainName] = tInfos[chainName] || {count : 0, currentIndex : -1, abort : false};

for(var i = 0; i {
asyncWhile(function(item, tIndex){
return function(){
if( tInfo.abort )
return false;
if( tInfo.currentIndex return true;
else if( tInfo.currentIndex == tIndex )
{
try{
item();
}
catch(e){
if ( abortWhenError ) tInfo.abort = true;
}
finally{
tInfo.currentIndex ;
}
}
else
{
if( abortWhenError ) tInfo. abort = true;
}
return false;
};
}(funcArray[i], tInfo.count ));
}

setTimeout(function() {
if( tInfo.count > 0 && tInfo.currentIndex == -1 )
tInfo.currentIndex = 0;
},20); // For debugging reasons, a delayed start is added
}

Thus, an asynchronous js function library supporting Copy&Paste is completed. Specific usage examples are as follows:
Copy code The code is as follows:

function testAsync()
{
asyncSeq([function(){println("aSyncSeq -0 ");}
, function(){println("aSyncSeq -1 ");}
, function(){println("aSyncSeq -2 ");}
, function(){println("aSyncSeq -3 ");}
, function(){println("aSyncSeq -4 ");}
, function(){println("aSyncSeq -5 ");}
, function(){println("aSyncSeq -6 ");}
, function(){println("aSyncSeq -7 ");}
, function(){println("aSyncSeq -8 ");}
, function(){println("aSyncSeq -9 ");}
, function(){println("aSyncSeq -10 ");}
, function(){println("aSyncSeq -11 ");}
, function(){println("aSyncSeq -12 ");}
, function(){println("aSyncSeq -13 ");}
, function(){println("aSyncSeq -14 ");}
, function(){println("aSyncSeq -15 ");}
, function(){println("aSyncSeq -16 ");}
, function(){println("aSyncSeq -17 ");}
, function(){println("aSyncSeq -18 ");}
, function(){println("aSyncSeq -19 ");}
, function(){println("aSyncSeq -20 ");}
, function(){println("aSyncSeq -21 ");}
, function(){println("aSyncSeq -22 ");}
, function(){println("aSyncSeq -23 ");}
, function(){println("aSyncSeq -24 ");}
, function(){println("aSyncSeq -25 ");}
, function(){println("aSyncSeq -26 ");}
, function(){println("aSyncSeq -27 ");}
, function(){println("aSyncSeq -28 ");}
, function(){println("aSyncSeq -29 ");}
]);

asyncSeq([function(){println("aSyncSeq test-chain -a0 ");}
, function(){println("aSyncSeq test-chain -a1 ");}
, function(){println("aSyncSeq test-chain -a2 ");}
, function(){println("aSyncSeq test-chain -a3 ");}
, function(){println("aSyncSeq test-chain -a4 ");}
, function(){println("aSyncSeq test-chain -a5 ");}
, function(){println("aSyncSeq test-chain -a6 ");}
, function(){println("aSyncSeq test-chain -a7 ");}
, function(){println("aSyncSeq test-chain -a8 ");}
], "test-chain");

asyncSeq([function(){println("aSyncSeq -a0 ");}
, function(){println("aSyncSeq -a1 ");}
, function(){println("aSyncSeq -a2 ");}
, function(){println("aSyncSeq -a3 ");}
, function(){println("aSyncSeq -a4 ");}
, function(){println("aSyncSeq -a5 ");}
, function(){println("aSyncSeq -a6 ");}
, function(){println("aSyncSeq -a7 ");}
, function(){println("aSyncSeq -a8 ");}
]);
}

var textArea = null;

function println(text)
{
if( textArea == null )
{
textArea = document.getElementById("text");
textArea.value = "";
}

textArea.value = textArea.value text "rn";
}

最后,要向大家说一声抱歉,很多只想拿代码的朋友恐怕要失望了,如果你真的不知道怎么处理这些多余的行号,你可以学习一下正则表达式的替换,推荐用UltraEdit。
Statement
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
快速应用:PHP 异步 HTTP 下载多个文件的实用开发案例分析快速应用:PHP 异步 HTTP 下载多个文件的实用开发案例分析Sep 12, 2023 pm 01:15 PM

快速应用:PHP异步HTTP下载多个文件的实用开发案例分析随着互联网的发展,文件下载功能已成为很多网站和应用程序的基本需求之一。而对于需要同时下载多个文件的场景,传统的同步下载方式往往效率低下且耗费时间。为此,使用PHP异步HTTP下载多个文件成为了一种越来越常见的解决方案。本文将通过一个实际的开发案例,详细分析如何使用PHP异步HTTP

Swoole如何支持异步SMTP操作Swoole如何支持异步SMTP操作Jun 25, 2023 pm 12:24 PM

随着互联网的不断发展和普及,电子邮件已经成为了人们生活和工作中必不可少的一部分,而SMTP(SimpleMailTransferProtocol,简单邮件传输协议)则是邮件发送的重要协议之一。Swoole作为PHP的一个异步网络通讯框架,可以很好地支持异步SMTP操作,使邮件发送更加高效和稳定。本文将介绍Swoole如何支持异步SMTP操作,包括使用步

Vue文档中的异步请求函数的使用方法Vue文档中的异步请求函数的使用方法Jun 20, 2023 pm 05:55 PM

Vue.js是一种流行的前端JavaScript框架,它提供了一种在应用程序中构建用户界面的方式。在Vue.js的文档中,我们可以找到很多有用的信息,特别是关于如何使用异步请求函数。异步请求函数是一种在应用程序中执行异步任务的方式。它们被用于从服务器获取数据、处理输入、验证表单等。一般情况下,异步请求函数需要与Promise、async和await等Java

Swoole如何支持异步AMQP操作Swoole如何支持异步AMQP操作Jun 25, 2023 am 08:22 AM

随着互联网业务量的不断增长,对于高并发和高性能的需求越来越高,而Swoole作为PHP的一款网络通信框架,也越来越受到开发者的青睐。其中,Swoole支持异步AMQP是比较常见的应用场景之一。那么我们来看看Swoole如何支持异步AMQP操作。首先,我们需要明确什么是AMQP。AMQP(AdvancedMessageQueuingProtocol)高级

如何使用ThinkPHP6进行异步日志记录操作?如何使用ThinkPHP6进行异步日志记录操作?Jun 12, 2023 am 09:57 AM

随着互联网的高速发展,日志记录服务成为了每个大型web应用必不可少的模块。为了方便错误排查、性能监控等各种需求,本文将介绍如何使用ThinkPHP6框架进行异步日志记录操作。1.什么是日志记录在计算机科学领域,日志记录是指将计算机系统中发生的事件和信息记录下来。通常,这些记录都以文件或数据库的形式存储。日志记录有助于了解系统运行状况,及时发现和解决

Swoole如何支持异步SSH操作Swoole如何支持异步SSH操作Jun 25, 2023 am 11:10 AM

Swoole是一个为高并发而设计的PHP扩展,可以大幅提升PHP的性能。它支持异步IO、协程、多进程等特性,在网络编程、高负载场景中表现出色。本文将介绍Swoole如何支持异步SSH操作。一、SSH介绍SSH(SecureShell)是一种加密的网络协议,用来在网络中进行安全地传输信息。SSH协议具有安全、可靠、跨平台等特点,广泛应用于远程登录、文件传输、

PHP8.1新增的异步HTTP客户端PHP8.1新增的异步HTTP客户端Jul 08, 2023 pm 07:24 PM

PHP8.1新增的异步HTTP客户端随着互联网的快速发展,各种Web应用程序的性能也变得越来越重要。为了提供更好的用户体验,开发人员需要使用高效的工具和技术来处理各种网络请求。幸运的是,PHP8.1引入了一个全新的功能,即异步HTTP客户端,它允许我们以非阻塞的方式执行HTTP请求,从而提高应用程序的性能。通过异步HTTP客户端,我们可以在发送请求后继续执行

如何利用Celery、Redis和Django实现异步任务队列如何利用Celery、Redis和Django实现异步任务队列Sep 27, 2023 pm 11:18 PM

如何利用Celery、Redis和Django实现异步任务队列引言:在Web开发中,经常需要处理一些耗时较长的任务,如发送邮件、生成报表、处理大量数据等。如果将这些任务直接放在视图函数中处理,会导致请求响应时间过长,用户体验不佳。为了提高系统的性能和响应速度,我们可以使用异步任务队列来处理这些耗时的任务。Celery是一个广泛使用的Python的异步任务队列

See all articles

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

AI Hentai Generator

AI Hentai Generator

Generate AI Hentai for free.

Hot Article

Repo: How To Revive Teammates
1 months agoBy尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Energy Crystals Explained and What They Do (Yellow Crystal)
2 weeks agoBy尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Adventure: How To Get Giant Seeds
4 weeks agoBy尊渡假赌尊渡假赌尊渡假赌

Hot Tools

Zend Studio 13.0.1

Zend Studio 13.0.1

Powerful PHP integrated development environment

Notepad++7.3.1

Notepad++7.3.1

Easy-to-use and free code editor

Atom editor mac version download

Atom editor mac version download

The most popular open source editor

SAP NetWeaver Server Adapter for Eclipse

SAP NetWeaver Server Adapter for Eclipse

Integrate Eclipse with SAP NetWeaver application server.

MinGW - Minimalist GNU for Windows

MinGW - Minimalist GNU for Windows

This project is in the process of being migrated to osdn.net/projects/mingw, you can continue to follow us there. MinGW: A native Windows port of the GNU Compiler Collection (GCC), freely distributable import libraries and header files for building native Windows applications; includes extensions to the MSVC runtime to support C99 functionality. All MinGW software can run on 64-bit Windows platforms.