Beautifl - Flash Gallery

Preview

パーティクルの応用で弾幕3 - 超最適化
coppieee 2009年8月30日 MIT License
?
コードのリファクタリング & 最適化  
Nicolasが最適化してくれました。  
http://wonderfl.net/code/63f88f2189846bdc7275a01d5d228b1607344e51  
BitmapData#draw() を BitmapData#coloyPixces() に変えるだけ!  
めっちゃ早いよ!これ!  
by coppieee  
  
さらに最適化!  
http://wonderfl.net/code/9dcd5e428a43c4b20f69f86873f2831fa9ec32f3  
に触発されて、私も最適化してみた。Linked listは使いたくなかったので別の方法で。  
これでもか!ってぐらいがんばった。  
  
Particleのプール用意して、  
削除のところ工夫して、  
回転の計算のアルゴリズム変更して、  
とかもろもろ。  
  
弾の大きさを変更したのはオマケ。(オマケなんだけど、小さければ小さいほど速くなるよ。)  
きっとこれ以上の最適化は無理だと思うよ!  
  
  
_canvas.copyPixels()のところを別の描画のアルゴリズムで置き換えたら、あるいはもっと速く・・・。  
by coppieee
      // forked from coppieee's forked from: パーティクルの応用で弾幕2 - 早くなったよ~
// forked from coppieeee's パーティクルの応用で弾幕2 - 早くなったよ~
// forked from coppieeee's 弾幕 - パーティクルの応用で弾幕

/*
 * コードのリファクタリング & 最適化
 * Nicolasが最適化してくれました。
 * http://wonderfl.net/code/63f88f2189846bdc7275a01d5d228b1607344e51
 * BitmapData#draw() を BitmapData#coloyPixces() に変えるだけ!
 * めっちゃ早いよ!これ!
 * by coppieee
 */
/*
 * さらに最適化!
 * http://wonderfl.net/code/9dcd5e428a43c4b20f69f86873f2831fa9ec32f3
 * に触発されて、私も最適化してみた。Linked listは使いたくなかったので別の方法で。
 * これでもか!ってぐらいがんばった。
 * 
 * Particleのプール用意して、
 * 削除のところ工夫して、
 * 回転の計算のアルゴリズム変更して、
 * とかもろもろ。
 * 
 * 弾の大きさを変更したのはオマケ。(オマケなんだけど、小さければ小さいほど速くなるよ。)
 * きっとこれ以上の最適化は無理だと思うよ!
 * 
 * 
 * _canvas.copyPixels()のところを別の描画のアルゴリズムで置き換えたら、あるいはもっと速く・・・。
 * by coppieee
 */

