java飛機(jī)大戰(zhàn)心得體會(huì)

| 小揚(yáng)21147

心得體會(huì)是指一種讀書、實(shí)踐后所寫的感受性文字。語言類讀書心得同數(shù)學(xué)札記相近;體會(huì)是指將學(xué)習(xí)的東西運(yùn)用到實(shí)踐中去,通過實(shí)踐反思學(xué)習(xí)內(nèi)容并記錄下來的文字,近似于經(jīng)驗(yàn)總結(jié)。下面是小編帶來的有關(guān)java飛機(jī)大戰(zhàn)心得體會(huì),希望大家喜歡

java飛機(jī)大戰(zhàn)心得體會(huì)1

1.首先有玩家類,窗口類,主函數(shù)類和圖片文件(.jpg)

2.然后是先行知識(shí),創(chuàng)建窗口(JFrame),設(shè)置窗口屬性;窗口上不能直接添加組件(鍵盤監(jiān)聽等),所以先在窗口上添加容器(Jpanel),將組件(KeyAdapter)添加到容器;

3.畫出玩家:重寫窗口類中的paintComponent方法,創(chuàng)建Graphics對(duì)象,調(diào)用drawImage方法可畫圖,調(diào)用drawString方法可標(biāo)注其名字

4.移動(dòng):在窗口類中創(chuàng)建鍵盤監(jiān)聽抽象類KeyAdapter(實(shí)現(xiàn)了接口的部分方法但無具體操作),需要重寫該類的Keypressed方法和KeyRleased方法,賦給按鍵變量真值,隨后將該對(duì)象添加到窗口

5.隨機(jī)生成初始坐標(biāo):在開始游戲后隨機(jī)給定玩家的x、y坐標(biāo),創(chuàng)建Random對(duì)象,調(diào)用Random.nextInt(n)方法,n代表從[0,n)區(qū)間的隨機(jī)值。

6.最后通過一個(gè)Timer.schedule(匿名內(nèi)部類對(duì)象,指定延遲后開始,周期)方法來實(shí)現(xiàn)移動(dòng)效果。匿名內(nèi)部類【TimerTask的子類,重寫了run方法,包括repaint方法(實(shí)則調(diào)用paintComponent)和yidong方法】來重畫、監(jiān)聽鍵盤的指令()并作出相應(yīng)動(dòng)作

下面是源代碼(有注釋):

容器類

package a;

import javax.swing._;

import java.awt._;

import java.awt.event.KeyAdapter;

import java.awt.event.KeyEvent;

import java.util.Random;

import java.util.Timer;

import java.util.TimerTask;

import static a.Newgame.frame;

//新建面板

