js同步程序是如何向異步程序演變的
js的異步調(diào)用很重要,凡是涉及到網(wǎng)絡(luò)調(diào)用和事件機(jī)制的代碼都會用到它。第一眼看上去的時候異步調(diào)用很特別,和之前設(shè)計(jì)程序使用的同步調(diào)用方法很不一樣。實(shí)質(zhì)上他們之前的區(qū)別沒有相像中那么大。本文嘗試用幾個例子說明同步程序是如何向異步程序演變的。
從C/C++的同步調(diào)用開始
1 使用C語言的編碼方式實(shí)現(xiàn)調(diào)用訪問遠(yuǎn)程的接口
view plaincopy to clipboardprint?
int get_data()
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
socket s = new Socket();
Connnect(s, ip, port);
send(s, bufCmd);
recv(s, bufRcv);
use(bufRcv);
return 0;
}
int get_data()
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
socket s = new Socket();
Connnect(s, ip, port);
send(s, bufCmd);
recv(s, bufRcv);
use(bufRcv);
return 0;
}
2 將通信過程封裝成獨(dú)立的函數(shù),簡化業(yè)務(wù)流程代碼
view plaincopy to clipboardprint?
// 發(fā)包收包的過程
int send_and_recv(struct addr, char* bufCmd, char* bufRcv)
{
socket s = new Socket();
Connnect(s, addr.ip, addr.port);
send(s, bufCmd);
recv(s, bufRcv);
}
// 原來的業(yè)務(wù)流程
int get_data_v2()
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
// addr={ip, port}
send_and_recv(addr, bufCmd, bufRcv);
use(bufRcv);
return 0;
}
// 發(fā)包收包的過程
int send_and_recv(struct addr, char* bufCmd, char* bufRcv)
{
socket s = new Socket();
Connnect(s, addr.ip, addr.port);
send(s, bufCmd);
recv(s, bufRcv);
}
// 原來的業(yè)務(wù)流程
int get_data_v2()
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
// addr={ip, port}
send_and_recv(addr, bufCmd, bufRcv);
use(bufRcv);
return 0;
}
3 將通信過程變成異步調(diào)用
view plaincopy to clipboardprint?
// 變成異步調(diào)用以后,原來的調(diào)用過程分成了兩段
// 前半段組裝參數(shù)調(diào)用發(fā)包過程
// 后半段處理返
// 這里假設(shè)send_and_recv是一個異步的網(wǎng)絡(luò)通信函數(shù)
void get_data_v3()
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv_async(addr, bufCmd, bufRcv, callback); } // end of get_data_v4
// definition of call back function
int callback(char* bufRcv) {
use(bufRcv);
return 0;
}
// 變成異步調(diào)用以后,原來的調(diào)用過程分成了兩段
// 前半段組裝參數(shù)調(diào)用發(fā)包過程
// 后半段處理返
// 這里假設(shè)send_and_recv是一個異步的網(wǎng)絡(luò)通信函數(shù)
void get_data_v3()
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv_async(addr, bufCmd, bufRcv, callback); } // end of get_data_v4
// definition of call back function
int callback(char* bufRcv) {
use(bufRcv);
return 0;
}
4 假設(shè)處理結(jié)果的時候依賴外部參數(shù)
view plaincopy to clipboardprint?
// 這里原來的業(yè)務(wù)流程需要外部傳進(jìn)來的兩個參數(shù)(a,b)來決定如何處理結(jié)果
int get_data_v4(int a, int b)
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv(addr, bufCmd, bufRcv);
use(bufRcv, a, b);
return 0;
}
// 這里原來的業(yè)務(wù)流程需要外部傳進(jìn)來的兩個參數(shù)(a,b)來決定如何處理結(jié)果
int get_data_v4(int a, int b)
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv(addr, bufCmd, bufRcv);
use(bufRcv, a, b);
return 0;
}
5 加上參數(shù)依賴后再變成異步調(diào)用
view plaincopy to clipboardprint?
// 需要參數(shù)的異步調(diào)用需要將參數(shù)透傳到后半段的回調(diào)函數(shù)中
void get_data_v5(int a, int b)
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv_async(addr, bufCmd, bufRcv, callback); } // end of get_data_v5
// definition of call back function
int callback(char* bufRcv, int a, int b) {
use(bufRcv, a, b);
return 0;
}
// 需要參數(shù)的異步調(diào)用需要將參數(shù)透傳到后半段的回調(diào)函數(shù)中
void get_data_v5(int a, int b)
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv_async(addr, bufCmd, bufRcv, callback); } // end of get_data_v5
// definition of call back function
int callback(char* bufRcv, int a, int b) {
use(bufRcv, a, b);
return 0;
}
6 使用一個closure對象打包過程中的參數(shù)
view plaincopy to clipboardprint?
// 為了統(tǒng)一回調(diào)函數(shù)的形式并且縮短回調(diào)的參數(shù)列表,將這種需要透傳的參數(shù)只有一個
// 統(tǒng)一的數(shù)據(jù)結(jié)構(gòu)打包
void get_data_v6(int a, int b)
{
char bufCmd[]="cmd=1001&uin=123456¶m=abc";
char bufRcv[4096];
send_and_recv(addr, bufCmd, bufRcv, callback); } // end of get_data_v6
// definition of call back function