package 
{
    import flash.display.Bitmap;
    import flash.display.BitmapData;
    import flash.display.Graphics;
    import flash.display.Shape;
    import flash.display.Sprite;
    import flash.events.Event;
    import flash.geom.ColorTransform;
    import flash.geom.Matrix;
    import flash.geom.Point;
    import flash.geom.Rectangle;
    import flash.text.TextField;
    import flash.text.TextFieldAutoSize;
    import net.hires.debug.Stats;
    
    [SWF(width="512",height="512", frameRate="60" )]
    public class PShooting extends Sprite 
    {
        //1fpsあたりの弾の生成数
        public static const BULLET_COUNT:int = 20;
        
        public static const WIDTH:Number = 512;
        public static const HEIGHT:Number = 512;
        
        //弾のビットマップキャッシュ
        private var _bulletImg:BitmapData;
        //キャンバス
        private var _canvas:BitmapData;
        //パーティクルリスト(Vector#unshift()がバグってて使えないので、しょうがないからArray使う。)
        private var _particles:/*Particle*/Array;
        //敵。というか弾の再生位置。
        private var _enemy:Particle;
        //パーティクルプール(いらなくなったパーティクルを溜めとくとこ)
        private var _particlesPool:Vector.<Particle>;
        
        public function PShooting()
        {
            Wonderfl.capture_delay( 7 );
            
            //キャンバスの生成
            _canvas = new BitmapData(WIDTH, HEIGHT,false,0x000000);
            var cb:Bitmap = new Bitmap(_canvas);
            addChild(cb);
            
            _particles = [];
            _enemy = new Particle();
            _particlesPool = new Vector.<Particle>();
            
            //弾のBitmapの生成
            //Shapeに円を書く。
            var shape:Shape = new Shape();
            var g:Graphics = shape.graphics;
            g.beginFill(0x00AAAA, 0.5);
            g.drawCircle(8, 8, 8);
            g.beginFill(0x55EEFF);
            g.drawCircle(8, 8, 4);
            g.endFill();
            
            //BitmapDataにdraw()
            _bulletImg = new BitmapData(shape.width, shape.height, true, 0xFFFFFF);
            _bulletImg.draw(shape);
            
            //Statsの生成
            addChild(new Stats());
            
            addEventListener(Event.ENTER_FRAME, onEnterFrame);
            
            //particlsの個数表示用のTextField生成
            var tf:TextField = new TextField();
            tf.y = 100;
            tf.textColor = 0xFFFFFF;
            tf.background = true;
            tf.backgroundColor = 0x000000;
            tf.autoSize = TextFieldAutoSize.LEFT;
            tf.multiline = true;
            addChild(tf);
            addEventListener(Event.ENTER_FRAME, function(e:Event):void {
                tf.text = "bullets:" + _particles.length +"\n pool:"+_particlesPool.length ;
            });
        }
        private var _radian:Number = 0;
        
        private function onEnterFrame(e:Event):void
        {
            _canvas.lock();
            
            var ct:ColorTransform = new ColorTransform (0.9, 0.9, 0.95);
            _canvas.colorTransform(_canvas.rect, ct);
            //下の方が軽いけど、見た目が悪くなる。
            //_canvas.fillRect(_canvas.rect, 0x000000);
            
            //弾の生成場所をマウスのところへ移動
            _enemy.vx = (stage.mouseX - _enemy.x) * 0.05;
            _enemy.vy = (stage.mouseY - _enemy.y) * 0.05;
            _enemy.x += _enemy.vx;
            _enemy.y += _enemy.vy;
            
            //回転
            _radian += (Math.PI / 180) *62.1;
            var bRadian:Number = _radian;
            //弾の生成
            for (var i:int = 0; i <BULLET_COUNT; i++ )
            {
                var vx:Number = Math.cos(bRadian) * 3;
                var vy:Number = Math.sin(bRadian) * 3;
                var newP:Particle;
                if (_particlesPool.length != 0)
                {
                    newP = _particlesPool.pop();
                }else{
                    newP = new Particle();
                }
                newP.x = _enemy.x;
                newP.y = _enemy.y;
                newP.vx = vx;
                newP.vy = vy;
                _particles.unshift(newP);
                
                bRadian += Math.PI *2 / BULLET_COUNT;
            }
            
            //必要な計算はforループの前にしておく。
            var bulletRect:Rectangle = _bulletImg.rect;
            var bulletPoint:Point = new Point();
            
            var phi:Number = Math.PI / 180 * 80;
            var cosPhi:Number = Math.cos(phi) * 0.02; //0.02はオマケ
            var sinPhi:Number = Math.sin(phi) * 0.02;
            
            var bWidth_2:int = _bulletImg.width / 2;
            var bHeight_2:int = _bulletImg.height / 2;
            var xMin:int = - bWidth_2 / 2;
            var yMin:int = - bHeight_2 / 2;
            var xMax:int = bWidth_2 + WIDTH;
            var yMax:int = bHeight_2 + HEIGHT;
            
            //弾の移動
            for (var j:int = _particles.length - 1; j != -1 ;j--)
            {
                var p:Particle = _particles[j];
                
                //ベクトルうめぇww
                p.ax = p.vx * cosPhi - p.vy * sinPhi;
                p.ay = p.vy * cosPhi + p.vx * sinPhi;
                p.x += p.vx;
                p.y += p.vy;
                p.vx += p.ax;
                p.vy += p.ay;
                
                //画面外に出たら削除
                if (p.x < xMin || p.x > xMax || p.y < yMin || p.y > yMax)
                {
                    _particlesPool.push(p);
                    _particles.splice(j, 1);
                    continue;
                }
                
                bulletPoint.x = p.x - bWidth_2;
                bulletPoint.y = p.y - bHeight_2;
                _canvas.copyPixels(_bulletImg, bulletRect, bulletPoint);
            }
            _canvas.unlock();
        }
    }
}

class Particle
{
    public var x:Number = 0;
    public var y:Number = 0;
    public var vx:Number = 0;
    public var vy:Number = 0;
    public var ax:Number = 0;
    public var ay:Number = 0;
}