public class War extends JPanel {

private Timer timer;

private boolean sUp, sDown, sRight, sLeft;//右飛機(jī)按鍵變量

private boolean sW, sD, sS, sA;//左飛機(jī)按鍵變量

private Player1 player1 = new Player1();

private Player2 player2 = new Player2();

private ImageIcon img11 = player1.img1;

private ImageIcon img22 = player2.img2;

@Override

protected void paintComponent(Graphics g) {

super.paintComponent(g);//此句調(diào)用父類方法進(jìn)行界面重繪,恢復(fù)到某個(gè)界面,下面的步驟再畫玩家;如果注釋掉會(huì)有重影

//在面板上畫出玩家

g.drawImage(img11.getImage(), player1.x, player1.y, 100, 100, this);

g.drawString("P1", player1.x, player1.y);

g.drawImage(img22.getImage(), player2.x, player2.y, 100, 100, this);

g.drawString("P2", player2.x, player2.y);

}

public void startGame() {

timer = new Timer();

timer.schedule(new TimerTask() {//匿名內(nèi)部類(TimerTask的子類)

@Override

public void run() {//重寫run()函數(shù)

repaint();//調(diào)用重寫的paintComponent來畫兩飛機(jī)

yidong();//并每次判斷按下哪個(gè)鍵,然后移動(dòng)

}

}, 0, 50);//安排指定的任務(wù)從指定的延遲后開始進(jìn)行重復(fù)的固定延遲執(zhí)行。50毫秒執(zhí)行一次

}

public void yidong() {

//因?yàn)槊看蛴∫淮慰赡苡脩舭聪乱粋€(gè)飛機(jī)的幾個(gè)鍵或者兩個(gè)飛機(jī)的幾個(gè)鍵,這些是都要檢測(cè)到的,改成else if后只能檢測(cè)到一個(gè)鍵,無法實(shí)現(xiàn)兩架飛機(jī)同時(shí)多方向移動(dòng)

if (sW && player1.y > 0) {

player1.y -= player1.speed;

}

if (sA && player1.x > 0) {

player1.x -= player1.speed;

}

if (sS && player1.y < 700) {

player1.y += player1.speed;

}

if (sD && player1.x < 900) {

player1.x += player1.speed;

}

if (sUp && player2.y > 0) {

player2.y -= player2.speed;

}

if (sDown && player2.y < 700) {

player2.y += player2.speed;

}

if (sLeft && player2.x > 0) {

player2.x -= player2.speed;

}

if (sRight && player2.x < 900) {

player2.x += player2.speed;

}

}

public void act() {

//隨機(jī)生成兩飛機(jī)的初始坐標(biāo)

Random rd = new Random();

player1.x = rd.nextInt(900);

player1.y = rd.nextInt(700);

player2.x = rd.nextInt(900);

player2.y = rd.nextInt(700);

//開始游戲后獲得計(jì)時(shí)器開始監(jiān)聽并重畫

startGame();

//KeyAdapter是KeyListener的實(shí)現(xiàn)類,重寫了所有方法但沒有具體操作

KeyAdapter keyAdapter = new KeyAdapter() {

@Override

public void keyPressed(KeyEvent e) {

super.keyPressed(e);

int key = e.getKeyCode();

switch (key) {

case KeyEvent.VK_W:

sW = true;

break;

case KeyEvent.VK_A:

sA = true;

break;

case KeyEvent.VK_D:

sD = true;

break;

case KeyEvent.VK_S:

sS = true;

break;

case KeyEvent.VK_RIGHT:

sRight = true;

break;

case KeyEvent.VK_LEFT:

sLeft = true;

break;

case KeyEvent.VK_DOWN:

sDown = true;

break;

case KeyEvent.VK_UP:

sUp = true;

break;

}

}

@Override

public void keyReleased(KeyEvent e) {

int key = e.getKeyCode();

switch (key) {

case KeyEvent.VK_W:

sW = false;

break;

case KeyEvent.VK_A:

sA = false;

break;

case KeyEvent.VK_D:

sD = false;

break;

case KeyEvent.VK_S:

sS = false;

break;

case KeyEvent.VK_RIGHT:

sRight = false;

break;

case KeyEvent.VK_LEFT:

sLeft = false;

break;

case KeyEvent.VK_DOWN:

sDown = false;

break;

case KeyEvent.VK_UP:

sUp = false;

break;

}

}

};

frame.addKeyListener(keyAdapter);

}

}

主函數(shù)類:

package a;

import javax.swing._;

public class Newgame {

public static JFrame frame;

public static void main(String[] args) {

frame = new JFrame("逃出生天");

frame.setSize(1000,800);

//絕對(duì)布局組件位置和形狀不會(huì)隨窗體改變,不設(shè)置布局管理器就可以使用setBounds()來控制位置

frame.setLayout(null);

//設(shè)置窗體關(guān)閉程序自動(dòng)關(guān)閉

frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);

//窗體居中顯示

frame.setLocationRelativeTo(null);

//添加主面板

War war = new War();

frame.add(war);

//設(shè)置面板大小

war.setBounds(0,0,1000,800);

//設(shè)置主面板可見

frame.add(war);

frame.setVisible(true);

war.act();

}

}

1234567891011121314151617181920212223242526

玩家類:

玩家1

package a;

import javax.swing._;

public class Player1 {

public int x;

public int y;

public int speed = 50;

public ImageIcon img1 = new ImageIcon("D:\\Program_Files\\Jetbrains\\workspace_idea\\src\\a\\plane.jpg");

}

12345678910

玩家2

package a;

import javax.swing._;

public class Player2 {

public int x;

public int y;

public int speed = 50;

public ImageIcon img2 = new ImageIcon("D:\\Program_Files\\Jetbrains\\workspace_idea\\src\\a\\plane2.jpg");

}

做成一個(gè)桌面小游戲:

點(diǎn)“+”號(hào)

選好主函數(shù)類后點(diǎn)ok

然后點(diǎn)Build中的Build Artificts

選擇jar包點(diǎn)Build即可

然后jar包就在你的workspace生成了,找到它發(fā)送桌面快捷方式,更改圖標(biāo)名字即可

圖片轉(zhuǎn)換成圖標(biāo)的網(wǎng)站附上

java飛機(jī)大戰(zhàn)心得體會(huì)2

用Java制作游戲之前,一定要做到方向明確,思路清晰。首先確定自己需要用到幾個(gè)類,類里的內(nèi)容大致是什么,用腦圖進(jìn)行表達(dá)展現(xiàn)。

Java語言是一種跨平臺(tái)、適合于分布式計(jì)算環(huán)境的面向?qū)ο缶幊陶Z言,具有簡(jiǎn)單性、面向?qū)ο?、分布式,多線程性等,其中面向?qū)ο笥蟹庋b,繼承,多態(tài)三大特性。

1)封裝:數(shù)據(jù)和基于數(shù)據(jù)的操作封裝

