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

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

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

    隨筆-2  評論-1  文章-3  trackbacks-0
    本篇旨在幫助準(zhǔn)備學(xué)習(xí) Java 以及剛接觸 Java 的朋友認(rèn)識、掌握和使用 static this super final 這幾個關(guān)鍵字的使用。 Java 博大精深,我也是一位正在學(xué)習(xí)和使用 Java 的愛好者,文中難免有不妥之處,歡迎指正。
    一、 static
    ????
    請先看下面這段程序:
    1. ??
    2. ???public?class?Hello{
    3. ?????public?static?void?main(String[]?args){?//(1)
    4. ???????System.out.println("Hello,world!");???//(2)
    5. ?????}
    6. ???}


    看過這段程序,對于大多數(shù)學(xué)過 Java? 的從來說,都不陌生。即使沒有學(xué)過 Java ,而學(xué)過其它的高 級語言,例如 C ,那你也應(yīng)該能看懂這段代碼的意思。它只是簡單的輸出 “Hello,world” ,一點(diǎn)別的用處都沒有,然而,它卻展示了 static 關(guān)鍵 字的主要用法。 ?
    1 處,我們定義了一個靜態(tài)的方法名為 main ,這就意味著告訴 Java 編譯器,我這個方法不需要創(chuàng)建一個此類的對象即可使用。你還得你是怎么運(yùn)行這個程序嗎?一般,我們都是在命令行下,打入如下的命令 ( 加下劃線為手動輸入 )

    javac?Hello.java
    java?Hello

    Hello,world!
    就是你運(yùn)行的過程,第一行用來編譯 Hello.java 這個文件,執(zhí)行完后,如果你查看當(dāng)前,會發(fā)現(xiàn)多了一個 Hello.class 文件,那就是第一行產(chǎn) 生的 Java 二進(jìn)制字節(jié)碼。第二行就是執(zhí)行一個 Java 程序的最普遍做法。執(zhí)行結(jié)果如你所料。在 2 中,你可能會想,為什么要這樣才能輸出。好,我們來分解 一下這條語句。(如果沒有安裝 Java 文檔,請到 Sun 的官方網(wǎng)站瀏覽 J2SE?API )首先, System 是位于 java.lang 包中的一 個核心類,如果你查看它的定義,你會發(fā)現(xiàn)有這樣一行: public?static?final? PrintStream?out; 接著在進(jìn)一步,點(diǎn)擊 PrintStream 這個超鏈接,在 METHOD 頁面,你會看到大量定義的方法,查找 println ,會有這樣一行:
    public?void?println(String?x)
    。好了,現(xiàn)在你應(yīng)該明白為什么我們要那樣調(diào)用了, out System 的一個靜態(tài)變量,所以可以直接使用,而 out 所屬的類有一個 println 方法。

    靜態(tài)方法 ?通常,在一個類中定義一個方法為static,那就是說,無需本類的對象即可調(diào)用此方法。如下所示:

    1. ?class?Simple{
    2. ????static?void?go(){
    3. ??????System.out.println("Go...");
    4. ????}
    5. ?}
    6. ?public?class?Cal{
    7. ???public?static?void?main(String[]?args){
    8. ?????Simple.go();
    9. ???}
    10. }
    11. 調(diào)用一個靜態(tài)方法就是 類名 . 方法名 ”, 靜態(tài)方法的使用很簡單如上所示。一般來說,靜態(tài)方法常常為應(yīng)用程序中的其它類提供一些實(shí)用工具所用,在 Java 的類庫中大量的靜態(tài)方法正是出于此目的而定義的。
      靜態(tài)變量
      靜態(tài)變量與靜態(tài)方法類似。所有此類實(shí)例共享此靜態(tài)變量,也就是說在類裝載時,只分配一塊存儲空間,所有此類的對象都可以操控此塊存儲空間,當(dāng)然對于 final 則另當(dāng)別論了。看下面這段代碼:

      1. class ?Value{
      2. ??static?int?c=0;
      3. ??static?void?inc(){
      4. ????c++;
      5. ??}
      6. }
      7. class ?Count{
      8. ??public?static?void?prt(String?s){
      9. ????System.out.println(s);
      10. ??}
      11. ??public?static?void?main(String[]?args){
      12. ????Value?v1,v2;
      13. ????v1=new?Value();
      14. ????v2=new?Value();
      15. ????prt("v1.c="+v1.c+"??v2.c="+v2.c);
      16. ????v1.inc();
      17. ????prt("v1.c="+v1.c+"??v2.c="+v2.c);??
      18. ??}
      19. }

    結(jié)果如下:
    v1.c=0??v2.c=0
    v1.c=1??v2.c=1
    由此可以證明它們共享一塊存儲區(qū)。 static 變量有點(diǎn)類似于 C 中的全局變量的概念。值得探討的是靜態(tài)變量的初始化問題。我們修改上面的程序:

    1. class ?Value{
    2. ??static?int?c=0;
    3. ??Value(){
    4. ????c=15;
    5. ??}
    6. ??Value(int?i){
    7. ????c=i;
    8. ??}
    9. ??static?void?inc(){
    10. ????c++;
    11. ??}
    12. }
    13. class ?Count{
    14. ??public?static?void?prt(String?s){
    1. ????System.out.println(s);
    2. ??}
    3. ????Value?v=new?Value(10);
    4. ????static?Value?v1,v2;
    5. ????static{
    6. ??????prt("v1.c="+v1.c+"??v2.c="+v2.c);
    7. ??????v1=new?Value(27);
    8. ??????prt("v1.c="+v1.c+"??v2.c="+v2.c);
    9. ??????v2=new?Value(15);
    10. ??????prt("v1.c="+v1.c+"??v2.c="+v2.c);
    11. ????}
    12. ??public?static?void?main(String[]?args){
    13. ????Count?ct=new?Count();
    14. ????prt("ct.c="+ct.v.c);
    15. ????prt("v1.c="+v1.c+"??v2.c="+v2.c);
    16. ????v1.inc();
    17. ????prt("v1.c="+v1.c+"??v2.c="+v2.c);
    18. ????prt("ct.c="+ct.v.c);
    19. ??}
    20. }


    運(yùn)行結(jié)果如下:
    v1.c=0??v2.c=0
    v1.c=27??v2.c=27
    v1.c=15??v2.c=15
    ct.c=10
    v1.c=10??v2.c=10
    v1.c=11??v2.c=11
    ct.c=11
    個程序展示了靜態(tài)初始化的各種特性。如果你初次接觸 Java ,結(jié)果可能令你吃驚。可能會對 static 后加大括號感到困惑。首先要告訴你的是, static 定義的變量會優(yōu)先于任何其它非 static 變量,不論其出現(xiàn)的順序如何。正如在程序中所表現(xiàn)的,雖然 v 出現(xiàn)在 v1 v2 的前面,但是結(jié)果卻是 v1 v2 的初始化在 v 的前面。在 static{ 后面跟著一段代碼,這是用來進(jìn)行顯式的靜態(tài)變量初始化,這段代碼只會初始化一次,且在類被 第一次裝載時。如果你能讀懂并理解這段代碼,會幫助你對 static 關(guān)鍵字的認(rèn)識。在涉及到繼承的時候,會先初始化父類的 static 變量,然后是子類 的,依次類推。非靜態(tài)變量不是本文的主題,在此不做詳細(xì)討論,請參考 Think?in?Java 中的講解。
    靜態(tài)類
    通常一個普通類不允許聲明為靜態(tài)的,只有一個內(nèi)部類才可以。這時這個聲明為靜態(tài)的內(nèi)部類可以直接作為一個普通類來使用,而不需實(shí)例一個外部類。如下代碼所示:

    1. public ?class?StaticCls{
    2. ??public?static?void?main(String[]?args){
    3. ????OuterCls.InnerCls?oi=new?OuterCls.InnerCls();
    4. ??}
    5. }
    class ?OuterCls{
    1. ??public?static?class?InnerCls{
    2. ????InnerCls(){
    3. ??????System.out.println("InnerCls");
    4. ????}
    5. ???}
    6. }

    輸出結(jié)果會如你所料:
    InnerCls
    和普通類一樣。內(nèi)部類的其它用法請參閱Think?in?Java中的相關(guān)章節(jié),此處不作詳解。
    ****************************************************************************

    ????????????????????????????????????????????????????????????? 二、this?&?super
    ****************************************************************************
    ???? 中 討論了static的種種用法,通過用static來定義方法或成員,為我們編程提供了某種便利,從某種程度上可以說它類似于C語言中的全局函數(shù)和全 局變量。但是,并不是說有了這種便利,你便可以隨處使用,如果那樣的話,你便需要認(rèn)真考慮一下自己是否在用面向?qū)ο蟮乃枷刖幊蹋约旱某绦蚴欠袷敲嫦驅(qū)ο?的。好了,現(xiàn)在開始討論this&super這兩個關(guān)鍵字的意義和用法。
    在Java中,this通常指當(dāng)前對象,super則指父類的。 當(dāng)你想要引用當(dāng)前對象的某種東西,比如當(dāng)前對象的某個方法,或當(dāng)前對象的某個成員,你便可以利用this來實(shí)現(xiàn)這個目的,當(dāng)然,this的另一個用途是調(diào) 用當(dāng)前對象的另一個構(gòu)造函數(shù),這些馬上就要討論。如果你想引用父類的某種東西,則非super莫屬。由于this與super有如此相似的一些特性和與生 俱來的某種關(guān)系,所以我們在這一塊兒來討論,希望能幫助你區(qū)分和掌握它們兩個。
    在一般方法中
    最普遍的情況就是,在你的方法 中的某個形參名與當(dāng)前對象的某個成員有相同的名字,這時為了不至于混淆,你便需要明確使用this關(guān)鍵字來指明你要使用某個成員,使用方法是“this. 成員名”,而不帶this的那個便是形參。另外,還可以用“this.方法名”來引用當(dāng)前對象的某個方法,但這時this就不是必須的了,你可以直接用方 法名來訪問那個方法,編譯器會知道你要調(diào)用的是那一個。下面的代碼演示了上面的用法:
    1. public ?class?DemoThis{
    2. ??private?String?name;
    3. ??private?int?age;
    4. ??DemoThis(String?name,int?age){
    5. ????setName(name);?//你可以加上this來調(diào)用方法,像這樣:this.setName(name);但這并不是必須的
    6. ????setAge(age);
    7. ????this.print();
    8. ??}???
    9. ??public?void?setName(String?name){
    10. ????this.name=name;//此處必須指明你要引用成員變量
    11. ??}
    12. ??public?void?setAge(int?age){
    13. ????this.age=age;
    14. ??}
    15. ??public?void?print(){
    16. ????System.out.println( "Name=" +name+ "?Age=" +age);//在此行中并不需要用this,因?yàn)闆]有會導(dǎo)致混淆的東西
    17. ??}
    18. ??public?static?void?main(String[]?args){
    19. ????DemoThis?dt=new?DemoThis( "Kevin" , "22" );
    20. ??}
    21. }

    這段代碼很簡單,不用解釋你也應(yīng)該能看明白。在構(gòu)造函數(shù)中你看到用this.print(),你完全可以用print()來代替它,兩者效果一樣。下面我們修改這個程序,來演示super的用法。
    1. class ?Person{
    2. ??public?int?c;
    3. ??private?String?name;
    4. ??private?int?age;
    5. ??protected?void?setName(String?name){
    6. ????this.name=name;
    7. ??}
    8. ??protected?void?setAge(int?age){
    9. ????this.age=age;
    10. ??}
    11. ??protected?void?print(){
    12. ????System.out.println( "Name=" +name+ "?Age=" +age);
    13. ??}
    14. }
    15. public ?class?DemoSuper?extends?Person{
    16. ??public?void?print(){
    17. ????System.out.println( "DemoSuper:" );
    18. ????super.print();
    19. ??}
    20. ??public?static?void?main(String[]?args){
    21. ????DemoSuper?ds=new?DemoSuper();
    22. ????ds.setName( "kevin" );
    23. ????ds.setAge(22);
    24. ????ds.print();
    25. ??}
    26. }

    在DemoSuper中,重新定義的print方法覆寫了父類的print方法,它首先做一些自己的事情,然后調(diào)用父類的那個被覆寫了的方法。輸出結(jié)果說明了這一點(diǎn):
    DemoSuper:
    Name=kevin?Age=22
    這樣的使用方法是比較常用的。另外如果父類的成員可以被子類訪問,那你可以像使用this一樣使用它,用“super.父類中的成員名”的方式,但常常你并不是這樣來訪問父類中的成員名的。
    在構(gòu)造函數(shù)中
    構(gòu)造函數(shù)是一種特殊的方法,在對象初始化的時候自動調(diào)用。在構(gòu)造函數(shù)中,this和super也有上面說的種種使用方式,并且它還有特殊的地方,請看下面的例子:
    1. class ?Person{
    2. ??public?static?void?prt(String?s){
    3. ????System.out.println(s);
    4. ??}
    5. ??Person(){
    6. ????prt( "A?Person." );
    7. ??}
    8. ??Person(String?name){
    9. ????prt( "A?person?name?is:" +name);
    10. ??}
    11. }
    12. public ?class?Chinese?extends?Person{
    13. ??Chinese(){
    14. ????super();??//調(diào)用父類構(gòu)造函數(shù)(1)
    15. ????prt( "A?chinese." );//(4)
    16. ??}
    17. ??Chinese(String?name){
    18. ????super(name);//調(diào)用父類具有相同形參的構(gòu)造函數(shù)(2)
    19. ????prt( "his?name?is:" +name);
    20. ??}
    21. ??Chinese(String?name,int?age){
    22. ????this(name);//調(diào)用當(dāng)前具有相同形參的構(gòu)造函數(shù)(3)
    23. ????prt( "his?age?is:" +age);
    24. ??}
    25. ??public?static?void?main(String[]?args){
    26. ????Chinese?cn=new?Chinese();
    27. ????cn=new?Chinese( "kevin" );
    28. ????cn=new?Chinese( "kevin" ,22);
    29. ??}
    30. }

    在 這段程序中,this和super不再是像以前那樣用“.”連接一個方法或成員,而是直接在其后跟上適當(dāng)?shù)膮?shù),因此它的意義也就有了變化。super后 加參數(shù)的是用來調(diào)用父類中具有相同形式的構(gòu)造函數(shù),如1和2處。this后加參數(shù)則調(diào)用的是當(dāng)前具有相同參數(shù)的構(gòu)造函數(shù),如3處。當(dāng)然,在Chinese 的各個重載構(gòu)造函數(shù)中,this和super在一般方法中的各種用法也仍可使用,比如4處,你可以將它替換為“this.prt”(因?yàn)樗^承了父類中的 那個方法)或者是“super.prt”(因?yàn)樗歉割愔械姆椒ㄇ铱杀蛔宇愒L問),它照樣可以正確運(yùn)行。但這樣似乎就有點(diǎn)畫蛇添足的味道了。
    最后,寫了這么多,如果你能對“this通常指代當(dāng)前對象,super通常指代父類”這句話牢記在心,那么本篇便達(dá)到了目的,其它的你自會在以后的編程實(shí)踐當(dāng)中慢慢體會、掌握。另外關(guān)于本篇中提到的繼承,請參閱相關(guān)Java教程。
    關(guān)于最后一個final的意義和用法,也會很快貼出,敬請閱讀指正。
    **********************************************************************************************
    ??????????????????????????????????????????????????????????????
    三? final
    **********************************************************************************************
    final在Java中并不常用,然而它卻為我們提供了諸如在C語言中定義常量的功能,不僅如 此,final還可以讓你控制你的成員、方法或者是一個類是否可被覆寫或繼承等功能,這些特點(diǎn)使final在Java中擁有了一個不可或缺的地位,也是學(xué) 習(xí)Java時必須要知道和掌握的關(guān)鍵字之一。
    final成員
    當(dāng)你在類中定義變量時,在其前面加上final關(guān)鍵字,那便是 說,這個變量一旦被初始化便不可改變,這里不可改變的意思對基本類型來說是其值不可變,而對于對象變量來說其引用不可再變。其初始化可以在兩個地方,一是 其定義處,也就是說在final變量定義時直接給其賦值,二是在構(gòu)造函數(shù)中。這兩個地方只能選其一,要么在定義時給值,要么在構(gòu)造函數(shù)中給值,不能同時既 在定義時給了值,又在構(gòu)造函數(shù)中給另外的值。下面這段代碼演示了這一點(diǎn):
    1. import ?java.util.List;
    2. import ?java.util.ArrayList;
    3. import ?java.util.LinkedList;
    4. public ?class?Bat{
    5. ????final?PI=3.14;??????????//在定義時便給址值
    6. ????final?int?i;????????????//因?yàn)橐跇?gòu)造函數(shù)中進(jìn)行初始化,所以此處便不可再給值
    7. ????final?List?list;????????//此變量也與上面的一樣
    8. ????Bat(){
    9. ????????i=100;
    10. ????????list=new?LinkedList();
    11. ????}
    12. ????Bat(int?ii,List?l){
    13. ????????i=ii;
    14. ????????list=l;
    15. ????}
    16. ????public?static?void?main(String[]?args){
    17. ????????Bat?b=new?Bat();
    18. ????????b.list.add(new?Bat());
    19. ????????//b.i=25;
    20. ????????//b.list=new?ArrayList();
    21. ????????System.out.println( "I=" +b.i+ "?List?Type:" +b.list.getClass());
    22. ????????b=new?Bat(23,new?ArrayList());
    23. ????????b.list.add(new?Bat());
    24. ????????System.out.println( "I=" +b.i+ "?List?Type:" +b.list.getClass());
    25. ????}
    26. }

    此 程序很簡單的演示了final的常規(guī)用法。在這里使用在構(gòu)造函數(shù)中進(jìn)行初始化的方法,這使你有了一點(diǎn)靈活性。如Bat的兩個重載構(gòu)造函數(shù)所示,第一個缺省 構(gòu)造函數(shù)會為你提供默認(rèn)的值,重載的那個構(gòu)造函數(shù)會根據(jù)你所提供的值或類型為final變量初始化。然而有時你并不需要這種靈活性,你只需要在定義時便給 定其值并永不變化,這時就不要再用這種方法。在main方法中有兩行語句注釋掉了,如果你去掉注釋,程序便無法通過編譯,這便是說,不論是i的值或是 list的類型,一旦初始化,確實(shí)無法再更改。然而b可以通過重新初始化來指定i的值或list的類型,輸出結(jié)果中顯示了這一點(diǎn):
    I=100?List?Type:class?java.util.LinkedList
    I=23?List?Type:class?java.util.ArrayList
    還 有一種用法是定義方法中的參數(shù)為final,對于基本類型的變量,這樣做并沒有什么實(shí)際意義,因?yàn)榛绢愋偷淖兞吭谡{(diào)用方法時是傳值的,也就是說你可以在 方法中更改這個參數(shù)變量而不會影響到調(diào)用語句,然而對于對象變量,卻顯得很實(shí)用,因?yàn)閷ο笞兞吭趥鬟f時是傳遞其引用,這樣你在方法中對對象變量的修改也會 影響到調(diào)用語句中的對象變量,當(dāng)你在方法中不需要改變作為參數(shù)的對象變量時,明確使用final進(jìn)行聲明,會防止你無意的修改而影響到調(diào)用方法。
    另外方法中的內(nèi)部類在用到方法中的參變量時,此參變也必須聲明為final才可使用,如下代碼所示:
    1. public ?class?INClass{
    2. ???void?innerClass(final?String?str){
    3. ????????class?IClass{
    4. ????????????IClass(){
    5. ????????????????System.out.println(str);
    6. ????????????}
    7. ????????}
    8. ????????IClass?ic=new?IClass();
    9. ????}
    10. ??public?static?void?main(String[]?args){
    11. ??????INClass?inc=new?INClass();
    12. ??????inc.innerClass( "Hello" );
    13. ??}
    14. }

    final方法
    將 方法聲明為final,那就說明你已經(jīng)知道這個方法提供的功能已經(jīng)滿足你要求,不需要進(jìn)行擴(kuò)展,并且也不允許任何從此類繼承的類來覆寫這個方法,但是繼承 仍然可以繼承這個方法,也就是說可以直接使用。另外有一種被稱為inline的機(jī)制,它會使你在調(diào)用final方法時,直接將方法主體插入到調(diào)用處,而不 是進(jìn)行例行的方法調(diào)用,例如保存斷點(diǎn),壓棧等,這樣可能會使你的程序效率有所提高,然而當(dāng)你的方法主體非常龐大時,或你在多處調(diào)用此方法,那么你的調(diào)用主 體代碼便會迅速膨脹,可能反而會影響效率,所以你要慎用final進(jìn)行方法定義。

    final類
    當(dāng)你將final用 于類身上時,你就需要仔細(xì)考慮,因?yàn)橐粋€final類是無法被任何人繼承的,那也就意味著此類在一個繼承樹中是一個葉子類,并且此類的設(shè)計(jì)已被認(rèn)為很完美 而不需要進(jìn)行修改或擴(kuò)展。對于final類中的成員,你可以定義其為final,也可以不是final。而對于方法,由于所屬類為final的關(guān)系,自然 也就成了final型的。你也可以明確的給final類中的方法加上一個final,但這顯然沒有意義。
    下面的程序演示了final方法和final類的用法:
    1. final ?class?final{
    2. ????final?String?str= "final?Data" ;
    3. ????public?String?str1= "non?final?data" ;
    4. ????final?public?void?print(){
    5. ????????System.out.println( "final?method." );
    6. ????}
    7. ????public?void?what(){
    8. ????????System.out.println(str+ "\n" +str1);
    9. ????}
    10. }
    11. public ?class?FinalDemo?{???//extends?final?無法繼承?
    12. ????public?static?void?main(String[]?args){
    13. ????????final?f=new?final();
    14. ????????f.what();
    15. ????????f.print();
    16. ????}
    17. }

    從程序中可以看出,final類與普通類的使用幾乎沒有差別,只是它失去了被繼承的特性。final方法與非final方法的區(qū)別也很難從程序行看出,只是記住慎用。
    final在設(shè)計(jì)模式中的應(yīng)用
    在設(shè)計(jì)模式中有一種模式叫做不變模式,在Java中通過final關(guān)鍵字可以很容易的實(shí)現(xiàn)這個模式,在講解final成員時用到的程序Bat.java就是一個不變模式的例子。如果你對此感興趣,可以參考閻宏博士編寫的《Java與模式》一書中的講解。

    到 此為止,this,static,supert和final的使用已經(jīng)說完了,如果你對這四個關(guān)鍵字已經(jīng)能夠大致說出它們的區(qū)別與用法,那便說明你基本已 經(jīng)掌握。然而,世界上的任何東西都不是完美無缺的,Java提供這四個關(guān)鍵字,給程序員的編程帶來了很大的便利,但并不是說要讓你到處使用,一旦達(dá)到濫用 的程序,便適得其反,所以在使用時請一定要認(rèn)真考慮。

    posted on 2007-02-14 22:59 魚飛揚(yáng) 閱讀(144) 評論(0)  編輯  收藏 所屬分類: Java

    只有注冊用戶登錄后才能發(fā)表評論。


    網(wǎng)站導(dǎo)航:
     
    主站蜘蛛池模板: 久久亚洲精品无码VA大香大香| 亚洲Av永久无码精品一区二区| 成年免费大片黄在线观看com| 午夜性色一区二区三区免费不卡视频| 国产亚洲人成A在线V网站| 亚洲乱色伦图片区小说| 最好看最新的中文字幕免费| 亚洲一区视频在线播放| 亚洲老熟女五十路老熟女bbw| 曰批全过程免费视频网址 | 亚洲一区二区三区91| 你懂的网址免费国产| 亚洲精品国产精品乱码不卞 | 黄色免费在线网址| 男男AV纯肉无码免费播放无码 | 青青久久精品国产免费看| 毛片免费在线播放| 亚洲第一精品电影网| 最近免费mv在线观看动漫| 亚洲免费日韩无码系列| 国产天堂亚洲国产碰碰| 在线不卡免费视频| 亚洲国产成人va在线观看网址| 无码国产精品一区二区免费3p | 国产一区二区三区免费观在线| 无码欧精品亚洲日韩一区夜夜嗨| 亚洲久热无码av中文字幕| 毛片a级毛片免费观看免下载| 亚洲成年人电影网站| 亚洲免费中文字幕| 亚洲视频一区在线| 99久久99热精品免费观看国产| 亚洲AV永久无码精品一百度影院| 岛国精品一区免费视频在线观看| 亚洲AV无码一区二区三区在线观看 | 久久亚洲私人国产精品vA| 在线观看人成视频免费无遮挡| 亚洲伊人久久综合影院| 欧洲美女大片免费播放器视频 | 亚洲精品字幕在线观看| 美女无遮挡拍拍拍免费视频|