• <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>
            隨筆 - 97, 文章 - 22, 評論 - 81, 引用 - 0
            數據加載中……

            Pygame游戲開發 之二

            Pygame游戲開發之二

            初窺門徑

            游戲中需要用到很多對象,比如人物、野怪、建筑物、UI控件等等,他們的行為不盡相同,但是總有那么些共同點,如果將所有的東西都封裝成各自的類難免會寫上很多重復的代碼,因為兩個對象如果具有相同的行為(比如人物和野怪都可以行走)如果寫兩個move()函數,以后改起來就要修改兩個地方,不容易維護而且擴展很麻煩,但是人物和野怪又有其它不相同的行為,比如技能之類的不同。所以比較好的處理方法是抽象出一個生物類,然后讓人物和野怪去繼承它,舉個例子:

            class RenderObject(pygame.sprite.Sprite) :

                # 靜態變量

                images = []

                def __init__(self, selfdata) :

                    self.image       = self.images[ int(selfdata[0])  ]

                    self.size         = [self.image.get_width(), self.image.get_height()]

                    self.offset       = [0,0]

                    self.position     = [0,0]

                    self.speed       = [0.0,0.0]

                def move(self, xgo, ygo) :

                    do_move()

                def draw(self, screen) :

                    do_draw()

            以上這個類RenderObject是自己定義的用于渲染的類的基類,它繼承自pygame.sprite.Sprite,這里有必要先介紹一下pygame.sprite,這個模塊在pygame中是選擇性使用的,他可以作為游戲中很多對象的基類,并且將它擁有的東西繪制到Surface上,還有一個Group類是Sprite類的容器類,可以包含很多Sprite

                   RenderObject有三個類函數,__init__Python中的特殊函數,類似C語言中的構造函數,當類實例被創建的時候,這個函數會被自動執行,它的目的是執行一些該對象的必要的初始化工作(我在這個函數中加入了一個selfdata的列表,以便于不同類型的控件添加初始化信息)。move函數是執行移動操作,而draw函數則負責渲染。

                   首先來看下__init__函數,在Python中使用變量不需要提前聲明,所以要用到的變量我把它放在了__init__函數中進行初始化,注意到每個函數都有一個self參數,這個就好比C語言中的this指針,但是它是顯式聲明,隱式調用的。在使用類內部變量的時候需要加self做修飾,我們可以看到該類的一些成員變量:

            self.image      # 該渲染對象對應的Surface

            self.size        # 該渲染對象在Surface的矩形區域的寬高

            self.offset       # 該渲染對象在Surface的左上角坐標

            self.position     # 該渲染對象將被繪制到屏幕的位置

            self.speed       # 該渲染對象的速度

                   之所以要用到offsetsize,是因為該Surface不是完全被用來顯示的,我們只取某一張Surface的某一部分用于顯示,如果不是很明白,那么參看Player類的相關描述就知道為什么了。

                   對于move(self, xgo, ygo)函數,我們暫時可以這樣實現:

            if xgo :

            self.position[0] += self.speed[0] * xgo

            if ygo :

            self.position[1] += self.speed[1] * ygo

            xgoygo都是三值變量(1,0,-1),分別表示在兩個坐標軸方向是否發生了位移,并且這個位移是正向的還是逆向的。于是我們可以通過鍵盤輸入來傳遞對應的參數,position的值就會隨之變化了。注意如果這里速度為0,則表示靜態物體,position不會做任何變化。

            對于draw(self, screen)函數,用于將當前對象繪制到屏幕上,screen參數是上一節中提到的屏幕Surface,我們調用Surfaceblit函數就可以在屏幕上繪制當前對象了,像這樣:

            screen.blit( self.image, self.position, (self.offset, self.size))

            (self.offset, self.size)表示self.image這個Surface上對應的矩形區域。

            基類完成后,我們需要派生出一些子類,例如:

            class Player(RenderObject) :

                actionwait = 50

                def __init__(self, selfdata) :

                    RenderObject.__init__(self, selfdata)

                    self.size[0] /= 4

                    self.size[1] /= 4

                    self.speed = [0.1,0.1]

                    self.direction=3

                    self.action=0

                def move(self, xgo, ygo) :

                    RenderObject.move(self, xgo, ygo)

                    if xgo or ygo :

                        self.action = (self.action + 1) % (self.actionwait * 4)

                    self.offset[0] = (self.action / self.actionwait) * self.size[0]

                    self.offset[1] = self.direction * self.size[1]

             Player類作為游戲中的操作玩家,繼承自RenderObject,并且重寫它的__init__move函數,引入directionaction變量分別表示當前Player的方向和動作,如圖2-1(資源來自66RPG,一個基于RPG Maker xp業余游戲制作的交流、合作、分享的網站,在此感謝66RPG為我們游戲開發者提供這么多的資源下載,淚流滿面啊||*_*||)。

            2-1

            該圖是游戲人物的四方向行走圖,每一行代表一個方向,每一列代表一個動作,我們只需要通過鍵盤或者鼠標的輸入事件不斷改變offset的值,就可以順利得進行人物動畫的切換。這就是為什么之前要引入offsetsize的原因了,因為這個對象只取當前圖片的某一部分用于顯示,而且每次是根據人物不同的狀態進行切換的,還可以發現Player類中引入了一個actionwait的靜態變量,之所以用到它是因為現在的電腦的處理速度實在是太快了,每次move調用,我們會將self.action的值+1,并且循環往復,如果沒有一個延時操作,人物的動作切換會非常快,顯得十分不自然。

            同時可以根據自己的喜好派生出很多不同類型的游戲元素,比如野怪、按鈕、菜單等等。

             

            說到這里,還沒講到images這個靜態變量。RenderObject中定義了一個靜態列表,表示所有圖片的Surface對象的集合,我們可以在游戲載入的時候將所有的圖片都轉化成Surface,然后每次調用的時候直接在images中尋找對應的Surface即可,這樣就不用每次都調用load_image函數了。

            但是還有個問題,比如我現在有以下幾張圖片:hero.pnggo.pngtitle.pngmenu.png,那么不免要寫下面這段代碼:

            Imgs = []

            Imgs.append( load_image( ‘hero.png’) )

            Imgs.append( load_image( ‘go.png’) )

            Imgs.append( load_image( ‘title.png’) )

            Imgs.append( load_image( ‘menu.png’) )

            RenderObject.images = Imgs

            append是列表的函數,表示在已有列表中加入一個新的成員。這樣就可以把所有圖片預先載入到RenderObject.images列表中,以后訪問的時候只需要用下標索引即可。這也就是問題所在,因為現在圖片只有四張,這個代碼沒什么大的問題,但是如果圖片的數量增加到四十張、乃至四百張,那么問題也就隨之而來了,你將不斷地添加同一段代碼Imgs.append( load_image( ‘XXX’) ),這會使程序日漸冗長,而且每次新增一張圖片就需要添加一段代碼,某張圖片不需要了,有需要找到那段代碼將其刪除,維護會變得越來越繁瑣。

            圖片的文件名其實是一系列的字符串數據,我們大可以把它放到配置文件中,然后利用Python的文件讀取接口來將它們讀入,這樣以后需要添加一張圖片的時候只需要在配置文件中加入一個文件名即可,方便了許多。

            我把所有的圖片的文件名放到了一個叫image_config.txt的文件中,具體的結構組織可以看個人喜好,以下是我的組織形式:

             

            /Control

            title 1

            menu 1

            /Player

            hero 1

            monster 0

            /Animation

            cloud 10

             

            ’/’開頭的表示文件夾,比如/Control表示在Control文件夾下有兩種類型的文件(titlemenu),否則是某個文件的前綴加上當前類型文件的數量,比如title 2表示的其實是title-0.pngtitle-1.png這兩個文件,cloud 4則是cloud-0.pngcloud-1.pngcloud-2.pngcloud-3.png這四個文件。之所以要用編號的形式,是因為這些圖片有可能是一個動畫,又或許是一類相同的對象,這樣一來添加新圖片的時候會方便許多。

                   然后就是按照你自己定的配置文件格式進行解析,從而將圖片進行載入。以下是我的圖片載入過程,我把它放在了InitializeImage函數中:

            def InitializeImage() :

                config = open('image_config.txt', 'r')

                config.seek(0)

                filelist = config.readlines()

                config.close()

                nowfolder = ''

                imgs      = []

                for filename in filelist :

                    if len(filename) < 2:

                        continue

                    if filename[0] == '/' :

                        nowfolder = filename[1:-1]

                    else :

                        idx = filename.find(' ')

                        pre = filename[:idx]

                        for x in range(  int(filename[idx+1:-1])  ) :

                            pfile = nowfolder + '\\' + pre + '-' + str(x) + '.png'

                            print pfile

                            imgs.append( data.load_image(pfile) )

                RenderObject.images = imgs

                   open函數是Python中的文件讀取接口,類似C語言中的fopen,參數也是一樣的,返回的則是一個文件對象,我們首先調用seek(0)將文件頭指針定位到文件開頭,然后將該文件中所有的行通過readlines()讀入到filelist列表中,并且關閉文件。這時候filelist就儲存了該配置文件中所有的圖片文件信息,接下來的步驟是對它進行解析。

                   通過遍歷filelist的每個成員,判斷當前的這一行表示的是文件夾信息還是圖片文件信息。如果是文件夾則更新nowfoldernowfolder表示當前文件夾。filename[1:-1]表示不包含第一個字符和最后一個字符的子串(-1是最后一個字符的索引),之所以不要最后一個字符是因為讀入的時候是通過讀行的,所以最后一個字符在Win32下是\n也就是換行;如果是圖片文件信息,那么必定是類似 XXX num 的串,那么我們首先要找到這個串中的空格(這里需要保證文件名中不包含空格),通過字符串內置的find(‘ ‘)函數可以找到第一個出現空格的字符的下標,然后我們就可以把這個串分成兩部分了,將第二個部分也就是找到的下標到結尾的子串,將它轉換成int類型,這就是當前類型圖片的數量,最后用一個for循環將所有當前類型的圖片創建出來。一個個appendimgs上去即可。

                   再回頭來看RenderObject.__init__(self, selfdata)函數,該函數中還有selfdata變量,我把它定義成一個列表,不同的子類將會擁有各自不同的列表元素和個數,就好比對于動畫Animation類來說,它的值就是[st, en],表示當前動畫需要表示的始末圖片在images中的索引編號。

                   Smpl = Animation([3, 12])

                   可以簡單得通過以上的語句,就構造出一個Animation的對象,并且可以通過Smpl訪問它所有的成員變量來進行初始化對象的起始位置以及其它信息。這里的動畫指的是最普通的動畫,是通過不同圖片的循環播放來實現的,以上的實例就是一個擁有10張圖片的動畫。(未完待續)

                以上內容均為原創 轉載請注明出處

             

            posted on 2011-04-24 19:55 英雄哪里出來 閱讀(6137) 評論(3)  編輯 收藏 引用 所屬分類: Pygame

            評論

            # re: Pygame游戲開發 之二  回復  更多評論   

            留爪,持續關注
            2011-04-24 23:11 | zhaoyg

            # re: Pygame游戲開發 之二  回復  更多評論   

            @zhaoyg
            閣下在學Pygame嗎?
            2011-04-25 10:12 | 英雄哪里出來

            # re: Pygame游戲開發 之二  回復  更多評論   

            不錯不錯,期待更新
            2011-04-27 00:44 | 賊寇在何方
            久久www免费人成看国产片| 久久久久久久女国产乱让韩| 久久精品亚洲精品国产色婷| 国产精品久久久久久久久| 国产精品99久久精品爆乳| 久久受www免费人成_看片中文| 久久人人爽人人人人爽AV| 久久精品国产亚洲77777| 国产毛片久久久久久国产毛片| 久久这里都是精品| 99久久婷婷国产综合精品草原| 亚洲综合日韩久久成人AV| 久久精品成人欧美大片| 久久久久成人精品无码中文字幕| 久久综合九色欧美综合狠狠| 国产国产成人精品久久| 久久久久se色偷偷亚洲精品av| 国产免费久久久久久无码| 久久国产色AV免费观看| 久久综合久久性久99毛片| 久久亚洲高清观看| 久久这里只有精品18| 亚洲国产成人精品91久久久| 97久久精品人人做人人爽| 精品久久久久久无码专区| 色狠狠久久AV五月综合| 伊人久久亚洲综合影院| 国产成人精品综合久久久| 婷婷综合久久狠狠色99h| 77777亚洲午夜久久多人| 久久毛片一区二区| 久久久这里有精品| 欧美精品九九99久久在观看| 久久一本综合| 日本高清无卡码一区二区久久 | 久久综合亚洲色一区二区三区| 亚洲国产天堂久久综合网站| 国产精品久久久久jk制服| 久久精品水蜜桃av综合天堂| 色欲综合久久躁天天躁蜜桃| 99久久夜色精品国产网站|