2) 繼承:一個(gè)對(duì)象直接使用另一個(gè)對(duì)象的屬性和方法,子類可以繼承父類

3) 多態(tài):一個(gè)程序中同名的多個(gè)不同方法,實(shí)現(xiàn)多態(tài)的常見方法 子類對(duì)父類的覆蓋,利用重載在同一個(gè)類中定義多個(gè)同名的不同方法;

Java游戲制作中必須有 public static void main(String[] args)的主方法,main()方法是程序執(zhí)行的入口。例如我的飛機(jī)大戰(zhàn)main(),便是游戲開始執(zhí)行的入口,寫在GameFace類中。

游戲框架的制作,其大部分的代碼都是固定形式,可以通用。使用setVisible(true)的方法,改變默認(rèn)的不可見狀態(tài),setLayout(null)取消默認(rèn)的管理布局,然后setLocation(),setSize(),setBound()等方法設(shè)置其大小。將按鈕添加監(jiān)聽,實(shí)現(xiàn)監(jiān)聽。在對(duì)應(yīng)方法中編寫代碼,,為監(jiān)聽者創(chuàng)建對(duì)象,完成注冊(cè),即在接口名字前添加“add”。

游戲版面要繼承JPanel,其原因是JPanel帶有雙緩沖技術(shù),可以解決閃爍的問題,需要加入到JFrame窗體中,JPanel默認(rèn)的布局管理器是FlowLayout。調(diào)用畫筆,將所要顯示的圖片畫出來。然后再根據(jù)自己的思路,在游戲版面內(nèi)添加敵機(jī),實(shí)現(xiàn)矩形碰撞,生命值判定,游戲結(jié)束對(duì)話框。要熟練掌握調(diào)用其他的類,畫出分?jǐn)?shù),畫出生命值。制作容器,裝子彈,飛機(jī),炸彈。

制作游戲思路一定要清晰,對(duì)于共有屬性,可以建立一個(gè)父類,例如在創(chuàng)建一個(gè)FlyingObject類,這樣可以避免代碼的反復(fù)編寫。對(duì)于子類myplane 、enemies可以直接繼承父類,子類可以根據(jù)自己的需求再增加新的變量。在子類myplane中,畫出我的飛機(jī),控制飛機(jī)飛行的邊界,調(diào)用子彈容器,畫出容器中子彈,與敵機(jī)進(jìn)行碰撞檢測(cè)。飛機(jī)大戰(zhàn)還應(yīng)該實(shí)現(xiàn)鍵盤監(jiān)聽,例如Enter鍵實(shí)現(xiàn)暫停,上下左右鍵實(shí)現(xiàn)移動(dòng)等。在敵機(jī)Enemies類中,定義敵機(jī)隨機(jī)出現(xiàn)坐標(biāo),敵機(jī)圖片的相對(duì)路徑,敵機(jī)速度,敵機(jī)子彈與我的plane碰撞出現(xiàn)的結(jié)果以及敵機(jī)子彈生成的時(shí)間等。

游戲制作時(shí)需要注意的事項(xiàng):1.JLabel的位置一定要放對(duì),應(yīng)當(dāng)先添加背景JLabel,再添加其它控件。否則其它控件將被JLabel所遮擋。

2.對(duì)于多次重復(fù)編寫的代碼,盡可能的用for循環(huán),簡(jiǎn)單方便。但是也要注意跳出語句的使用。

3.調(diào)用函數(shù)時(shí)要注意順序

4.類名,方法名要遵循命名規(guī)范性,盡量做到見名知義,也要注意大小寫。

java飛機(jī)大戰(zhàn)心得體會(huì)3

線程:進(jìn)程(process)就是一塊包含了某些資源的內(nèi)存區(qū)域。操作系統(tǒng)利用進(jìn)程把它的工作劃分為一些功能單元。 線程:進(jìn)程中所包含的一個(gè)或多個(gè)執(zhí)行單元稱為線程(thread)。進(jìn)程還擁有一個(gè)私有的虛擬地址空間,該空間僅能被它所包含的線程訪問。 線程和進(jìn)程的區(qū)別如下: 1)一個(gè)進(jìn)程至少有一個(gè)線程。線程的劃分尺度小于進(jìn)程,使得多線程程序的并發(fā)性高。 另外,進(jìn)程在執(zhí)行過程中擁有獨(dú)立的內(nèi)存單元,而多個(gè)線程共享內(nèi)存,從而極大地提高了程序的運(yùn)行效率。 2)線程在執(zhí)行過程中與進(jìn)程的區(qū)別在于每個(gè)獨(dú)立的線程有一個(gè)程序運(yùn)行的入口、順序執(zhí)行序列和程序的出口。但是線程不能夠獨(dú)立執(zhí)行,必須依存在應(yīng)用程序中,由應(yīng)用程序提供多個(gè)線程執(zhí)行控制。 3)從邏輯角度來看,多線程的意義在于一個(gè)應(yīng)用程序中,有多個(gè)執(zhí)行部分可以同時(shí)執(zhí)行。但操作系統(tǒng)并沒有將多個(gè)線程看做多個(gè)獨(dú)立的應(yīng)用來實(shí)現(xiàn)進(jìn)程的調(diào)度和管理以及資源分配。

