[转]小黑之天地培训学习手记——第三、四周

[url]http://bbs.9ria.com/viewthread.php?tid=79563&extra=page%3D1%26amp%3Borderby%3Ddateline%26amp%3Bfilter%3D2592000[/url]

最近忙得很……在干一些不为人知的事。。=。 =。菜头帮的兄弟都叫我快点更新贴子,并喷得我体无完肤……其实我也想发,现在好了,终于可以写到了。
这次要写2周的东西。嘛,中途放了清明节,所以课程也不会很多。而且主要讲的是OOP思想神马的,所以不会很多实际上的知识。(想问各位愚人节过得怎么样。)
上周再上周4是由黑羽大叔……大哥亲自上阵……教我们殿堂之路之失落的60页。
黑羽讲的课基本是没有写什么代码的,都是OOP思想的一些法则,为了使我们这些新手少走弯路。这是内功啊~
编程有4条法则,新手们最好按照这个来做。
1、 宁用复合不用继承。
2、 200行代码需警惕,500行有问题
3、 继承不过5层
我擦,忘记了1条……对此我表示淡定。
对应的法则我来解说一下:
1、 新手用继承可能会产生混乱,例如一台电脑,它需要刻录东西的时候,就需要刻录功能了,于是继承了刻录机,就有了刻录功能。但换一种方式思考,我电脑继承刻录机……电脑是刻录机的儿子……好吧,这感觉不太爽,电脑会抗议的。这是新手容易犯的错误。虽然这样做可以达到目的,但这样就太委屈电脑了吧。
再看复合,我电脑缺少刻录功能,于是把刻录机拿过来集成在里面。这下爽了,下次内存不足的时候也直接拿一条好了。

2、 新手写程序时,类不要分得太多,也不要把所有东西放在一个类中。因为这很容易牵一发动全身的。而且代码不能重用。最主要的是读那代码会很辛苦的。
这让我想起以前到现在都在研究的一个飞机游戏。单单类文档就580行,而且很少注释。看着它我感到很茫然,作为一个新手的我感到鸭梨很大……

3、 如果真的要用继承的话,新手们,不要超过3层了,否则逻辑很容易混乱的,黑羽说,6层以上的是架构师们的事了。除非你是爱因斯坦吧(貌似现在英国有个13岁的小女孩智商比爱因斯坦还高?)。

最近菜头帮上老在讨论着设计模式。其实嘛,作为新手就对这个不需要太过于执着,但也不太过于冷淡。设计模式在我们写代码过程中就很自然地形成的。这时cobersky就来句惊人发言了:让你写一辈子的HelloWorld也写不出一个windows。

(我……我什么时候才能写出一个HelloWorld啊T^T……)

好吧,但先让我们新手写好这个HelloWorld好么?
要学设计模式就去看秋神的贴子吧。
http://bbs.9ria.com/thread-77927-1-1.html

另外黑羽要求我们去看看这些
OO五大原则(1.SRP 单一职责原则)
OO五大原则(2.OCP——开闭原则)
OO五大原则(3.LSP——里氏替换原则)
“依赖倒转”原则(DIP)
OOD的设计原则--接口隔离原则


好吧,这星期讲了一下二维数组。
所谓二维数组。也只是
Var ary1:array = new array();
Var ary2:array = new array();
Var ary:array = new array();
ary.push(ary1);
ary.push(ary2);
这样而已。
把一个数组放到另一个数组。
(坑爹啊。大学时的老师有木有这样说!!!!!!!纯洁的计算机学生你伤不起啊!!!!!!!!!!)
二维矩形使用实例如附件,不解说了。无力啊……囧~
array.rar (1018 Bytes)

第二个实例,永恒的打砖头游戏……
前面已经放过一个大砖头的代码了,但那个只写在时间轴上。这次我们分开类来写。
首先,我们需要砖头(Brick),小球(Ball),挡板(Bar),文档类(BlickBreak)
我们让他们做自己的东西,然后在文档类结合起来。
1、 砖头,砖头什么都不用做,纯粹的小受,那也太无聊了,我们就给个HP给他。
private var HP:int = 4*Math.random();
这个设置成私有,否则就对砖头太不负责任了,那么文档类怎么获得他呢?
用setter和getter吧。
写好setter和getter后就够了。

package
{
import flash.display.Graphics;
import flash.display.Sprite;
public class Brick extends Sprite
{
private var HP:int = 4*Math.random();

public function Brick()
{
drawBrick();
}
//画画
private function drawBrick():void
{
var g:Graphics = this.graphics;
g.beginFill(0x000000);
g.drawRect(0,0,50,20);
g.endFill();
}
//我们只需要知道HP是否为0
public function get getHP():Boolean
{
return HP == 0;
}
//设置HP
public function set setHP(Num:int):void
{
HP = HP - Num;
}
}
}


