This program also calls the getArticle() function internally. However, it should be noted that this version of the getArticle() function designed for asynchronous communication must receive a function as the second parameter. When calling the getArticle() function, a request was sent to the server as before. The difference is that the function will now return quickly instead of waiting for the server's response. This means that when the execution rights are returned to the calling program, the server has not received a response. In this way, the thread can execute other tasks until it obtains a server response and call the callback function at this time. Once the server response is obtained, the second parameter of getArticle() will be called as a predefined callback function, and the server's return value is its parameter. Similarly, getArticleWithCache() also needs to make some changes, defining a callback parameter as its second parameter. This callback function will be called in the callback function passed to getArticle(), so it can be executed after the server communication is over.
You may already think that the above changes are quite complicated, but the changes to the backgroundLoad() function will be more complicated, and it will also be rewritten to the form that can handle callback functions:
function backgroundLoad ( ids, callback ) {
var i = 0;
function l ( ) {
if ( i < ) {
getArticleWithCache(ids[i++], function( a ){
backgroundLoad(, l);
});
} else {
callback();
}
}
l();
}
The modified backgroundLoad() function looks far from our previous function, but the functions they implement are no different. This means that both functions accept ID arrays as parameters, and call getArticleWithCache() for each element in the array, and then call backgroundLoad() recursively using the sub-article ID that has been obtained. However, it is also a loop access to the array, so it is not easy to identify the new function. In the previous program, it was completed with a for loop statement. Why are the two sets of functions that implement the same function so different?
This difference stems from the fact that any function must return immediately after encountering a situation where it is necessary to communicate with the server, such as getArticleWithCache(). Unless the original function is not being executed, the callback function that should accept the server response cannot be called. For JavaScript, it is impossible to interrupt the program during the loop and continue to execute the program later from this breakpoint, such as a for statement. Therefore, this example uses a recursive pass callback function to implement a loop structure instead of a traditional loop statement. For those familiar with the continuous transfer style (CPS), this is a manual implementation of CPS. Because loop syntax cannot be used, even a simple program as traversing the tree mentioned above has to be very complicated to write. The problem related to event driver design is the control flow problem: loops and other control flow expressions can be difficult to understand.
There is another problem here: if you convert a function that does not apply asynchronous communication to a function that uses asynchronous communication, the rewritten function will require a callback function as a new parameter, which creates a big problem for existing APIs, because the internal changes do not limit the impact to the internal, but lead to overall confusing APIs and changes to other users of the API.
What is the fundamental reason for these problems? Yes, it is the JavaScript single-threaded mechanism that causes these problems. Executing asynchronous communication in a single thread requires event driver design and complex statements. If another thread can handle user requests while the program is waiting for the server's response, then the above complex technology is not needed.
Try multithreaded programming
Let me introduce it, it is a library that allows JavaScript to perform multithreaded programming, and applying it can greatly alleviate the difficulties associated with asynchronous communication in AJAX development mentioned above. This is a free software library written in JavaScript. The prerequisite for using it is to comply with the two protocols, Mozilla Public License and GNU General Public License. You can download the source code from their website.
Download and use the source code now! Suppose you have saved the downloaded source code to a folder named, and before performing any operations, run the following program first. This is a very simple function implementation:
<script type="text/javascript" src=""> </script>
<script type="text/javascript">
(function(){
var i = 0;
while ( 1 ) {
+= i++ + " <br>";
}
});
</script>
Execute this program will display numbers starting from 0 in sequence, and they appear one by one, and you can scroll to see it. Now let's take a closer look at the code. It uses the while(1) condition to create a loop that will not abort. Usually, constantly using a JavaScript program like this and is the only thread will cause the browser to look like it is frozen, and naturally it will not allow you to scroll. So why does the above program allow you to do this? The key is the () statement above while(1), which is a method provided by this library, which can create a new thread. The function passed in as a parameter is executed in this new thread. Let us make the following fine-tuning of the program:
<script type="text/javascript" src=""> </script>
<script type="text/javascript">
function f ( i ){
while ( 1 ) {
+= i++ + " <br>";
}
}
(f, 0);
(f, 100000);
</script>
In this program, there is a new function f() that can display numbers repeatedly. It is defined at the beginning of the program segment. Then, the create() method is called twice with f() as the parameter. The second parameter passed to the create() method will be passed to f() without modification. When you execute this program, you will first see some decimals starting from 0, followed by some large numbers starting from 100,000, and then the numbers in the order of the previous decimals. You can observe that the program is displaying decimals and large numbers alternately, which means that the two threads are running at the same time.
Let me show another usage. The above example calls the create() method to create a new thread. It is also possible to achieve this without calling any APIs in Curry. For example, the previous example can be written like this:
<script type="text/javascript" src=""> </script>
<script type="text/-js">
var i = 1;
while ( 1 ) {
+= i++ + " <br>";
}
</script>
Within the script tag, it is very simple to write an infinite loop in JavaScript. You should notice the type attribute in the tag, which is a very unfamiliar value (text/x- -js). If this attribute is placed in the script tag, the program between tags will be executed in a new thread. You should remember that in this case, the library must be included.
With this, it is possible to freely switch between threads of the execution environment, even if your program is long and has strong continuity. We can briefly discuss how to do this. In short, code conversion is required. Roughly speaking, first, convert the function passed to create() into a string, and then rewrite it until it can be executed in batches. These programs can then be executed step by step according to the scheduler. The scheduler is responsible for coordinating multithreading, in other words, it can make adjustments at the right time so that every modified function will have the same chance to run. In fact, no new thread was created, but a multi-threaded environment was simulated based on the original single thread.
Although the converted function seems to run in a different thread, there is actually only one thread doing all of this. Performing synchronous communication within the converted function will still cause the browser to freeze, and you may think that the previous problems have not been solved at all. But you don't have to worry about it. It provides a customized communication library implemented using JavaScript asynchronous communication method. It is designed to allow other threads to run when a thread is waiting for the server's response. This communication is in stock below. Its usage is as follows:
<script type="text/javascript" src=""> </script>
<script type="text/-js">
var req = (url, ["Accept", "*"]);
if ( == 200) {
alert();
} else {
alert();
}
</script>
The get() method, as its name implies, can obtain the contents of the specified URL through the HTTP GET method, which takes the target URL as the first parameter and an array representing the HTTP request header as the optional second parameter. The get() method interacts with the server. When the server's response is obtained, it returns an XMLHttpRequest object as the return value. When the get() method returns, the server response has been received, so there is no need to use the callback function to receive the result. Of course, there is no need to worry about the browser freezing when the program waits for the server to respond. In addition, there is also a post() method that can be used to send data to the server:
<script type="text/javascript" src=""> </script>
<script type="text/-js">
var req = (url, "key1=val1&key2=val2");
alert();
</script>
The post() method takes the destination URL as the first parameter and the content to be sent as the second parameter. Like the get() method, you can also use the request header as an optional third parameter.
If you use this communication library to implement the getArticle() method in the first example, then you will soon be able to write getArticleWithCache(), backgroundLoad() and other functions that call the getArticle() method using the simple method example at the beginning of the article. Even if that version of backgroundLoad() is reading the article data, as usual, there is another thread that can respond to user requests, so the browser will not freeze. Now, can you understand how practical it is to apply multithreading in JavaScript?
Want to know more
I've introduced you to a library that can apply multithreading in JavaScript:. The content of this article is just a very basic thing. If you want to have a deeper understanding, I recommend you to go to the tutorial. It provides more about usage and lists documentation available for advanced users, making it the best material to get started. It's also nice to visit their website, where more information is provided.
Related Authors
Daisuke Maki: After graduating from the Natural Sciences Division of the School of Arts and Arts at International Christian University (obtaining a bachelor's degree in arts), he studied for a master's degree in information from the Graduate School of Electro-Communications University. AJAX specializes in web development and application of JavaScript. He developed it. This design was applied in the 2006 fiscal year in the Exploratory Software Project, a project directed by the Japan Information Technology Promotion Agency (IPA).
He currently holds a master's degree in engineering and is enrolled in the graduate school of Electro-Communications University to pursue his PhD.
Download address///jiaoben/
Previous page12Read the full text