2 簡(jiǎn)述線程的狀態(tài)及其轉(zhuǎn)換 1)New,創(chuàng)建一個(gè)線程,但是線程并沒有進(jìn)行任何的操作。 2)Runnable,新線程從New狀態(tài),調(diào)用start方法轉(zhuǎn)換到Runnable狀態(tài)。線程調(diào)用start方法向線程調(diào)度程序(JVM或者是操作系統(tǒng))注冊(cè)一個(gè)線程,這個(gè)時(shí)候一切就緒只等CPU的時(shí)間。 3)Running,從Runnable狀態(tài)到Running狀態(tài),線程調(diào)度根據(jù)調(diào)度策略的不同調(diào)度不同的線程,被調(diào)度執(zhí)行的線程進(jìn)入Running狀態(tài),執(zhí)行run方法。 4)Dead狀態(tài),從Running狀態(tài)到Runnable,run方法運(yùn)行完畢后,線程就會(huì)被拋棄,線程就進(jìn)入Dead狀態(tài)。 5)Block狀態(tài),從Running狀態(tài)到Block狀態(tài),如果線程在運(yùn)行的狀態(tài)中因?yàn)镮/O阻塞、調(diào)用了線程的sleep方法以及調(diào)用對(duì)象的wait方法則線程將進(jìn)入阻塞狀態(tài),直到這些阻塞原因被結(jié)束,線程進(jìn)入到Runnable狀態(tài)。

3 簡(jiǎn)述線程的兩種創(chuàng)建方式以及它們的區(qū)別 創(chuàng)建線程的兩種方式: 1)使用Thread創(chuàng)建線程。Thread類是線程類,其每一個(gè)實(shí)例表示一個(gè)可以并發(fā)運(yùn)行的線程。我們可以通過繼承該類并重寫run方法來定義一個(gè)具體的線程。其中重寫run方法的目的是定義該線程要執(zhí)行的邏輯。啟動(dòng)線程時(shí)調(diào)用線程的start()方法而非直接調(diào)用run()方法。start()方法會(huì)將當(dāng)前線程納入線程調(diào)度,使當(dāng)前線程可以開始并發(fā)運(yùn)行。當(dāng)線程獲取時(shí)間片段后會(huì)自動(dòng)開始執(zhí)行run方法中的邏輯。 2)使用Runnable創(chuàng)建線程。實(shí)現(xiàn)Runnable接口并重寫run方法來定義線程體,然后在創(chuàng)建線程的時(shí)候?qū)unnable的實(shí)例傳入并啟動(dòng)線程。 兩種創(chuàng)建線程方式的區(qū)別: 使用Thread創(chuàng)建線程,編寫簡(jiǎn)單,可以直接操縱線程,無需使用Thread.currentThread(),但是不能夠再繼承其他類。 使用Runnable創(chuàng)建線程可以將線程與線程要執(zhí)行的任務(wù)分離開減少耦合,同時(shí)Java是單繼承的,定義一個(gè)類實(shí)現(xiàn)Runnable接口,這樣該類還可以繼承自其他類。

多線程實(shí)現(xiàn)方法

使用Thread創(chuàng)建線并啟動(dòng)線程

java.lang.Thread類是線程類,其每一個(gè)實(shí)例表示一個(gè)可以并發(fā)運(yùn)行的線程。我們可以通過繼承該類并重寫run方法來定義一個(gè)具體的線程。其中 重寫run方法的目的是定義該線程要執(zhí)行的邏輯。啟動(dòng)線程時(shí)調(diào)用線程的start()方法而非直接調(diào)用run()方法。start()方法會(huì)將當(dāng)前線程納入線程調(diào) 度,使當(dāng)前線程可以開始并發(fā)運(yùn)行。當(dāng)線程獲取時(shí)間片段后會(huì)自動(dòng)開始執(zhí)行run方法中的邏輯。

public class TestThread extends Thread{

@Override

public void run() {

for(int i=0;i<100;i++){

System.out.println("我是線程");

}

}

}

創(chuàng)建預(yù)啟動(dòng)線程

Thread thread = new TestThread();//實(shí)例化線程

thread.start();//啟動(dòng)線程

使用Runnable創(chuàng)建并啟動(dòng)線程

