• <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>

            雖不能至,心向往之

            —— 巴人也,操C++口音,混跡于京師,勉強(qiáng)度日……《史記·corelito列傳》
            posts - 8, comments - 15, trackbacks - 0, articles - 0

            使用V8——Google Chrome 的 JavaScript 引擎(5)

            Posted on 2008-10-23 10:45 sufan 閱讀(3440) 評(píng)論(7)  編輯 收藏 引用 所屬分類: 翻譯
            使用訪問器訪問 C++ 對(duì)象

            為我們的類設(shè)置環(huán)境
            怎樣使用C++把一個(gè)類映射成為 JavaScript?首先來看看如下的例子: 
            //Sample class mapped to v8
            class Point 

            public
                
            //constructor
                Point(int x, int y):x_(x),y_(y){}

                
            //internal class functions
                
            //just increment x_
                void Function_A(){++x_;    }

                
            //increment x_ by the amount
                void Function_B(int vlr){x_+=vlr;}

                
            //variables
                int x_; 
            };

             
            為了保證整個(gè)類映射后完全基于 JavaScript,我們對(duì)類的成員函數(shù)和變量都做映射。第一步,在上下文環(huán)境(context)中映射一個(gè)類模板:

            Handle<FunctionTemplate> point_templ = FunctionTemplate::New();
            point_templ
            ->SetClassName(String::New("Point"));


            從字面上似乎看出:我們只是創(chuàng)建了一個(gè)“函數(shù)(function)”模板,但實(shí)際上它能夠被看作一個(gè)類。它的名字是“Point”,我們將會(huì)在后面用到。
            然后,通過訪問原型(prototype) 類模板,我們能為這個(gè)類添加內(nèi)建的方法:

            Handle<ObjectTemplate> point_proto = point_templ->PrototypeTemplate();
            point_proto
            ->Set("method_a", FunctionTemplate::New(PointMethod_A));
            point_proto
            ->Set("method_b", FunctionTemplate::New(PointMethod_B));


            類能通過上面的代碼“知道”它有兩個(gè)方法。但這仍然是類的原型,我們必須要實(shí)例化這個(gè)類之后,才能使用這兩個(gè)方法。

            Handle<ObjectTemplate> point_inst = point_templ->InstanceTemplate();
            point_inst
            ->SetInternalFieldCount(1);


            SetInternalFieldCount 函數(shù)為類指針申請(qǐng)空間(在后面會(huì)再一次提到)。一旦有了類的實(shí)例,我們就能為類中的變量添加訪問器了:

            point_inst->SetAccessor(String::New("x"), GetPointX, SetPointX);

            至此,舞臺(tái)算是搭好了,主角該上場(chǎng)了:
            Point* p = new Point(00);

            新類已經(jīng)建好了,但只能在C++中訪問。要對(duì)其進(jìn)行訪問,我們需要:
            Handle<Function> point_ctor = point_templ->GetFunction();
            Local
            <Object> obj = point_ctor->NewInstance();
            obj
            ->SetInternalField(0, External::New(p));


            GetFunction 返回 point 的構(gòu)造器,有了它,我們可以使用 NewInstance 函數(shù)來創(chuàng)建一個(gè)新的實(shí)例。然后,使用類指針來設(shè)置內(nèi)部域(這個(gè)域的空間我們?cè)谇懊嬉呀?jīng)使用 SetInternalFieldCount 函數(shù)申請(qǐng)好了)。這樣一來,JavaScript 就能通過指針訪問該對(duì)象。
            有一點(diǎn)我們漏掉了,我們想從 JavaScript 中訪問它,但只有類模板和實(shí)例,沒有名字:

            context->Global()->Set(String::New("point"), obj);


            最后一步將“point”這個(gè)名字和 obj 實(shí)例聯(lián)系起來。到這里,我們僅僅是完成了在腳本中使用“point”名字來創(chuàng)建 Point 類的過程。

            在 JavaScript 中訪問類方法
            這里講的并不是我們?cè)鯓釉?Point 類中訪問 Function_A……

            先來看看回調(diào)函數(shù) PointMethod_A:

            Handle<Value> PointMethod_A(const Arguments& args)
            {
                Local
            <Object> self = args.Holder();
                Local
            <External> wrap = Local<External>::Cast(self->GetInternalField(0));
                
            void* ptr = wrap->Value();
                static_cast
            <Point*>(ptr)->Function_A();
                
            return Integer::New(static_cast<Point*>(ptr)->x_);
            }


            和普通的訪問器類似,我們需對(duì)參數(shù)進(jìn)行操作。而且,要能訪問我們自己的類,還必須得從內(nèi)部域中獲取相關(guān)的類指針。在將內(nèi)部域映射到“wrap”之后,就可以通過獲取它的“值(Value)”來得到類指針,然后再進(jìn)行類型轉(zhuǎn)換。得到“point”類指針之后,可以通過它來調(diào)用 method_a ,method_a 來調(diào)用回調(diào)函數(shù) PointMethod_A。

            示例 v8_embedded_demo_with_object.zip  (Visual C++ Express 2008 )完整的包含了上述各個(gè)步驟。

            最后,我非常希望這篇文章能對(duì)你有所幫助。如果感覺文中有錯(cuò)誤或者不清楚的地方,歡迎和我進(jìn)行討論~~

            Feedback

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2008-10-25 16:56 by 金山詞霸2008
            真是一個(gè)好方法。

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2009-04-27 15:44 by ccqcn
            我怎么按照你這個(gè),運(yùn)行,沒有任何效果啊。

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2009-04-27 15:46 by ccqcn
            請(qǐng)問,在js文件里,直接 var p=new Point() 構(gòu)造C++里面的對(duì)象,怎么辦呢??

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2009-05-24 17:13 by pro.z
            這篇文章對(duì)我很有幫助,首先感謝博主,我還有個(gè)問題想請(qǐng)問下,就是關(guān)于document.write這一類的怎么處理,V8自身好像沒定義,我用你的方法映射的話,貌似返回值有問題,麻煩求助下您!

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2009-06-24 17:57 by Heaven
            如果我想實(shí)現(xiàn) A.B.function();怎么做呢?請(qǐng)教下?

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2011-09-09 09:21 by 有趣之極
            好東西是好東西,但最新版本用vs2010編譯不出來啊。。。

            # re: 使用V8——Google Chrome 的 JavaScript 引擎(5)  回復(fù)  更多評(píng)論   

            2011-11-03 13:19 by 門外漢
            樓主翻譯辛苦了
            亚洲嫩草影院久久精品| 亚洲国产精品婷婷久久| 久久99国产一区二区三区| 18岁日韩内射颜射午夜久久成人| 国产成人精品久久亚洲高清不卡| 一本色道久久HEZYO无码| 欧美亚洲另类久久综合婷婷| 久久综合久久综合久久综合| 97精品久久天干天天天按摩| 亚洲乱码精品久久久久..| 日韩美女18网站久久精品| 久久亚洲色一区二区三区| 国产精品免费久久久久久久久| 国产精品美女久久久久网| 精品久久久久久中文字幕| 日本福利片国产午夜久久| 91精品国产乱码久久久久久| 2022年国产精品久久久久| 久久久久国产精品| 久久久久久综合一区中文字幕| 久久精品九九亚洲精品天堂| 日本久久久久久久久久| 四虎国产精品免费久久| 亚洲国产天堂久久综合| 久久无码AV中文出轨人妻| 尹人香蕉久久99天天拍| 国产精品久久久久久久app| 色狠狠久久综合网| 亚洲中文字幕无码久久2020| 久久久久99精品成人片直播| 久久综合88熟人妻| AV无码久久久久不卡蜜桃| 国产麻豆精品久久一二三| 日韩精品久久久久久| 无码人妻久久一区二区三区蜜桃| 色天使久久综合网天天| 亚洲国产精品久久久天堂| 久久国产精品国产自线拍免费| 93精91精品国产综合久久香蕉 | 三上悠亚久久精品| 国产成年无码久久久久毛片|