2、 我们小球需要运动,于是就需要增加一个EnterFrame事件了(可以不在小球那里添加enterFrane事件,只在小球里加了个公开的函数让文档类的enterframe调用,效果是差不多的。不过这样貌似增加了文档类和小球的耦合,改代码会麻烦点。)
运动嘛,就需要一个速度,X轴速度和Y轴速度,并且要写一个函数让小球跑到边界时会反弹。小球基本完成,以后需要什么再添加

package
{
import flash.display.Graphics;
import flash.display.Sprite;
import flash.events.Event;
import flash.events.EventDispatcher;

public class Ball extends Sprite
{
//球的各种速度
private var speedX:int = 10;
private var speedY:int = 10;
public function Ball()
{
draw();
this.addEventListener(Event.ENTER_FRAME,gameStep);
}

//画小球
private function draw():void
{
var g:Graphics = this.graphics;
g.beginFill(0x000000);
g.drawCircle(-6,-6,6);
g.endFill();
}
//游戏运行时
public function gameStep(evt:Event):void
{
ballMove();
checkBall();
}
//小球移动
private function ballMove():void
{
this.x += speedX;
this.y += speedY;
}
//检测小球有没出界
private function checkBall():void
{
if(this.x > stage.stageWidth - 2 || this.x < 0) speedX = -speedX;
if(this.y > stage.stageHeight - 2 || this.y < 0)speedY = -speedY;
}
//设置小球X坐标和Y坐标
/**
* 每次Y轴速度只需要取相反的。所以每次改变速度只需要传入-1就好了。
* 而X轴可能会改变,可能不变,因为撞到砖头的话X轴速度是不变的。
* 于是我们设一个不会取到的默认值,再判断一下就好了。
* */
public function speedXY(Y:Number,X:Number = 100):void
{
speedY = speedY*Y;
if(X != 100) speedX = X;
}

}
}


3、 挡板,无非就是左右移动了。然后就是左移动和右移动,移动可以用鼠标移动,也可以用键盘移动,键盘移动的话要有一个X轴速度。我用了键盘移动,所以要写个控制移动的接口函数。

package
{
import flash.display.Graphics;
import flash.display.Sprite;

public class Bar extends Sprite
{
//挡板移动速度
private var speed:int = 7;

public function Bar()
{
draw();

}

//画画
private function draw():void
{
var g:Graphics = this.graphics;
g.beginFill(0x000000);
g.drawRect(-40,0,70,10);
g.endFill();
}
//向左移动
public function moveLeft():void
{
this.x -= speed;
}
//向右移动
public function moveRight():void
{
this.x += speed;
}
}
}


4、 最后就是文档类了,这货任重而道远。首先在他上面添加各种我们需要的物件。接着是添加各种侦听,enterframe和键盘侦听。
文档类在enterframe中要做的事是检测砖头状态(检测砖头有没被撞,血量多少),检测挡板有没被撞,撞后怎样报复小球-b-……还有是挡板的控制。