實(shí)現(xiàn)Runnable接口并重寫run方法來定義線程體,然后在創(chuàng)建線程的時(shí)候?qū)unnable的實(shí)例傳入并啟動(dòng)線程。 這樣做的好處在于可以將線程與線程要執(zhí)行的任務(wù)分離開減少耦合,同時(shí)java是單繼承的,定義一個(gè)類實(shí)現(xiàn)Runnable接口這樣的做法可以更好的 去實(shí)現(xiàn)其他父類或接口。因?yàn)榻涌谑嵌嗬^承關(guān)系。

public class TestRunnable implements Runnable{

@Override

public void run() {

for(int i=0;i<100;i++){

System.out.println("我是線程");

}

}

}

啟動(dòng)線程的方法:

Runnable runnable = new TestRunnable();

Thread thread = new Thread(runnable);//實(shí)例化線程并傳入線程體

thread.start();//啟動(dòng)線程

使用內(nèi)部類創(chuàng)建線程

通常我們可以通過匿名內(nèi)部類的方式創(chuàng)建線程,使用該方式可以簡(jiǎn)化編寫代碼的復(fù)雜度,當(dāng)一個(gè)線程僅需要一個(gè)實(shí)例時(shí)我們通常使用這種方式來 創(chuàng)建。 例如: 繼承Thread方式:

Thread thread = new Thread(){ //匿名類方式創(chuàng)建線程

public void run(){

//線程體

}

};

thread.start();//啟動(dòng)線程

Runnable方式:

Runnable runnable = new Runnable(){ //匿名類方式創(chuàng)建線程

public void run(){

}

};

Thread thread = new Thread(runnable);

thread.start();//啟動(dòng)線程

線程的方法

currentThread:方法可以用于獲取運(yùn)行當(dāng)前代碼片段的線程

Thread current = Thread.currentThread();

獲取線程信息 Thread提供了 獲取線程信息的相關(guān)方法: long getId():返回該線程的標(biāo)識(shí)符 String getName():返回該線程的名稱 int getPriority():返回線程的優(yōu)先級(jí) Thread.state getState():獲取線程的狀態(tài) boolean isAlive():測(cè)試線程是否處于活動(dòng)狀態(tài) boolean isDaemon():測(cè)試線程是否為守護(hù)線程 boolean isInterrupted():測(cè)試線程是否已經(jīng)中斷

線程優(yōu)先級(jí)

線程的切換是由線程調(diào)度控制的,我們無法通過代碼來干涉,但是我們可以通過提高線程的優(yōu)先級(jí)來最大程度的改善線程獲取時(shí)間片的幾率。 線程的優(yōu)先級(jí)被劃分為10級(jí),值分別為1-10,其中1最低,10最高。線程提供了3個(gè)常量來表示最低,最高,以及默認(rèn)優(yōu)先級(jí): Thread.MIN_PRIORITY, Thread.MAX_PRIORITY, Thread.NORM_PRIORITY 設(shè)置優(yōu)先級(jí)的方法為:

void setPriority(int priority)

守護(hù)線程

守護(hù)線程與普通線程在表現(xiàn)上沒有什么區(qū)別,我們只需要通過Thread提供的方法來設(shè)定即可: __void setDaemon(boolean )__ 當(dāng)參數(shù)為true時(shí)該線程為守護(hù)線程。 守護(hù)線程的特點(diǎn)是,當(dāng)進(jìn)程中只剩下守護(hù)線程時(shí),所有守護(hù)線程強(qiáng)制終止。 GC就是運(yùn)行在一個(gè)守護(hù)線程上的。 需要注意的是,設(shè)置線程為后臺(tái)線程要在該線程啟動(dòng)前設(shè)置。

Thread daemonThread = new Thread();

daemonThread.setDaemon(true);

daemonThread.start();

sleep方法

Thread的靜態(tài)方法sleep用于使當(dāng)前線程進(jìn)入阻塞狀態(tài): __static void sleep(long ms)__ 該方法會(huì)使當(dāng)前線程進(jìn)入阻塞狀態(tài)指定毫秒,當(dāng)指定毫秒阻塞后,當(dāng)前線程會(huì)重新進(jìn)入Runnable狀態(tài),等待分配時(shí)間片。 該方法聲明拋出一個(gè)InterruptException。所以在使用該方法時(shí)需要捕獲這個(gè)異常 改程序可能會(huì)出現(xiàn)"跳秒"現(xiàn)象,因?yàn)樽枞幻牒缶€程并非是立刻回到running狀態(tài),而是出于runnable狀態(tài),等待獲取時(shí)間片。那么這段等待 時(shí)間就是"誤差"。所以以上程序并非嚴(yán)格意義上的每隔一秒鐘執(zhí)行一次輸出。

yield方法:

Thread的靜態(tài)方法yield: __static void yield()__ 該方法用于使當(dāng)前線程主動(dòng)讓出當(dāng)次CPU時(shí)間片回到Runnable狀態(tài),等待分配時(shí)間片。

