• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            posts - 195,  comments - 30,  trackbacks - 0
            原地址:http://www.cnblogs.com/xiaoli0414/archive/2007/11/27/974534.html

            最經(jīng)公司工作需要調(diào)用一個外部的webservice,同時要將傳出的數(shù)據(jù)進行保存,以自己以前的習(xí)慣,就打算逐步操作,失敗啊,完全沒考慮過用戶體驗效果,在同事指點下,意識到使用異步調(diào)用的好處,隨便將自己找的一些資料留以保存,以戒后誤!
                    我們要明確,
            為什么要進行異步回調(diào)?眾所周知,普通方法運行,是單線程的,如果中途有大型操作(如:讀取大文件,大批量操作數(shù)據(jù)庫,網(wǎng)絡(luò)傳輸?shù)龋紩?dǎo)致方法阻塞,表現(xiàn)在界面上就是,程序卡或者死掉,界面元素不動了,不響應(yīng)了。異步方法很好的解決了這些問題,異步執(zhí)行某個方法,程序立即開辟一個新線程去運行你的方法,主線程包括界面就不會死掉了。異步如何開始,好理解,現(xiàn)在我們討論的是如何結(jié)束這個異步出來的新線程。
                   首先,異步出來的新線程,必須回收,不回收是浪費資源的可恥行為,.NET也是不允許的,所以你別想鉆空子,俗話說,請神容易送神難,就是這個道理。下面你可以很容易想到,回收分為2種情況:主動回收和被動回收(當(dāng)然,這是我自己的理解,微軟可不是這么說的),主動回收就是,你去監(jiān)視那個線程,并且等待,當(dāng)異步方法完成了,就把異步線程回收,焦點回歸主線程,實際上就是上篇文章《C#異步初步》的那種情況,BeginInvoke之后又EndInvoke,如果在EndInvoke的時候,該異步線程沒有完成操作,那么整個程序,包括主線程,又在阻塞了,又會出現(xiàn)界面“死”的情況。要想解決這個問題,就使用“被動回收”方式,其中一個重要的辦法就是“異步回調(diào)”。
                     
            核心有二:
                
            A、   用回調(diào)函數(shù)(本例中為CallBackMethod
            ),異步結(jié)束后,自動調(diào)用此回調(diào)函數(shù)。    
                
            B、   
            而不在主線程中手工等待異步結(jié)束,如上兩例中在主線程中調(diào)用EndInvoke。此種方法,是在回調(diào)函數(shù)中調(diào)用EndInvoke的。
                 異步回調(diào)的大概流程是這樣的:首先啟動異步,啟動參數(shù)加上異步結(jié)束時執(zhí)行的方法,然后這個異步線程就不用管了,最后當(dāng)這個異步線程自己完成工作了,就自動執(zhí)行啟動參數(shù)里的那個方法,這樣確實很省心,可是代碼寫起來,就很復(fù)雜了。
                下面是搜藏的代碼:

             //
            首先準備好,要進行異步的方法(能異步的,最好不多線程)
            private string MethodName(int Num, out int Num2)
            {
                           Num2 = Num;
                           return "HelloWorld";
            }

            //程序終點
            //異步完成時,執(zhí)行的方法(回調(diào)方法),此方法只能有IAsyncResult一個參數(shù),但是該參數(shù)幾乎萬能,可以傳遞object
            private void CallBackMethod(IAsyncResult ar)
            {
                           //從異步狀態(tài)ar.AsyncState中,獲取委托對象
                           DelegateName dn = (DelegateName)ar.AsyncState;
                           //輸出參數(shù)
                           int i;

                           //一定要EndInvoke,否則你的下場很慘
                           string r = dn.EndInvoke(out i, ar);
                           MessageBox.Show("異步完成嘍!i的值是" i.ToString() ",r的值是" r);
            }

            //定義與方法同簽名的委托
            private delegate string DelegateName(int Num, out int Num2);

            //程序入口
            private void Run()
            {
                           //實例化委托并初賦值
                           DelegateName dn = new DelegateName(MethodName);
                           //輸出參數(shù)
                           int i;
                           //實例化回調(diào)方法
                           //把AsyncCallback看成Delegate你就懂了,實際上AsyncCallback是一種特殊的Delegate,就像Event似的
                           AsyncCallback acb = new AsyncCallback(CallBackMethod);
                           //異步開始
                           //如果參數(shù)acb換成null則表示沒有回調(diào)方法
                           //最后一個參數(shù)dn的地方,可以換成任意對象,該對象可以被回調(diào)方法從參數(shù)中獲取出來,寫成null也可以。參數(shù)dn相當(dāng)于該線程的ID,如果有多個異步線程,可以都是null,但是絕對不能一樣,不能是同一個object,否則異常
                           IAsyncResult iar = dn.BeginInvoke(1, out i, acb, dn);
                           //去做別的事
                           //…………
            }

            //最后的結(jié)果應(yīng)該是:i=1,r="HelloWorld"


            另外,如果可以,定義委托的時候可以選擇不用過多的修飾:

                    /// <summary>
                    /// 定義委托
                    /// </summary>
                    /// <returns></returns>
                    public delegate bool Asyncdelegate();

                    /// <summary>
                    /// Callback method must have the same signature as the
                    /// AsyncCallback delegate
                    /// </summary>
                    /// <param name="ar"></param>
                    private void CallbackMethod(IAsyncResult ar)
                    {
                        // Retrieve the delegate.
                        Asyncdelegate dlgt = (Asyncdelegate)ar.AsyncState;

                        // Call EndInvoke to retrieve the results.
                        dlgt.EndInvoke(ar);
                    }

              其他方法中調(diào)用:
                    //異步執(zhí)行      
                    //指定委托方法      
                    Asyncdelegate isgt = new Asyncdelegate(icpInfo.Insert);
                    IAsyncResult ar = isgt.BeginInvoke(new AsyncCallback(CallbackMethod), isgt);
                    
            posted on 2011-03-14 11:13 luis 閱讀(366) 評論(0)  編輯 收藏 引用

            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            <2011年11月>
            303112345
            6789101112
            13141516171819
            20212223242526
            27282930123
            45678910

            常用鏈接

            留言簿(3)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            友情鏈接

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            久久丫精品国产亚洲av| 久久超碰97人人做人人爱| 久久久久久狠狠丁香| 亚洲综合精品香蕉久久网97| 国产亚洲精午夜久久久久久| 久久精品国产亚洲一区二区三区| 日韩久久无码免费毛片软件| 思思久久好好热精品国产| 亚洲AV无码久久精品成人| 婷婷综合久久狠狠色99h| 性做久久久久久久久浪潮| 少妇内射兰兰久久| 久久人人爽人人爽人人片AV麻豆 | 久久精品无码一区二区三区| 国产精品欧美亚洲韩国日本久久| 久久亚洲国产精品成人AV秋霞| av无码久久久久久不卡网站| 青青草原综合久久大伊人导航| 狠狠88综合久久久久综合网| 人妻丰满?V无码久久不卡| 久久精品国产清高在天天线| 亚洲人成网站999久久久综合| 777久久精品一区二区三区无码| 亚洲国产美女精品久久久久∴| 久久99精品国产麻豆不卡| 久久精品一区二区三区不卡| 一本一道久久综合狠狠老| 中文字幕无码久久久| 色婷婷噜噜久久国产精品12p| 精品国际久久久久999波多野| 久久强奷乱码老熟女网站| 武侠古典久久婷婷狼人伊人| 国产精品免费久久| 久久精品国产一区二区电影| 久久综合综合久久97色| 久久国产精品国产自线拍免费| 精品乱码久久久久久久| 精品少妇人妻av无码久久| 国产精品毛片久久久久久久| 久久精品无码专区免费青青| 亚洲国产精品一区二区三区久久 |