package
{
import flash.display.DisplayObject;
import flash.display.Sprite;
import flash.events.Event;
import flash.events.KeyboardEvent;


public class BlickBreak extends Sprite
{

private var brick:Brick; //砖头
private var ball:Ball = new Ball(); //小球
private var bar:Bar = new Bar(); //挡板
//砖头数量
private var BrickNum:int = 20;
//存放砖头的数组
private var brickArray:Array = new Array();

//按左和按右
private var KeyLeft:Boolean ;
private var KeyRight:Boolean;


public function BlickBreak()
{
addSomething();
//添加各种侦听
stage.addEventListener(KeyboardEvent.KEY_DOWN,KeyDownHandler);
stage.addEventListener(KeyboardEvent.KEY_UP,KeyUpHandler);
this.addEventListener(Event.ENTER_FRAME,gameStep);
}
//键盘按下
private function KeyDownHandler(evt:KeyboardEvent):void
{
if(evt.keyCode == 65) KeyLeft = true;
if(evt.keyCode == 68) KeyRight = true;
}
//抬起
private function KeyUpHandler(evt:KeyboardEvent):void
{
if(evt.keyCode == 65) KeyLeft = false;
if(evt.keyCode == 68) KeyRight = false;
}
//各种addChild
private function addSomething():void
{
addBrick(BrickNum);
addChild(ball);
ball.x = 200;
ball.y = 200;
addChild(bar);
bar.x = 350;
bar.y = 350;
}
//添加砖头
private function addBrick(Num:int):void
{
for(var i:int = 0; i < Num ;i++){
brick = new Brick();
addChild(brick);
brick.x = i%int(stage.stageWidth/50)*53;
brick.y = int(i/int(stage.stageWidth/50))*24;
brickArray.push(brick);
}
}
//游戏进行时
private function gameStep(evt:Event):void
{
checkBrick();
checkHitbar();
control();

}
//检测小球与挡板碰撞
private function checkHitbar():void
{
if(hitTest(bar,ball)){
//碰撞后改变小球运动状态。
ball.speedXY(-1,(ball.x - bar.x)/bar.width*20)
}
}

//键盘控制挡板
private function control():void
{
if(KeyLeft == true) bar.moveLeft();
if(KeyRight == true) bar.moveRight();
}

//检测砖头状态
private function checkBrick():void
{
for (var i:int = 0;i < brickArray.length;i++){
var mc:Brick = brickArray[i];
if(hitTest(ball,mc)){
ball.speedXY(-1);
checkHP(mc,i);

}
}
}
//检查砖头剩余HP
private function checkHP(mc:Brick,i:int):void
{
if(mc.getHP){
reMoveChild(mc);
brickArray.splice(i,1);
}else mc.setHP = 1;
}
//移除某东东
private function reMoveChild(mc:DisplayObject):void
{
if(this.contains(mc)) this.removeChild(mc);
}


//碰撞检测
private function hitTest(mc1:DisplayObject,mc2:DisplayObject):Boolean
{
return mc1.hitTestObject(mc2);
}
}
}


各个类都做着自己的事,除了文档类= =。该公开的公开,该自己做的事自己做。。

实例三。纯爷都应该做的(是男人就下1000层。还有别的名字么。。= =)
算了,这个不说了,直接放源码……
我用了5个类,文档类,玩家类,控制类,地图,踏板。
要说的几点,首先是检测有没踏上踏板。
//判断是否踩在踏板上
               private function hitTest(plate:Plate):Boolean
{
return char.Top < plate.Top &&
char.Right > plate.Left &&
char.Left < plate.Right &&
char.Bottom > plate.Top
}


我翻译一下,画个图吧。
一下是玩家的图示。

[img]http://dl.iteye.com/upload/attachment/462398/55a7d66b-794f-3a87-9180-b472c9ecf11c.jpg[/img]


那么上面的碰撞是怎样判断呢。我们先看图,假如,玩家与踏板之间还有5点像素间隙。而玩家现在的速度是6像素。

[img]http://dl.iteye.com/upload/attachment/462400/8fe39e1b-d2bd-3ecc-95ef-3adfd48bef14.jpg[/img]


这样的话下一帧玩家肯定会落在踏板上吧。那么char.Bottom > plate.Top这句就返回true了。
再看,char.Left < plate.Right,玩家的左边界的X轴坐标现在已经比踏板小了,下一帧也一样。于是这句也返回true。那么怎样才是false呢?如图。

[img]http://dl.iteye.com/upload/attachment/462402/7913a79a-281a-3e49-923c-24ac4193403a.jpg[/img]


char.Right > plate.Left同上。不解释

最后是char.Top < plate.Top。这个,老师不是这样写的,我只是省时间。老师是记录碰撞后的上一帧位置作判断的。当然是老师的好了,我这种会有个猫腻,我只是懒。。=。 =。。那么为什么这样写也可以呢?试想一个,如果要条件不成立是怎样。

[img]http://dl.iteye.com/upload/attachment/462404/a3c0306a-e657-3770-b994-97daef87bf7c.jpg[/img]

明白了没?这是对立事件,如果条件为false就说明已经在踏板下面了,怎么碰啊?
至少我说这样的判断有猫腻,是因为如果玩家的人物长高一点,那人物就算脚踏不上踏板,但可能会用手“爬”上踏板。自己去理解。。。


最重要的说过了,然后说说我为什么把东西都放到map上,只因为这样的话,以后添加一些规则,例如gameover之类的就可以直接移除map达到移除其他的东西。而且现在我是每个踏板(虽然只有3个,因为循环利用,超出范围的踏板都被我重新拉回舞台下面重新飘上来了。)都放一个enterframe,所以资源有点浪费,如果只在map上添加一个的话就省些了。

最后,我使用了一个Control控制键盘,是我写的一个专门侦听键盘事件的类。教程如下:
[url]http://bbs.9ria.com/thread-77596-1-1.html[/url]
今天讲了位图,切图啊,粒子特效啊神马的,下周见吧。。=,=!
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值