join方法

__void join()__ 該方法用于等待當(dāng)前線程結(jié)束。此方法是一個(gè)阻塞方法。 該方法聲明拋出InterruptException。

線程同步

synchronized關(guān)鍵字 多個(gè)線程并發(fā)讀寫同一個(gè)臨界資源時(shí)候會(huì)發(fā)生"線程并發(fā)安全問題“ 常見的臨界資源: 多線程共享實(shí)例變量 多線程共享靜態(tài)公共變量 若想解決線程安全問題,需要將異步的操作變?yōu)橥讲僮鳌?所謂異步操作是指多線程并發(fā)的操作,相當(dāng)于各干各的。 所謂同步操作是指有先后順序的操作,相當(dāng)于你干完我再干。

同步代碼塊(synchronized 關(guān)鍵字 ),同步代碼塊包含兩部分:一個(gè)作為鎖的對(duì)象的引用,一個(gè)作為由這個(gè)鎖保護(hù)的代碼塊 這個(gè)比較難理解故寫了下面代碼幫助理解

/__

_ 多線程并發(fā)安全問題

_ 當(dāng)多個(gè)線程同時(shí)操作同一資源時(shí),由于

_ 線程切換時(shí)機(jī)不確定,導(dǎo)致出現(xiàn)邏輯混亂。

_ 嚴(yán)重時(shí)可能導(dǎo)致系統(tǒng)崩潰。

_ @author ylg

_

_/

public class SyncDemo1 {

public static void main(String[] args) {

/_

_ 當(dāng)一個(gè)方法中的局部?jī)?nèi)部類想引用該方法

_ 的其他局部變量時(shí),這個(gè)變量必須被聲明

_ 為final的

_/

final Table table = new Table();

Thread t1 = new Thread(){

public void run(){

while(true){

int bean = table.getBean();

Thread.yield();//模擬線程切換

System.out.println(

getName()+":"+bean

);

}

}

};

Thread t2 = new Thread(){

public void run(){

while(true){

int bean = table.getBean();

Thread.yield();//模擬線程切換

System.out.println(

getName()+":"+bean

);

}

}

};

t1.start();

t2.start();

}

}

class Table{

//20個(gè)豆子

private int beans = 20;

/__

_ 當(dāng)一個(gè)方法被synchronized修飾后,該方法

_ 成為"同步方法"。多個(gè)線程不能同時(shí)進(jìn)入到

_ 方法內(nèi)部。

_ @return

_/

public synchronized int getBean(){

if(beans==0){

throw new RuntimeException("沒有豆子了!");

}

Thread.yield();//模擬線程切換

return beans--;

}

}

/__

_ 有效的縮小同步范圍可以保證在

_ 安全的前提下提高了并發(fā)的效率

_ @author ylg

_

_/

public class SyncDemo2 {

public static void main(String[] args) {

final Shop shop = new Shop();

Thread t1 = new Thread(){

public void run(){

shop.buy();

}

};

Thread t2 = new Thread(){

public void run(){

shop.buy();

}

};

t1.start();

t2.start();

}

}

class Shop{

/_

_ 在方法上使用synchroinzed,同步監(jiān)視器對(duì)象即當(dāng)前方法所屬對(duì)象:this

_/

// public synchronized void buy(){

public void buy(){

try{

Thread t = Thread.currentThread();

System.out.println(t+"正在挑選衣服..");

Thread.sleep(5000);

/_

_ 同步塊可以縮小同步范圍。

_ 但是必須保證"同步監(jiān)視器"即:"上鎖對(duì)象"是同一個(gè)才可以。

_ 通常,在一個(gè)方法中使用this所謂同步監(jiān)視器對(duì)象即可。

_/

synchronized (this) {

System.out.println(t+"正在試衣服..");

Thread.sleep(5000);

}

System.out.println(t+"結(jié)賬離開");

}catch(Exception e){

}

}

}

/__

_ synchronized也成為"互斥鎖"

_ 當(dāng)synchronzed修飾的是兩段代碼,但是"鎖對(duì)象"相同時(shí),這兩段代碼就是互斥的。

_ @author ylg

_

_/

public class SyncDemo4 {

public static void main(String[] args) {

final Boo b = new Boo();

Thread t1 = new Thread(){

public void run(){

b.methodA();

}

};

Thread t2 = new Thread(){

public void run(){

b.methodB();

}

};

t1.start();

t2.start();

}

}

