欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

第十三章 Perl的面向?qū)ο缶幊?/p>

by flamephoenix

一、模塊簡(jiǎn)介
二、Perl中的類
三、創(chuàng)建類
四、構(gòu)造函數(shù)
  • 實(shí)例變量

  • 五、方法
    六、方法的輸出
    七、方法的調(diào)用
    八、重載
    九、析構(gòu)函數(shù)
    十、繼承
    十一、方法的重載
    十二、Perl類和對(duì)象的一些注釋

        本章介紹如何使用Perl的面向?qū)ο缶幊?OOP)特性及如何構(gòu)建對(duì)象,還包括繼承、方法重載和數(shù)據(jù)封裝等內(nèi)容。
    一、模塊簡(jiǎn)介
        模塊(module)就是Perl包(pachage)。Perl中的對(duì)象基于對(duì)包中數(shù)據(jù)項(xiàng)的引用。(引用見(jiàn)第x章引用)。
    詳見(jiàn)http://www.metronet.com的perlmod和perlobj。
        在用其它語(yǔ)言進(jìn)行面向?qū)ο缶幊虝r(shí),先聲明一個(gè)類然后創(chuàng)建該類的對(duì)象(實(shí)例),特定類所有對(duì)象的行為方式是相同的,由類方法確定,可以通過(guò)定義新類或從現(xiàn)存類繼承來(lái)創(chuàng)建類。已熟悉面向?qū)ο缶幊痰娜丝梢栽诖擞龅皆S多熟悉的術(shù)語(yǔ)。Perl一直是一個(gè)面向?qū)ο蟮恼Z(yǔ)言,在Perl5中,語(yǔ)法略有變動(dòng),更規(guī)范化了對(duì)象的使用。
        下面三個(gè)定義對(duì)理解對(duì)象、類和方法在Perl中如何工作至關(guān)重要。
        .類是一個(gè)Perl包,其中含提供對(duì)象方法的類。
        .方法是一個(gè)Perl子程序,類名是其第一個(gè)參數(shù)。
        .對(duì)象是對(duì)類中數(shù)據(jù)項(xiàng)的引用。
    二、Perl中的類
        再?gòu)?qiáng)調(diào)一下,一個(gè)Perl類是僅是一個(gè)包而已。當(dāng)你看到Perl文檔中提到“類”時(shí),把它看作“包”就行了。Perl5的語(yǔ)法可以創(chuàng)建類,如果你已熟悉C++,那么大部分語(yǔ)法你已經(jīng)掌握了。與Perl4不同的概念是用雙冒號(hào)(::)來(lái)標(biāo)識(shí)基本類和繼承類(子類)。
        面向?qū)ο蟮囊粋(gè)重要特性是繼承。Perl中的繼承特性與其它面向?qū)ο笳Z(yǔ)言不完全一樣,它只繼承方法,你必須用自己的機(jī)制來(lái)實(shí)現(xiàn)數(shù)據(jù)的繼承。
        因?yàn)槊總(gè)類是一個(gè)包,所以它有自己的名字空間及自己的符號(hào)名關(guān)聯(lián)數(shù)組(詳見(jiàn)第x章關(guān)聯(lián)數(shù)組),每個(gè)類因而可以使用自己的獨(dú)立符號(hào)名集。與包的引用結(jié)合,可以用單引號(hào)(')操作符來(lái)定位類中的變量,類中成員的定位形式如:$class'$member。在Perl5中,可用雙冒號(hào)替代單引號(hào)來(lái)獲得引用,如:$class'$member與$class::$member相同。
    三、創(chuàng)建類。
        本節(jié)介紹創(chuàng)建一個(gè)新類的必要步驟。下面使用的例子是創(chuàng)建一個(gè)稱為Cocoa的簡(jiǎn)單的類,其功能是輸出一個(gè)簡(jiǎn)單的Java應(yīng)用的源碼的必要部分。放心,這個(gè)例子不需要你有Java的知識(shí),但也不會(huì)使你成為Java專家,其目的是講述創(chuàng)建類的概念。
        首先,創(chuàng)建一個(gè)名為Cocoa.pm的包文件(擴(kuò)展名pm是包的缺省擴(kuò)展名,意為Perl Module)。一個(gè)模塊就是一個(gè)包,一個(gè)包就是一個(gè)類。在做其它事之前,先加入“1;”這樣一行,當(dāng)你增加其它行時(shí),記住保留“1;”為最后一行。這是Perl包的必需條件,否則該包就不會(huì)被Perl處理。下面是該文件的基本結(jié)構(gòu)。
    package Cocoa;

    #
    # Put "require" statements in for all required,imported packages
    #

    #
    # Just add code here
    #

    1; # terminate the package with the required 1;
        接下來(lái),我們往包里添加方法使之成為一個(gè)類。第一個(gè)需添加的方法是new(),它是創(chuàng)建對(duì)象時(shí)必須被調(diào)用的,new()方法是對(duì)象的構(gòu)造函數(shù)。
    四、構(gòu)造函數(shù)
        構(gòu)造函數(shù)是類的子程序,它返回與類名相關(guān)的一個(gè)引用。將類名與引用相結(jié)合稱為“祝!币粋(gè)對(duì)象,因?yàn)榻⒃摻Y(jié)合的函數(shù)名為bless(),其語(yǔ)法為:
        bless YeReference [,classname]
        YeReference是對(duì)被“祝福”的對(duì)象的引用,classname是可選項(xiàng),指定對(duì)象獲取方法的包名,其缺省值為當(dāng)前包名。
        創(chuàng)建一個(gè)構(gòu)建函數(shù)的方法為返回已與該類結(jié)合的內(nèi)部結(jié)構(gòu)的引用,如:
    sub new {
      my $this = {}; # Create an anonymous hash, and #self points to it.
      bless $this; # Connect the hash to the package Cocoa.
      return $this; # Return the reference to the hash.
    }

    1;
        {}創(chuàng)建一個(gè)對(duì)不含鍵/值對(duì)的哈希表(即關(guān)聯(lián)數(shù)組)的引用,返回值被賦給局域變量$this。函數(shù)bless()取出該引用,告訴對(duì)象它引用的是Cocoa,最后返回該引用。函數(shù)的返回值現(xiàn)在指向這個(gè)匿名哈希表。
        從new()函數(shù)返回后,$this引用被銷毀,但調(diào)用函數(shù)保存了對(duì)該哈希表的引用,因此該哈希表的引用數(shù)不會(huì)為零,從而使Perl在內(nèi)存中保存該哈希表。創(chuàng)建對(duì)象可如下調(diào)用:
        $cup = new Cocoa;
        下面語(yǔ)句為使用該包創(chuàng)建對(duì)象的例子:
    1 #!/usr/bin/perl
    2 push (@INC,'pwd');
    3 use Cocoa;
    4 $cup = new Cocoa;
        第一行指出Perl解釋器的位置,第二行中,將當(dāng)前目錄加到路徑尋找列表@INC中供尋找包時(shí)使用。你也可以在不同的目錄中創(chuàng)建你的模塊并指出該絕對(duì)路徑。例如,如果在/home/test/scripts/創(chuàng)建包,第二行就應(yīng)該如下:
        push (@INC , "/home/test/scripts");
        在第三行中,包含上包Cocoa.pm以獲取腳本中所需功能。use語(yǔ)句告訴Perl在@INC路徑尋找文件Cocoa.pm并包含到解析的源文件拷貝中。use語(yǔ)句是使用類必須的。第四行調(diào)用new函數(shù)創(chuàng)建對(duì)象,這是Perl的妙處,也是其易混淆之處,也是其強(qiáng)大之處。創(chuàng)建對(duì)象的方法有多種,可以這樣寫(xiě):
        $cup = cocoa->new();
        如果你是C程序員,可以用雙冒號(hào)強(qiáng)制使用Cocoa包中的new()函數(shù),如:
        $cup = Cocoa::new();
        可以在構(gòu)造函數(shù)中加入更多的代碼,如在Cocoa.pm中,可以在每個(gè)對(duì)象創(chuàng)建時(shí)輸出一個(gè)簡(jiǎn)單聲明,還可以用構(gòu)造函數(shù)初始化變量或設(shè)置數(shù)組或指針。
    注意:
    1、一定要在構(gòu)造函數(shù)中初始化變量;
    2、一定要用my函數(shù)在方法中創(chuàng)建變量;
    3、一定不要在方法中使用local,除非真的想把變量傳遞給其它子程序;
    4、一定不要在類模塊中使用全局變量。
        加上聲明的Cocoa構(gòu)造函數(shù)如下:
    sub new {
      my $this = {};
      print "\n /* \n ** Created by Cocoa.pm \n ** Use at own risk";
      print "\n ** Did this code even get pass the javac compiler? ";
      print "\n **/ \n";
      bless $this;
      return $this;
    }
        也可以簡(jiǎn)單地調(diào)用包內(nèi)或包外的其它函數(shù)來(lái)做更多的初始化工作,如:
    sub new {
      my $this = {}
      bless $this;
      $this->doInitialization();
      return $this;
    }
        創(chuàng)建類時(shí),應(yīng)該允許它可被繼承,應(yīng)該可以把類名作為第一個(gè)參數(shù)來(lái)調(diào)用new函數(shù),那么new函數(shù)就象下面的語(yǔ)句:
    sub new {
      my $class = shift; # Get the request class name
      my $this = {};
      bless $this, $class # Use class name to bless() reference
      $this->doInitialization(); return $this;
    }
        此方法使用戶可以下列三種方式之一來(lái)進(jìn)行調(diào)用:
  • Cocoa::new()
  • Cocoa->new()
  • new Cocoa
  •     可以多次bless一個(gè)引用對(duì)象,然而,新的將被bless的類必然把對(duì)象已被bless的引用去掉,對(duì)C和Pascal程序員來(lái)說(shuō),這就象把一個(gè)指針賦給分配的一塊內(nèi)存,再把同一指針賦給另一塊內(nèi)存而不釋放掉前一塊內(nèi)存。總之,一個(gè)Perl對(duì)象每一時(shí)刻只能屬于一個(gè)類。
        對(duì)象和引用的真正區(qū)別是什么呢?Perl對(duì)象被bless以屬于某類,引用則不然,如果引用被bless,它將屬于一個(gè)類,也便成了對(duì)象。對(duì)象知道自己屬于哪個(gè)類,引用則不屬于任何類。

  • 實(shí)例變量


  •     作為構(gòu)造函數(shù)的new()函數(shù)的參數(shù)叫做實(shí)例變量。實(shí)例變量在創(chuàng)建對(duì)象的每個(gè)實(shí)例時(shí)用于初始化,例如可以用new()函數(shù)為對(duì)象的每個(gè)實(shí)例起個(gè)名字。
        可以用匿名哈希表或匿名數(shù)組來(lái)保存實(shí)例變量。
        用哈希表的代碼如下:
    sub new {
    my $type = shift;
    my %parm = @_;
    my $this = {};
    $this->{'Name'} = $parm{'Name'};
    $this->{'x'} = $parm{'x'};
    $this->{'y'} = $parm{'y'};
    bless $this, $type;
    }
        用數(shù)組保存的代碼如下:
    sub new {
    my $type = shift;
    my %parm = @_;
    my $this = [];
    $this->[0] = $parm{'Name'};
    $this->[1] = $parm{'x'};
    $this->[2] = $parm{'y'};
    bless $this, $type;
    }
        構(gòu)造對(duì)象時(shí),可以如下傳遞參數(shù):
        $mug = Cocoa::new( 'Name' => 'top','x' => 10,'y' => 20 );
        操作符=>與逗號(hào)操作服功能相同,但=>可讀性好。訪問(wèn)方法如下:
        print "Name=$mug->{'Name'}\n";
        print "x=$mug->{'x'}\n";
        print "y=$mug->{'y'}\n";
    五、方法
        Perl類的方法只不過(guò)是一個(gè)Perl子程序而已,也即通常所說(shuō)的成員函數(shù)。Perl的方法定義不提供任何特殊語(yǔ)法,但規(guī)定方法的第一個(gè)參數(shù)為對(duì)象或其被引用的包。Perl有兩種方法:靜態(tài)方法和虛方法。
        靜態(tài)方法第一個(gè)參數(shù)為類名,虛方法第一個(gè)參數(shù)為對(duì)象的引用。方法處理第一個(gè)參數(shù)的方式?jīng)Q定了它是靜態(tài)的還是虛的。靜態(tài)方法一般忽略掉第一個(gè)參數(shù),因?yàn)樗鼈円呀?jīng)知道自己在哪個(gè)類了,構(gòu)造函數(shù)即靜態(tài)方法。虛方法通常首先把第一個(gè)參數(shù)shift到變量self或this中,然后將該值作普通的引用使用。如:
    1. sub nameLister {
    2.     my $this = shift;
    3.     my ($keys ,$value );
    4.     while (($key, $value) = each (%$this)) {
    5.         print "\t$key is $value.\n";
    6.     }
    7. }
    六、方法的輸出
        如果你現(xiàn)在想引用Cocoa.pm包,將會(huì)得到編譯錯(cuò)誤說(shuō)未找到方法,這是因?yàn)镃ocoa.pm的方法還沒(méi)有輸出。輸出方法需要Exporter模塊,在包的開(kāi)始部分加上下列兩行:
        require Exporter;
        @ISA = qw (Exporter);
        這兩行包含上Exporter.pm模塊,并把Exporter類名加入@ISA數(shù)組以供查找。接下來(lái)把你自己的類方法列在@EXPORT數(shù)組中就可以了。例如想輸出方法closeMain和declareMain,語(yǔ)句如下:
        @EXPORT = qw (declareMain , closeMain);
        Perl類的繼承是通過(guò)@ISA數(shù)組實(shí)現(xiàn)的。@ISA數(shù)組不需要在任何包中定義,然而,一旦它被定義,Perl就把它看作目錄名的特殊數(shù)組。它與@INC數(shù)組類似,@INC是包含文件的尋找路徑。@ISA數(shù)組含有類(包)名,當(dāng)一個(gè)方法在當(dāng)前包中未找到時(shí)就到@ISA中的包去尋找。@ISA中還含有當(dāng)前類繼承的基類名。
        類中調(diào)用的所有方法必須屬于同一個(gè)類或@ISA數(shù)組定義的基類。如果一個(gè)方法在@ISA數(shù)組中未找到,Perl就到AUTOLOAD()子程序中尋找,這個(gè)可選的子程序在當(dāng)前包中用sub定義。若使用AUTOLOAD子程序,必須用use Autoload;語(yǔ)句調(diào)用autoload.pm包。AUTOLOAD子程序嘗試從已安裝的Perl庫(kù)中裝載調(diào)用的方法。如果AUTOLOAD也失敗了,Perl再到UNIVERSAL類做最后一次嘗試,如果仍失敗,Perl就生成關(guān)于該無(wú)法解析函數(shù)的錯(cuò)誤。
    七、方法的調(diào)用
        調(diào)用一個(gè)對(duì)象的方法有兩種方法,一是通過(guò)該對(duì)象的引用(虛方法),一是直接使用類名(靜態(tài)方法)。當(dāng)然該方法必須已被輸出,F(xiàn)在給Cocoa類增加一些方法,代碼如下:
    package Cocoa;
    require Exporter;
    @ISA = qw(Exporter);
    @EXPORT = qw(setImports, declareMain, closeMain);
    #
    # This routine creates the references for imports in Java functions
    #
    sub setImports{
      my $class = shift @_;
      my @names = @_;
      foreach (@names) {
        print "import " . $_ . ";\n";
      }
    }
    #
    # This routine declares the main function in a Java script
    #
    sub declareMain{
      my $class = shift @_;
      my ( $name, $extends, $implements) = @_;
      print "\n public class $name";
      if ($extends) {
        print " extends " . $extends;
      }
      if ($implements) {
        print " implements " . $implements;
      }
      print " { \n";
    }
    #
    # This routine declares the main function in a Java script
    #
    sub closeMain{
      print "} \n";
    }
    #
    # This subroutine creates the header for the file.
    #
    sub new {
      my $this = {};
      print "\n /* \n ** Created by Cocoa.pm \n ** Use at own risk \n */ \n";
      bless $this;
      return $this;
    }

    1;
        現(xiàn)在,我們寫(xiě)一個(gè)簡(jiǎn)單的Perl腳本來(lái)使用該類的方法,下面是創(chuàng)建一個(gè)Java applet源代碼骨架的腳本代碼:
    #!/usr/bin/perl
    use Cocoa;
    $cup = new Cocoa;
    $cup->setImports( 'java.io.InputStream', 'java.net.*');
    $cup->declareMain( "Msg" , "java.applet.Applet", "Runnable");
    $cup->closeMain();
        這段腳本創(chuàng)建了一個(gè)叫做Msg的Java applet,它擴(kuò)展(extend)了java.applet.Applet小應(yīng)用程序并使之可運(yùn)行(runnable),其中最后三行也可以寫(xiě)成如下:
    Cocoa::setImports($cup, 'java.io.InputStream', 'java.net.*');
    Cocoa::declareMain($cup, "Msg" , "java.applet.Applet", "Runnable");
    Cocoa::closeMain($cup);
        其運(yùn)行結(jié)果如下:
    /*
    ** Created by Cocoa.pm
    ** Use at own risk
    */
    import java.io.InputStream;
    import java.net.*;

    public class Msg extends java.applet.Applet implements Runnable {
    }
        注意:如果用->操作符調(diào)用方法(也叫間接調(diào)用),參數(shù)必須用括號(hào)括起來(lái),如:$cup->setImports( 'java.io.InputStream', 'java.net.*');而雙冒號(hào)調(diào)用如:Cocoa::setImports($cup, 'java.io.InputStream', 'java.net.*');也可去掉括號(hào)寫(xiě)成:Cocoa::setImports $cup, 'java.io.InputStream', 'java.net.*' ;
    八、重載
        有時(shí)需要指定使用哪個(gè)類的方法,如兩個(gè)不同的類有同名方法的時(shí)候。假設(shè)類Espresso和Qava都定義了方法grind,可以用::操作符指定使用Qava的方法:
        $mess = Qava::grind("whole","lotta","bags");
        Qava::grind($mess, "whole","lotta","bags");
        可以根據(jù)程序的運(yùn)行情況來(lái)選擇使用哪個(gè)類的方法,這可以通過(guò)使用符號(hào)引用去調(diào)用來(lái)實(shí)現(xiàn):
        $method = $local ? "Qava::" : "Espresso::";
        $cup->{$method}grind(@args);
    九、析構(gòu)函數(shù)
        Perl跟蹤對(duì)象的鏈接數(shù)目,當(dāng)某對(duì)象的最后一個(gè)應(yīng)用釋放到內(nèi)存池時(shí),該對(duì)象就自動(dòng)銷毀。對(duì)象的析構(gòu)發(fā)生在代碼停止后,腳本將要結(jié)束時(shí)。對(duì)于全局變量而言,析構(gòu)發(fā)生在最后一行代碼運(yùn)行之后。
        如果你想在對(duì)象被釋放之前獲取控制權(quán),可以定義DESTROY()方法。DESTROY()在對(duì)象將釋放前被調(diào)用,使你可以做一些清理工作。DESTROY()函數(shù)不自動(dòng)調(diào)用其它DESTROY()函數(shù),Perl不做內(nèi)置的析構(gòu)工作。如果構(gòu)造函數(shù)從基類多次bless,DESTROY()可能需要調(diào)用其它類的DESTROY()函數(shù)。當(dāng)一個(gè)對(duì)象被釋放時(shí),其內(nèi)含的所有對(duì)象引用自動(dòng)釋放、銷毀。
        一般來(lái)說(shuō),不需要定義DESTROY()函數(shù),如果需要,其形式如下:
    sub DESTROY {
    #
    # Add code here.
    #
    }
        因?yàn)槎喾N目的,Perl使用了簡(jiǎn)單的、基于引用的垃圾回收系統(tǒng)。任何對(duì)象的引用數(shù)目必須大于零,否則該對(duì)象的內(nèi)存就被釋放。當(dāng)程序退出時(shí),Perl的一個(gè)徹底的查找并銷毀函數(shù)進(jìn)行垃圾回收,進(jìn)程中的一切被簡(jiǎn)單地刪除。在UNIX類的系統(tǒng)中,這像是多余的,但在內(nèi)嵌式系統(tǒng)或多線程環(huán)境中這確實(shí)很必要。
    十、繼承
        類方法通過(guò)@ISA數(shù)組繼承,變量的繼承必須明確設(shè)定。下例創(chuàng)建兩個(gè)類Bean.pm和Coffee.pm,其中Coffee.pm繼承Bean.pm的一些功能。此例演示如何從基類(或稱超類)繼承實(shí)例變量,其方法為調(diào)用基類的構(gòu)造函數(shù)并把自己的實(shí)例變量加到新對(duì)象中。
        Bean.pm代碼如下:
    package Bean;
    require Exporter;
    @ISA = qw(Exporter);
    @EXPORT = qw(setBeanType);

    sub new {
      my $type = shift;
      my $this = {};
      $this->{'Bean'} = 'Colombian';
      bless $this, $type;
      return $this;
    }

    #
    # This subroutine sets the class name
    sub setBeanType{
      my ($class, $name) = @_;
      $class->{'Bean'} = $name;
      print "Set bean to $name \n";
    }
    1;
        此類中,用$this變量設(shè)置一個(gè)匿名哈希表,將'Bean'類型設(shè)為'Colombian'。方法setBeanType()用于改變'Bean'類型,它使用$class引用獲得對(duì)對(duì)象哈希表的訪問(wèn)。
        Coffee.pm代碼如下:
    1  #
    2  # The Coffee.pm file to illustrate inheritance.
    3  #
    4  package Coffee;
    5  require Exporter;
    6  require Bean;
    7  @ISA = qw(Exporter, Bean);
    8  @EXPORT = qw(setImports, declareMain, closeMain);
    9  #
    10 # set item
    11 #
    12 sub setCoffeeType{
    13   my ($class,$name) = @_;
    14   $class->{'Coffee'} = $name;
    15   print "Set coffee type to $name \n";
    16   }
    17 #
    18 # constructor
    19 #
    20 sub new {
    21   my $type = shift;
    22   my $this = Bean->new(); ##### <- LOOK HERE!!! ####
    23   $this->{'Coffee'} = 'Instant'; # unless told otherwise
    24   bless $this, $type;
    25   return $this;
    26   }
    27 1;
        第6行的require Bean;語(yǔ)句包含了Bean.pm文件和所有相關(guān)函數(shù),方法setCoffeeType()用于設(shè)置局域變量$class->{'Coffee'}的值。在構(gòu)造函數(shù)new()中,$this指向Bean.pm返回的匿名哈希表的指針,而不是在本地創(chuàng)建一個(gè),下面兩個(gè)語(yǔ)句分別為創(chuàng)建不同的哈希表從而與Bean.pm構(gòu)造函數(shù)創(chuàng)建的哈希表無(wú)關(guān)的情況和繼承的情況:
        my $this = {}; #非繼承
        my $this = $theSuperClass->new(); #繼承
        下面代碼演示如何調(diào)用繼承的方法:
    1  #!/usr/bin/perl
    2  push (@INC,'pwd');
    3  use Coffee;
    4  $cup = new Coffee;
    5  print "\n -------------------- Initial values ------------ \n";
    6  print "Coffee: $cup->{'Coffee'} \n";
    7  print "Bean: $cup->{'Bean'} \n";
    8  print "\n -------------------- Change Bean Type ---------- \n";
    9  $cup->setBeanType('Mixed');
    10 print "Bean Type is now $cup->{'Bean'} \n";
    11 print "\n ------------------ Change Coffee Type ---------- \n";
    12 $cup->setCoffeeType('Instant');
    13 print "Type of coffee: $cup->{'Coffee'} \n";
        該代碼的結(jié)果輸出如下:
    -------------------- Initial values ------------
    Coffee: Instant
    Bean: Colombian
    -------------------- Change Bean Type ----------
    Set bean to Mixed
    Bean Type is now Mixed
    ------------------ Change Coffee Type ----------
    Set coffee type to Instant
    Type of coffee: Instant
        上述代碼中,先輸出對(duì)象創(chuàng)建時(shí)哈希表中索引為'Bean'和'Coffee'的值,然后調(diào)用各成員函數(shù)改變值后再輸出。
        方法可以有多個(gè)參數(shù),現(xiàn)在向Coffee.pm模塊增加函數(shù)makeCup(),代碼如下:
    sub makeCup {
      my ($class, $cream, $sugar, $dope) = @_;
      print "\n================================== \n";
      print "Making a cup \n";
      print "Add cream \n" if ($cream);
      print "Add $sugar sugar cubes\n" if ($sugar);
      print "Making some really addictive coffee ;-) \n" if ($dope);
      print "================================== \n";
    }
        此函數(shù)可有三個(gè)參數(shù),不同數(shù)目、值的參數(shù)產(chǎn)生不同的結(jié)果,例如:
    1  #!/usr/bin/perl
    2  push (@INC,'pwd');
    3  use Coffee;
    4  $cup = new Coffee;
    5  #
    6  # With no parameters
    7  #
    8  print "\n Calling with no parameters: \n";
    9  $cup->makeCup;
    10 #
    11 # With one parameter
    12 #
    13 print "\n Calling with one parameter: \n";
    14 $cup->makeCup('1');
    15 #
    16 # With two parameters
    17 #
    18 print "\n Calling with two parameters: \n";
    19 $cup->makeCup(1,'2');
    20 #
    21 # With all three parameters
    22 #
    23 print "\n Calling with three parameters: \n";
    24 $cup->makeCup('1',3,'1');
        其結(jié)果輸出如下:
    Calling with no parameters:
    ==================================
    Making a cup
    ==================================
    Calling with one parameter:
    ==================================
    Making a cup
    Add cream
    ==================================
    Calling with two parameters:
    ==================================
    Making a cup
    Add cream
    Add 2 sugar cubes
    ==================================
    Calling with three parameters:
    ==================================
    Making a cup
    Add cream
    Add 3 sugar cubes
    Making some really addictive coffee ;-)
    ==================================
        在此例中,函數(shù)makeCup()的參數(shù)既可為字符串也可為整數(shù),處理結(jié)果相同,你也可以把這兩種類型的數(shù)據(jù)處理區(qū)分開(kāi)。在對(duì)參數(shù)的處理中,可以設(shè)置缺省的值,也可以根據(jù)實(shí)際輸入?yún)?shù)值的個(gè)數(shù)給予不同處理。
    十一、子類方法的重載
        繼承的好處在于可以獲得基類輸出的方法的功能,而有時(shí)需要對(duì)基類的方法重載以獲得更具體或不同的功能。下面在Bean.pm類中加入方法printType(),代碼如下:
    sub printType {
      my $class = shift @_;
      print "The type of Bean is $class->{'Bean'} \n";
    }
        然后更新其@EXPORT數(shù)組來(lái)輸出:
        @EXPORT = qw ( setBeanType , printType );
        現(xiàn)在來(lái)調(diào)用函數(shù)printType(),有三種調(diào)用方法:
    $cup->Coffee::printType();
    $cup->printType();
    $cup->Bean::printType();
        輸出分別如下:
    The type of Bean is Mixed
    The type of Bean is Mixed
    The type of Bean is Mixed
        為什么都一樣呢?因?yàn)樵谧宇愔袥](méi)有定義函數(shù)printType(),所以實(shí)際均調(diào)用了基類中的方法。如果想使子類有其自己的printType()函數(shù),必須在Coffee.pm類中加以定義:
    #
    # This routine prints the type of $class->{'Coffee'}
    #
    sub printType {
      my $class = shift @_;
      print "The type of Coffee is $class->{'Coffee'} \n";
    }
        然后更新其@EXPORT數(shù)組:
        @EXPORT = qw(setImports, declareMain, closeMain, printType);
        現(xiàn)在輸出結(jié)果變成了:
    The type of Coffee is Instant
    The type of Coffee is Instant
    The type of Bean is Mixed
        現(xiàn)在只有當(dāng)給定了Bean::時(shí)才調(diào)用基類的方法,否則直接調(diào)用子類的方法。
        那么如果不知道基類名該如何調(diào)用基類方法呢?方法是使用偽類保留字SUPER::。在類方法內(nèi)使用語(yǔ)法如:$this->SUPER::function(...argument list...); ,它將從@ISA列表中尋找。剛才的語(yǔ)句用SUPER::替換Bean::可以寫(xiě)為$cup->SUPER::printType(); ,其結(jié)果輸出相同,為:
    The type of Bean is Mixed
    十二、Perl類和對(duì)象的一些注釋
        OOP的最大好處就是代碼重用。OOP用數(shù)據(jù)封裝來(lái)隱藏一些復(fù)雜的代碼,Perl的包和模塊通過(guò)my函數(shù)提供數(shù)據(jù)封裝功能,但是Perl并不保證子類一定不會(huì)直接訪問(wèn)基類的變量,這確實(shí)減少了數(shù)據(jù)封裝的好處,雖然這種動(dòng)作是可以做到的,但卻是個(gè)很壞的編程風(fēng)格。
    注意:
    1、一定要通過(guò)方法來(lái)訪問(wèn)類變量。
    2、一定不要從模塊外部直接訪問(wèn)類變量。
        當(dāng)編寫(xiě)包時(shí),應(yīng)該保證方法所需的條件已具備或通過(guò)參數(shù)傳遞給它。在包內(nèi)部,應(yīng)保證對(duì)全局變量的訪問(wèn)只用通過(guò)方法傳遞的引用來(lái)訪問(wèn)。對(duì)于方法要使用的靜態(tài)或全局?jǐn)?shù)據(jù),應(yīng)該在基類中用local()來(lái)定義,子類通過(guò)調(diào)用基類來(lái)獲取。有時(shí),子類可能需要改變這種數(shù)據(jù),這時(shí),基類可能就不知道怎樣去尋找新的數(shù)據(jù),因此,這時(shí)最好定義對(duì)該數(shù)據(jù)的引用,子類和基類都通過(guò)引用來(lái)改變?cè)摂?shù)據(jù)。
        最后,你將看到如下方式來(lái)使用對(duì)象和類:
        use coffee::Bean;
        這句語(yǔ)句的含義是“在@INC數(shù)組所有目錄的Coffee子目錄來(lái)尋找Bean.pm”。如果把Bean.pm移到./Coffee目錄,上面的例子將用這一use語(yǔ)句來(lái)工作。這樣的好處是有條理地組織類的代碼。再如,下面的語(yǔ)句:
        use Another::Sub::Menu;
        意味著如下子目錄樹(shù):
        ./Another/Sub/Menu.pm

    上一章 下一章 目錄