<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    JAVA & XML & JAVASCRIPT & AJAX & CSS

    Web 2.0 技術(shù)儲備............

      BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
      77 隨筆 :: 17 文章 :: 116 評論 :: 0 Trackbacks

    八、JavaScript面向?qū)ο蟮闹С?br />~~~~~~~~~~~~~~~~~~
    (續(xù))

    ?4). 需要用戶維護的另一個屬性:constructor
    ?------
    ?回顧前面的內(nèi)容,我們提到過:
    ?? - (如果正常地實現(xiàn)繼承模型,)對象實例的constructor屬性指向構(gòu)造器
    ?? - obj.constructor.prototype指向該對象的原型
    ?? - 通過Object.constructor屬性,可以檢測obj2與obj1是否是相同類型的實例

    ? 與原型鏈要通過用戶代碼來維護prototype屬性一樣,實例的構(gòu)造器屬性constructor
    也需要用戶代碼維護。

    ? 對于JavaScript的內(nèi)置對象來說,constructor屬性指向內(nèi)置的構(gòu)造器函數(shù)。如:
    //---------------------------------------------------------
    // 內(nèi)置對象實例的constructor屬性
    //---------------------------------------------------------
    var _object_types = {
    ? 'function'? : Function,
    ? 'boolean'?? : Boolean,
    ? 'regexp'??? : RegExp,
    // 'math'???? : Math,
    // 'debug'??? : Debug,
    // 'image'??? : Image;
    // 'undef'??? : undefined,
    // 'dom'????? : undefined,
    // 'activex'? : undefined,
    ? 'vbarray'?? : VBArray,
    ? 'array'???? : Array,
    ? 'string'??? : String,
    ? 'date'????? : Date,
    ? 'error'???? : Error,
    ? 'enumerator': Enumerator,
    ? 'number'??? : Number,
    ? 'object'??? : Object
    }

    function objectTypes(obj) {
    ? if (typeof obj !== 'object') return typeof obj;
    ? if (obj === null) return 'null';

    ? for (var i in _object_types) {
    ??? if (obj.constructor===_object_types[i]) return i;
    ? }
    ? return 'unknow';
    }

    // 測試數(shù)據(jù)和相關(guān)代碼
    function MyObject() {
    }
    function MyObject2() {
    }
    MyObject2.prototype = new MyObject();

    window.execScript(''+
    'Function CreateVBArray()' +
    '? Dim a(2, 2)' +
    '? CreateVBArray = a' +
    'End Function', 'VBScript');

    document.writeln('<div id=dom style="display:none">dom<', '/div>');

    // 測試代碼
    var ax = new ActiveXObject("Microsoft.XMLHTTP");
    var dom = document.getElementById('dom');
    var vba = new VBArray(CreateVBArray());
    var obj = new MyObject();
    var obj2 = new MyObject2();

    document.writeln(objectTypes(vba), '<br>');
    document.writeln(objectTypes(ax), '<br>');
    document.writeln(objectTypes(obj), '<br>');
    document.writeln(objectTypes(obj2), '<br>');
    document.writeln(objectTypes(dom), '<br>');

    在這個例子中,我們發(fā)現(xiàn)constructor屬性被實現(xiàn)得并不完整。對于DOM對象、ActiveX對象
    來說這個屬性都沒有正確的返回。

    確切的說,DOM(包括Image)對象與ActiveX對象都不是標(biāo)準(zhǔn)JavaScript的對象體系中的,
    因此它們也可能會具有自己的constructor屬性,并有著與JavaScript不同的解釋。因此,
    JavaScript中不維護它們的constructor屬性,是具有一定的合理性的。

    另外的一些單體對象(而非構(gòu)造器),也不具有constructor屬性,例如“Math”和“Debug”、
    “Global”和“RegExp對象”。他們是JavaScript內(nèi)部構(gòu)造的,不應(yīng)該公開構(gòu)造的細(xì)節(jié)。

    我們也發(fā)現(xiàn)實例obj的constructor指向function MyObject()。這說明JavaScript維護了對
    象的constructor屬性。——這與一些人想象的不一樣。

    然而再接下來,我們發(fā)現(xiàn)MyObject2()的實例obj2的constructor仍然指向function MyObject()。
    盡管這很說不通,然而現(xiàn)實的確如此。——這到底是為什么呢?

    事實上,僅下面的代碼:
    --------
    function MyObject2() {
    }

    obj2 = new MyObject2();
    document.writeln(MyObject2.prototype.constructor === MyObject2);
    --------
    構(gòu)造的obj2.constructor將正確的指向function MyObject2()。事實上,我們也會注意到這
    種情況下,MyObject2的原型屬性的constructor也正確的指向該函數(shù)。然而,由于JavaScript
    要求指定prototype對象來構(gòu)造原型鏈:
    --------
    function MyObject2() {
    }
    MyObject2.prototype = new MyObject();

    obj2 = new MyObject2();
    --------
    這時,再訪問obj2,將會得到新的原型(也就是MyObject2.prototype)的constructor屬性。
    因此,一切很明了:原型的屬性影響到構(gòu)造過程對對象的constructor的初始設(shè)定。

    作為一種補充的解決問題的手段,JavaScript開發(fā)規(guī)范中說“need to remember to reset
    the constructor property',要求用戶自行設(shè)定該屬性。

    所以你會看到更規(guī)范的JavaScript代碼要求這樣書寫:
    //---------------------------------------------------------
    // 維護constructor屬性的規(guī)范代碼
    //---------------------------------------------------------
    function MyObject2() {
    }
    MyObject2.prototype = new MyObject();
    MyObject2.prototype.constructor = MyObject2;

    obj2 = new MyObject2();


    更外一種解決問題的方法,是在function MyObject()中去重置該值。當(dāng)然,這樣會使
    得執(zhí)行效率稍低一點點:
    //---------------------------------------------------------
    // 維護constructor屬性的第二種方式
    //---------------------------------------------------------
    function MyObject2() {
    ? this.constructor = arguments.callee;
    ? // or, this.constructor = MyObject2;

    ? // ...
    }
    MyObject2.prototype = new MyObject();

    obj2 = new MyObject2();


    ?5). 析構(gòu)問題
    ?------
    ?JavaScript中沒有析構(gòu)函數(shù),但卻有“對象析構(gòu)”的問題。也就是說,盡管我們不
    知道一個對象什么時候會被析構(gòu),也不能截獲它的析構(gòu)過程并處理一些事務(wù)。然而,
    在一些不多見的時候,我們會遇到“要求一個對象立即析構(gòu)”的問題。

    問題大多數(shù)的時候出現(xiàn)在對ActiveX Object的處理上。因為我們可能在JavaScript
    里創(chuàng)建了一個ActiveX Object,在做完一些處理之后,我們又需要再創(chuàng)建一個。而
    如果原來的對象供應(yīng)者(Server)不允許創(chuàng)建多個實例,那么我們就需要在JavaScript
    中確保先前的實例是已經(jīng)被釋放過了。接下來,即使Server允許創(chuàng)建多個實例,而
    在多個實例間允許共享數(shù)據(jù)(例如OS的授權(quán),或者資源、文件的鎖),那么我們在新
    實例中的操作就可能會出問題。

    可能還是有人不明白我們在說什么,那么我就舉一個例子:如果創(chuàng)建一個Excel對象,
    打開文件A,然后我們save它,然后關(guān)閉這個實例。然后我們再創(chuàng)建Excel對象并打開
    同一文件。——注意這時JavaScript可能還沒有來得及析構(gòu)前一個對象。——這時我們
    再想Save這個文件,就發(fā)現(xiàn)失敗了。下面的代碼示例這種情況:
    //---------------------------------------------------------
    // JavaScript中的析構(gòu)問題(ActiveX Object示例)
    //---------------------------------------------------------
    <script>
    var strSaveLocation = 'file:///E:/1.xls'

    function createXLS() {
    ? var excel = new ActiveXObject("Excel.Application");
    ? var wk = excel.Workbooks.Add();
    ? wk.SaveAs(strSaveLocation);
    ? wk.Saved = true;

    ? excel.Quit();
    }

    function writeXLS() {
    ? var excel = new ActiveXObject("Excel.Application");
    ? var wk = excel.Workbooks.Open(strSaveLocation);
    ? var sheet = wk.Worksheets(1);
    ? sheet.Cells(1, 1).Value = '測試字符串';
    ? wk.SaveAs(strSaveLocation);
    ? wk.Saved = true;

    ? excel.Quit();
    }
    </script>

    <body>
    ? <button onclick="createXLS()">創(chuàng)建</button>
    ? <button onclick="writeXLS()">重寫</button>
    </body>

    在這個例子中,在本地文件操作時并不會出現(xiàn)異常。——最多只是有一些內(nèi)存垃
    圾而已。然而,如果strSaveLocation是一個遠程的URL,這時本地將會保存一個
    文件存取權(quán)限的憑證,而且同時只能一個(遠程的)實例來開啟該excel文檔并存
    儲。于是如果反復(fù)點擊"重寫"按鈕,就會出現(xiàn)異常。

    ——注意,這是在SPS中操作共享文件時的一個實例的簡化代碼。因此,它并非
    “學(xué)術(shù)的”無聊討論,而且工程中的實際問題。

    解決這個問題的方法很復(fù)雜。它涉及到兩個問題:
    ? - 本地憑證的釋放
    ? - ActiveX Object實例的釋放

    下面我們先從JavaScript中對象的“失效”問題說起。簡單的說:
    ? - 一個對象在其生存的上下文環(huán)境之外,即會失效。
    ? - 一個全局的對象在沒有被執(zhí)用(引用)的情況下,即會失效。

    例如:
    //---------------------------------------------------------
    // JavaScript對象何時失效
    //---------------------------------------------------------
    function testObject() {
    ? var _obj1 = new Object();
    }

    function testObject2() {
    ? var _obj2 = new Object();
    ? return _obj2;
    }

    // 示例1
    testObject();

    // 示例2
    testObject2()

    // 示例3
    var obj3 = testObject2();
    obj3 = null;

    // 示例4
    var obj4 = testObject2();
    var arr = [obj4];
    obj3 = null;
    arr = [];

    在這四個示例中:
    ? - “示例1”在函數(shù)testObject()中構(gòu)造了_obj1,但是在函數(shù)退出時,
    ??? 它就已經(jīng)離開了函數(shù)的上下文環(huán)境,因此_obj1失效了;
    ? - “示例2”中,testObject2()中也構(gòu)造了一個對象_obj2并傳出,因
    ??? 此對象有了“函數(shù)外”的上下文環(huán)境(和生存周期),然而由于函數(shù)
    ??? 的返回值沒有被其它變量“持有”,因此_obj2也立即失效了;
    ? - “示例3”中,testObject2()構(gòu)造的_obj2被外部的變量obj3持用了,
    ??? 這時,直到“obj3=null”這行代碼生效時,_obj2才會因為引用關(guān)系
    ??? 消失而失效。
    ? - 與示例3相同的原因,“示例4”中的_obj2會在“arr=[]”這行代碼
    ??? 之后才會失效。

    但是,對象的“失效”并不等會“釋放”。在JavaScript運行環(huán)境的內(nèi)部,沒
    有任何方式來確切地告訴用戶“對象什么時候會釋放”。這依賴于JavaScript
    的內(nèi)存回收機制。——這種策略與.NET中的回收機制是類同的。

    在前面的Excel操作示例代碼中,對象的所有者,也就是"EXCEL.EXE"這個進程
    只能在“ActiveX Object實例的釋放”之后才會發(fā)生。而文件的鎖,以及操作
    系統(tǒng)的權(quán)限憑證是與進程相關(guān)的。因此如果對象僅是“失效”而不是“釋放”,
    那么其它進程處理文件和引用操作系統(tǒng)的權(quán)限憑據(jù)時就會出問題。

    ——有些人說這是JavaScript或者COM機制的BUG。其實不是,這是OS、IE
    和JavaScript之間的一種復(fù)雜關(guān)系所導(dǎo)致的,而非獨立的問題。

    Microsoft公開了解決這種問題的策略:主動調(diào)用內(nèi)存回收過程。

    在(微軟的)JScript中提供了一個CollectGarbage()過程(通常簡稱GC過程),
    GC過程用于清理當(dāng)前IE中的“失效的對象失例”,也就是調(diào)用對象的析構(gòu)過程。

    在上例中調(diào)用GC過程的代碼是:
    //---------------------------------------------------------
    // 處理ActiveX Object時,GC過程的標(biāo)準(zhǔn)調(diào)用方式
    //---------------------------------------------------------
    function writeXLS() {
    ? //(略...)

    ? excel.Quit();
    ? excel = null;
    ? setTimeout(CollectGarbage, 1);
    }

    第一行代碼調(diào)用excel.Quit()方法來使得excel進程中止并退出,這時由于JavaScript
    環(huán)境執(zhí)有excel對象實例,因此excel進程并不實際中止。

    第二行代碼使excel為null,以清除對象引用,從而使對象“失效”。然而由于
    對象仍舊在函數(shù)上下文環(huán)境中,因此如果直接調(diào)用GC過程,對象仍然不會被清理。

    第三行代碼使用setTimeout()來調(diào)用CollectGarbage函數(shù),時間間隔設(shè)為'1',只
    是使得GC過程發(fā)生在writeXLS()函數(shù)執(zhí)行完之后。這樣excel對象就滿足了“能被
    GC清理”的兩個條件:沒有引用和離開上下文環(huán)境。

    GC過程的使用,在使用了ActiveX Object的JS環(huán)境中很有效。一些潛在的ActiveX
    Object包括XML、VML、OWC(Office Web Componet)、flash,甚至包括在JS中的VBArray。
    從這一點來看,ajax架構(gòu)由于采用了XMLHTTP,并且同時要滿足“不切換頁面”的
    特性,因此在適當(dāng)?shù)臅r候主動調(diào)用GC過程,會得到更好的效率用UI體驗。

    事實上,即使使用GC過程,前面提到的excel問題仍然不會被完全解決。因為IE還
    緩存了權(quán)限憑據(jù)。使頁的權(quán)限憑據(jù)被更新的唯一方法,只能是“切換到新的頁面”,
    因此事實上在前面提到的那個SPS項目中,我采用的方法并不是GC,而是下面這一
    段代碼:
    //---------------------------------------------------------
    // 處理ActiveX Object時采用的頁面切換代碼
    //---------------------------------------------------------
    function writeXLS() {
    ? //(略...)

    ? excel.Quit();
    ? excel = null;
    ?
    ? // 下面代碼用于解決IE call Excel的一個BUG, MSDN中提供的方法:
    ? //?? setTimeout(CollectGarbage, 1);
    ? // 由于不能清除(或同步)網(wǎng)頁的受信任狀態(tài), 所以將導(dǎo)致SaveAs()等方法在
    ? // 下次調(diào)用時無效.
    ? location.reload();
    }

    最后之最后,關(guān)于GC的一個補充說明:在IE窗體被最小化時,IE將會主動調(diào)用一次
    CollectGarbage()函數(shù)。這使得IE窗口在最小化之后,內(nèi)存占用會有明顯改善。

    posted on 2006-03-20 09:37 Web 2.0 技術(shù)資源 閱讀(1282) 評論(0)  編輯  收藏 所屬分類: Javascript
    主站蜘蛛池模板: 国产成人aaa在线视频免费观看| 99在线观看免费视频| 亚洲人成电影网站久久| 亚洲成人免费在线| 亚洲av日韩av高潮潮喷无码| 国产免费阿v精品视频网址| 亚洲一级片内射网站在线观看| 国产尤物在线视精品在亚洲| 日本a级片免费看| 国产亚洲精品91| 亚洲av午夜成人片精品电影| 一级毛片免费观看不收费| 久久久久亚洲爆乳少妇无| 久久久久免费视频| 亚洲成年人在线观看| 亚洲免费在线视频播放| 亚洲男人的天堂久久精品| 最近最好的中文字幕2019免费| 成人区精品一区二区不卡亚洲| 成年在线观看免费人视频草莓| 亚洲AV无码一区二区三区性色| 免费国产在线观看不卡| 国产精品视频全国免费观看| 国产亚洲精AA在线观看SEE| 91精品啪在线观看国产线免费| 亚洲婷婷在线视频| 日韩一级视频免费观看| 一区二区三区免费电影| 亚洲爱情岛论坛永久| 黄页网站在线观看免费高清| MM1313亚洲国产精品| 久久精品国产亚洲网站| 国产无人区码卡二卡三卡免费| 国产亚洲日韩在线a不卡| 亚洲人成亚洲人成在线观看| 97免费人妻在线视频| 国产午夜亚洲精品不卡免下载| 亚洲日本va中文字幕久久| 国产大片线上免费观看 | 亚洲精品国产手机| 免费无码看av的网站|