class Boo{

public synchronized void methodA(){

Thread t = Thread.currentThread();

System.out.println(t+"正在調(diào)用方法A");

try {

Thread.sleep(5000);

} catch (InterruptedException e) {

}

System.out.println(t+"調(diào)用方法A完畢");

}

public synchronized void methodB(){

Thread t = Thread.currentThread();

System.out.println(t+"正在調(diào)用方法B");

try {

Thread.sleep(5000);

} catch (InterruptedException e) {

}

System.out.println(t+"調(diào)用方法B完畢");

java飛機(jī)大戰(zhàn)心得體會(huì)4

開發(fā)登錄界面、設(shè)計(jì)地圖加載、子彈飛機(jī)的加載、主控制程序的開發(fā)登錄界面的開發(fā),通常是在界面上顯示游戲菜單選項(xiàng),用戶可根據(jù)菜單項(xiàng)選擇需要的操作。菜單項(xiàng)是和游戲的功能緊密相關(guān)的。手機(jī)游戲的開發(fā)往往在界面和操作方式上有較高的要求,Java ME提供了低級(jí)界面開發(fā)的API,可用于游戲開發(fā)。低級(jí)界面畫布類主要有Canvas和GameCanvas,其中Canvas是屬于MIDP 1.0,GameCanvas則是屬于MIDP 2.0。低級(jí)用戶界面技術(shù)為用戶提供了靈活的開發(fā)方法,可以進(jìn)行一些較為底層的操作,例如:按鍵的處理事件更為豐富,組件位置的設(shè)置更為靈活。手機(jī)上的屏幕坐標(biāo)系和我們常規(guī)數(shù)學(xué)的坐標(biāo)系不相同。對(duì)于坐標(biāo)(x,y),若x值越大,則越向右,y值越大,則越向下。對(duì)于飛機(jī)射擊游戲,飛機(jī)在飛行過程中經(jīng)過不同的地理環(huán)境,例如大海、小島、石礁等作為地圖的背景。地圖的寬度通常要求能夠自動(dòng)匹配不同類型的手機(jī)寬度,高度則是以一個(gè)關(guān)卡所需的時(shí)間為依據(jù)進(jìn)行設(shè)計(jì)。在此我們采用簡(jiǎn)化操作方法,給地圖寬度和高度設(shè)置一個(gè)具體值。子彈的射擊應(yīng)該支持不同的方向,例如我方飛機(jī)自下而上發(fā)射子彈,敵方飛機(jī)自上而下發(fā)射子彈。當(dāng)子彈與飛機(jī)碰撞之后,飛機(jī)的血量相應(yīng)地要減少,如果飛機(jī)的血量為0,則產(chǎn)生爆炸效果,子彈和飛機(jī)都消失。飛機(jī)主角在開始游戲時(shí)出現(xiàn)在屏幕最底部的中間處,玩家可以通過上、下、左、右鍵控制其飛行方向,按下確認(rèn)鍵后可以發(fā)射子彈。敵機(jī)在游戲開始后從背景上部出現(xiàn),飛行,并發(fā)射子彈。在本任務(wù)中需要考慮敵機(jī)出現(xiàn)的位置,出現(xiàn)的數(shù)量和頻率,飛行的速度和子彈的發(fā)射方向。

為了更好地控制游戲的主體邏輯,可以統(tǒng)一對(duì)游戲地圖進(jìn)行設(shè)置,對(duì)主角飛機(jī)、敵方飛機(jī)初始化,并在多線程中實(shí)現(xiàn)游戲的邏輯控制。

游戲主框架的首要模塊的安排:

1.游戲的初始化。主要是完成游戲資源的加載,各種游戲?qū)ο?、變量取值的初始化,游戲運(yùn)行環(huán)境的獲取和設(shè)置,歷史記錄的讀取等。

2.游戲的主循環(huán)。執(zhí)行游戲處理的主要代碼,直到滿足退出要求才停止循環(huán),例如:玩家選擇退出游戲,玩家游戲失敗,玩家最終完成游戲等。主要實(shí)現(xiàn)分為三步:(1)獲得游戲的輸入信息;(2)處理游戲的邏輯;(3)更新游戲的畫面

3.退出游戲

Canvas為抽象類,負(fù)責(zé)圖形圖像的繪制和用戶交互。進(jìn)行低級(jí)玩家界面的開發(fā)通常需要繼承Canvas類,主要方法如下:

1.getHeight():獲取Canvas繪圖區(qū)域的高度。

2.getWidth():獲取Canvas繪圖區(qū)域的寬度。

3.paint(Graphics g):渲染畫布,向屏幕畫圖,通常將畫圖的操作放在該方法中實(shí)現(xiàn)。當(dāng)屏幕需要重新繪制時(shí),Java ME主線程會(huì)自動(dòng)調(diào)用paint方法,程序員不能在代碼中直接聲明調(diào)用該方法。

4.repaint():主動(dòng)向系統(tǒng)請(qǐng)求刷新界面,具體的刷新操作實(shí)際是通過調(diào)用paint方法來完成。

5.isDoubleBuffered():判斷手機(jī)設(shè)備是否匹配雙緩沖。有些手機(jī)匹配雙緩沖技術(shù),有些則不匹配。

6.getGameAction(int keyCode):將手機(jī)的鍵值切換為游戲動(dòng)作。

7.getKeyName(keyCode):得到按鍵的名字。

Graphics提供2D渲染能力,作用是在屏幕上繪制圖形,類似于一支畫筆。

Graphics繪制的圖形不能夠直接顯示,必須通過Canvas或GameCanvas才能顯示在屏幕上,因此Canvas和GameCanvas類似于可以顯示圖形的畫布

Graphics類支持繪制圖形主要包括以下3種:

(1)文本:可以設(shè)置文本的顏色、字體大小等

(2)圖像:可以直接繪制圖像文件或者從緩存中繪制

(3)2D幾何圖形:繪制點(diǎn)、直線、平面圖形等

Graphics類沒有構(gòu)造方式,獲取對(duì)象的途徑有3種:

1.Canvas類中的paint方法有一個(gè)Graphics對(duì)象參數(shù),系統(tǒng)會(huì)自動(dòng)調(diào)用paint方法,并傳進(jìn)一個(gè)Graphics對(duì)象,因此可以在paint方法中使用Graphics對(duì)象編寫繪圖代碼。

2.在GameCanvas類中通過getGraphics方法來獲取一個(gè)Graphics對(duì)象,因此可以在要求的地方靈活地編寫與繪圖有關(guān)的代碼。

3.Image對(duì)象的getGraphics()方法得到Graphics對(duì)象,可用于編寫雙緩沖區(qū)代碼。

1.直接用整幅圖片作為背景,再在上面重疊一層加入物件、擺設(shè)等,優(yōu)點(diǎn)在于圖形相對(duì)豐富、漂亮,但消耗資源較多,受手機(jī)硬件條件的影響不能做太大的圖。

2.游戲地圖是用一個(gè)個(gè)圖塊重復(fù)拼接成,而在程序中就可以通過一個(gè)較小的圖像文件和一個(gè)二維數(shù)組,繪制出一幅較大的地圖。具體的做法是將圖像文件劃分為若干個(gè)相同大小的圖塊(一般每個(gè)圖塊是16像素 × 16像素或者32像素 × 32像素),每個(gè)圖塊給一個(gè)索引值,例如:1表示草地圖塊,2表示磚頭圖塊。而二維數(shù)據(jù)中記錄的數(shù)字就是圖像文件中的圖塊索引值,例如:某個(gè)數(shù)組元素的數(shù)值若為1,則表示在此處畫草地。此方法的益處在于比較節(jié)約系統(tǒng)資源,可做出來的地圖相對(duì)比較一般。

地圖編輯器能夠幫助將地圖最后轉(zhuǎn)變成程序直接使用,所以一個(gè)好的地圖編輯器能夠加速游戲的開發(fā)周期。

業(yè)界已經(jīng)推出多款地圖編輯器,例如Mappy(MapWin)、Tiled、TILE STUDIO等,其中Mappy功能比較強(qiáng)大,可以很方便地對(duì)2D地圖進(jìn)行編輯。

下載mayppy軟件:HTTP:/ / www.tilemap.uk /mappy

假如需支持png圖片,那么還需下載兩個(gè)dll文件,這兩個(gè)文件也都放在Mappy軟件的網(wǎng)址上

zlib.dll用于文件壓縮

libpng12.dll是PNG圖像壓縮庫

將這兩個(gè)文件下載復(fù)制到Mappy可執(zhí)行文件的同一個(gè)目錄下即可,否則在導(dǎo)入PNG文件時(shí),會(huì)報(bào)圖5-23的錯(cuò)誤

制作游戲還需要用到圖像素材,因此除了Mappy軟件之外,還需要用到圖像處理軟件Photoshop來制作原始游戲素材。

J2SE中提供了多個(gè)接口和類管理集合,例如有Collection、List、Set、Map接口,實(shí)現(xiàn)接口的集合類有LinkedList、ArrayList、Vector、Hashtable、HashMap和WeakHashMap類。

在Java ME中只有java.util包提供Vector類,其功能和J2SE的Vector類似,實(shí)現(xiàn)的是一個(gè)動(dòng)態(tài)增長的數(shù)組,可以在程序代碼中調(diào)整或者裁減集合的大小,能向集合插入、刪除和修改元素。

每一個(gè)集合中的元素都被分派一個(gè)整數(shù)索引號(hào),能夠直接根據(jù)索引號(hào)插入和刪除一個(gè)元素,也能夠修改和得到一個(gè)元素的值。

為了更好地控制游戲的主體邏輯,可以統(tǒng)一對(duì)游戲地圖進(jìn)行設(shè)置,對(duì)主角飛機(jī)、敵方飛機(jī)初始化,并在多線程中實(shí)現(xiàn)游戲的邏輯控制。

46890