android 飞机大战

  1. package com.pbicv.ddpx.game;
    
    import android.content.Context;
    import android.content.res.TypedArray;
    import android.graphics.Bitmap;
    import android.graphics.BitmapFactory;
    import android.graphics.Canvas;
    import android.graphics.Paint;
    import android.graphics.Rect;
    import android.graphics.RectF;
    import android.text.TextPaint;
    import android.util.AttributeSet;
    import android.view.MotionEvent;
    import android.view.View;
    
    import com.pbicv.ddpx.R;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    import java.util.List;
    
    
    public class GameView extends View {
    
        private Paint paint;
        private Paint textPaint;
        private CombatAircraft combatAircraft = null;
        private List<Sprite> sprites = new ArrayList<Sprite>();
        private List<Sprite> spritesNeedAdded = new ArrayList<Sprite>();
        //0:combatAircraft
        //1:explosion
        //2:yellowBullet
        //3:blueBullet
        //4:smallEnemyPlane
        //5:middleEnemyPlane
        //6:bigEnemyPlane
        //7:bombAward
        //8:bulletAward
        //9:pause1
        //10:pause2
        //11:bomb
        private List<Bitmap> bitmaps = new ArrayList<Bitmap>();
        private float density = getResources().getDisplayMetrics().density;//屏幕密度
        public static final int STATUS_GAME_STARTED = 1;//游戏开始
        public static final int STATUS_GAME_PAUSED = 2;//游戏暂停
        public static final int STATUS_GAME_OVER = 3;//游戏结束
        public static final int STATUS_GAME_DESTROYED = 4;//游戏销毁
        private int status = STATUS_GAME_DESTROYED;//初始为销毁状态
        private long frame = 0;//总共绘制的帧数
        private long score = 0;//总得分
        private float fontSize = 12;//默认的字体大小,用于绘制左上角的文本
        private float fontSize2 = 20;//用于在Game Over的时候绘制Dialog中的文本
        private float borderSize = 2;//Game Over的Dialog的边框
        private Rect continueRect = new Rect();//"继续"、"重新开始"按钮的Rect
    
        //触摸事件相关的变量
        private static final int TOUCH_MOVE = 1;//移动
        private static final int TOUCH_SINGLE_CLICK = 2;//单击
        private static final int TOUCH_DOUBLE_CLICK = 3;//双击
        //一次单击事件由DOWN和UP两个事件合成,假设从down到up间隔小于200毫秒,我们就认为发生了一次单击事件
        private static final int singleClickDurationTime = 200;
        //一次双击事件由两个点击事件合成,两个单击事件之间小于300毫秒,我们就认为发生了一次双击事件
        private static final int doubleClickDurationTime = 300;
        private long lastSingleClickTime = -1;//上次发生单击的时刻
        private long touchDownTime = -1;//触点按下的时刻
        private long touchUpTime = -1;//触点弹起的时刻
        private float touchX = -1;//触点的x坐标
        private float touchY = -1;//触点的y坐标
    
        public GameView(Context context) {
            super(context);
            init(null, 0);
        }
    
        public GameView(Context context, AttributeSet attrs) {
            super(context, attrs);
            init(attrs, 0);
        }
    
        public GameView(Context context, AttributeSet attrs, int defStyle) {
            super(context, attrs, defStyle);
            init(attrs, defStyle);
        }
    
        private void init(AttributeSet attrs, int defStyle) {
            final TypedArray a = getContext().obtainStyledAttributes(
                    attrs, R.styleable.GameView, defStyle, 0);
            a.recycle();
            //初始化paint
            paint = new Paint();
            paint.setStyle(Paint.Style.FILL);
            //设置textPaint,设置为抗锯齿,且是粗体
            textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG | Paint.FAKE_BOLD_TEXT_FLAG);
            textPaint.setColor(0xff000000);
            fontSize = textPaint.getTextSize();
            fontSize *= density;
            fontSize2 *= density;
            textPaint.setTextSize(fontSize);
            borderSize *= density;
        }
    
        public void start(int[] bitmapIds){
            destroy();
            for(int bitmapId : bitmapIds){
                Bitmap bitmap = BitmapFactory.decodeResource(getResources(), bitmapId);
                bitmaps.add(bitmap);
            }
            startWhenBitmapsReady();
        }
        
        private void startWhenBitmapsReady(){
            combatAircraft = new CombatAircraft(bitmaps.get(0));
            //将游戏设置为开始状态
            status = STATUS_GAME_STARTED;
            postInvalidate();
        }
        
        private void restart(){
            destroyNotRecyleBitmaps();
            startWhenBitmapsReady();
        }
    
        public void pause(){
            //将游戏设置为暂停状态
            status = STATUS_GAME_PAUSED;
        }
    
        private void resume(){
            //将游戏设置为运行状态
            status = STATUS_GAME_STARTED;
            postInvalidate();
        }
    
        private long getScore(){
            //获取游戏得分
            return score;
        }
    
        /*-------------------------------draw-------------------------------------*/
    
        @Override
        protected void onDraw(Canvas canvas) {
            //我们在每一帧都检测是否满足延迟触发单击事件的条件
            if(isSingleClick()){
                onSingleClick(touchX, touchY);
            }
    
            super.onDraw(canvas);
    
            if(status == STATUS_GAME_STARTED){
                drawGameStarted(canvas);
            }else if(status == STATUS_GAME_PAUSED){
                drawGamePaused(canvas);
            }else if(status == STATUS_GAME_OVER){
                drawGameOver(canvas);
            }
        }
    
        //绘制运行状态的游戏
        private void drawGameStarted(Canvas canvas){
    
            drawScoreAndBombs(canvas);
    
            //第一次绘制时,将战斗机移到Canvas最下方,在水平方向的中心
            if(frame == 0){
                float centerX = canvas.getWidth() / 2;
                float centerY = canvas.getHeight() - combatAircraft.getHeight() / 2;
                combatAircraft.centerTo(centerX, centerY);
            }
    
            //将spritesNeedAdded添加到sprites中
            if(spritesNeedAdded.size() > 0){
                sprites.addAll(spritesNeedAdded);
                spritesNeedAdded.clear();
            }
    
            //检查战斗机跑到子弹前面的情况
            destroyBulletsFrontOfCombatAircraft();
    
            //在绘制之前先移除掉已经被destroyed的Sprite
            removeDestroyedSprites();
    
            //每隔30帧随机添加Sprite
            if(frame % 30 == 0){
                createRandomSprites(canvas.getWidth());
            }
            frame++;
    
            //遍历sprites,绘制敌机、子弹、奖励、爆炸效果
            Iterator<Sprite> iterator = sprites.iterator();
            while (iterator.hasNext()){
                Sprite s = iterator.next();
    
                if(!s.isDestroyed()){
                    //在Sprite的draw方法内有可能会调用destroy方法
                    s.draw(canvas, paint, this);
                }
    
                //我们此处要判断Sprite在执行了draw方法后是否被destroy掉了
                if(s.isDestroyed()){
                    //如果Sprite被销毁了,那么从Sprites中将其移除
                    iterator.remove();
                }
            }
    
            if(combatAircraft != null){
                //最后绘制战斗机
                combatAircraft.draw(canvas, paint, this);
                if(combatAircraft.isDestroyed()){
                    //如果战斗机被击中销毁了,那么游戏结束
                    status = STATUS_GAME_OVER;
                }
                //通过调用postInvalidate()方法使得View持续渲染,实现动态效果
                postInvalidate();
            }
        }
    
        //绘制暂停状态的游戏
        private void drawGamePaused(Canvas canvas){
            drawScoreAndBombs(canvas);
    
            //调用Sprite的onDraw方法,而非draw方法,这样就能渲染静态的Sprite,而不让Sprite改变位置
            for(Sprite s : sprites){
                s.onDraw(canvas, paint, this);
            }
            if(combatAircraft != null){
                combatAircraft.onDraw(canvas, paint, this);
            }
    
            //绘制Dialog,显示得分
            drawScoreDialog(canvas, "continue");
    
            if(lastSingleClickTime > 0){
                postInvalidate();
            }
        }
    
        //绘制结束状态的游戏
        private void drawGameOver(Canvas canvas){
            //Game Over之后只绘制弹出窗显示最终得分
            drawScoreDialog(canvas, "restart");
    
            if(lastSingleClickTime > 0){
                postInvalidate();
            }
        }
    
        private void drawScoreDialog(Canvas canvas, String operation){
            int canvasWidth = canvas.getWidth();
            int canvasHeight = canvas.getHeight();
            //存储原始值
            float originalFontSize = textPaint.getTextSize();
            Paint.Align originalFontAlign = textPaint.getTextAlign();
            int originalColor = paint.getColor();
            Paint.Style originalStyle = paint.getStyle();
            /*
            W = 360
            w1 = 20
            w2 = 320
            buttonWidth = 140
            buttonHeight = 42
            H = 558
            h1 = 150
            h2 = 60
            h3 = 124
            h4 = 76
            */
            int w1 = (int)(20.0 / 360.0 * canvasWidth);
            int w2 = canvasWidth - 2 * w1;
            int buttonWidth = (int)(140.0 / 360.0 * canvasWidth);
    
            int h1 = (int)(150.0 / 558.0 * canvasHeight);
            int h2 = (int)(60.0 / 558.0 * canvasHeight);
            int h3 = (int)(124.0 / 558.0 * canvasHeight);
            int h4 = (int)(76.0 / 558.0 * canvasHeight);
            int buttonHeight = (int)(42.0 / 558.0 * canvasHeight);
    
            canvas.translate(w1, h1);
            //绘制背景色
            paint.setStyle(Paint.Style.FILL);
            paint.setColor(0xFFD7DDDE);
            Rect rect1 = new Rect(0, 0, w2, canvasHeight - 2 * h1);
            canvas.drawRect(rect1, paint);
            //绘制边框
            paint.setStyle(Paint.Style.STROKE);
            paint.setColor(0xFF515151);
            paint.setStrokeWidth(borderSize);
            //paint.setStrokeCap(Paint.Cap.ROUND);
            paint.setStrokeJoin(Paint.Join.ROUND);
            canvas.drawRect(rect1, paint);
            //绘制文本"飞机大战分数"
            textPaint.setTextSize(fontSize2);
            textPaint.setTextAlign(Paint.Align.CENTER);
            canvas.drawText("Final score", w2 / 2, (h2 - fontSize2) / 2 + fontSize2, textPaint);
            //绘制"飞机大战分数"下面的横线
            canvas.translate(0, h2);
            canvas.drawLine(0, 0, w2, 0, paint);
            //绘制实际的分数
            String allScore = String.valueOf(getScore());
            canvas.drawText(allScore, w2 / 2, (h3 - fontSize2) / 2 + fontSize2, textPaint);
            //绘制分数下面的横线
            canvas.translate(0, h3);
            canvas.drawLine(0, 0, w2, 0, paint);
            //绘制按钮边框
            Rect rect2 = new Rect();
            rect2.left = (w2 - buttonWidth) / 2;
            rect2.right = w2 - rect2.left;
            rect2.top = (h4 - buttonHeight) / 2;
            rect2.bottom = h4 - rect2.top;
            canvas.drawRect(rect2, paint);
            //绘制文本"继续"或"重新开始"
            canvas.translate(0, rect2.top);
            canvas.drawText(operation, w2 / 2, (buttonHeight - fontSize2) / 2 + fontSize2, textPaint);
            continueRect = new Rect(rect2);
            continueRect.left = w1 + rect2.left;
            continueRect.right = continueRect.left + buttonWidth;
            continueRect.top = h1 + h2 + h3 + rect2.top;
            continueRect.bottom = continueRect.top + buttonHeight;
    
            //重置
            textPaint.setTextSize(originalFontSize);
            textPaint.setTextAlign(originalFontAlign);
            paint.setColor(originalColor);
            paint.setStyle(originalStyle);
        }
    
        //绘制左上角的得分和左下角炸弹的数量
        private void drawScoreAndBombs(Canvas canvas){
            //绘制左上角的暂停按钮
            Bitmap pauseBitmap = status == STATUS_GAME_STARTED ? bitmaps.get(9) : bitmaps.get(10);
            RectF pauseBitmapDstRecF = getPauseBitmapDstRecF();
            float pauseLeft = pauseBitmapDstRecF.left;
            float pauseTop = pauseBitmapDstRecF.top;
            canvas.drawBitmap(pauseBitmap, pauseLeft, pauseTop, paint);
            //绘制左上角的总得分数
            float scoreLeft = pauseLeft + pauseBitmap.getWidth() + 20 * density;
            float scoreTop = fontSize + pauseTop + pauseBitmap.getHeight() / 2 - fontSize / 2;
            canvas.drawText(score + "", scoreLeft, scoreTop, textPaint);
    
            //绘制左下角
            if(combatAircraft != null && !combatAircraft.isDestroyed()){
                int bombCount = combatAircraft.getBombCount();
                if(bombCount > 0){
                    //绘制左下角的炸弹
                    Bitmap bombBitmap = bitmaps.get(11);
                    float bombTop = canvas.getHeight() - bombBitmap.getHeight();
                    canvas.drawBitmap(bombBitmap, 0, bombTop, paint);
                    //绘制左下角的炸弹数量
                    float bombCountLeft = bombBitmap.getWidth() + 10 * density;
                    float bombCountTop = fontSize + bombTop + bombBitmap.getHeight() / 2 - fontSize / 2;
                    canvas.drawText("X " + bombCount, bombCountLeft, bombCountTop, textPaint);
                }
            }
        }
    
        //检查战斗机跑到子弹前面的情况
        private void destroyBulletsFrontOfCombatAircraft(){
            if(combatAircraft != null){
                float aircraftY = combatAircraft.getY();
                List<Bullet> aliveBullets = getAliveBullets();
                for(Bullet bullet : aliveBullets){
                    //如果战斗机跑到了子弹前面,那么就销毁子弹
                    if(aircraftY <= bullet.getY()){
                        bullet.destroy();
                    }
                }
            }
        }
    
        //移除掉已经destroyed的Sprite
        private void removeDestroyedSprites(){
            Iterator<Sprite> iterator = sprites.iterator();
            while (iterator.hasNext()){
                Sprite s = iterator.next();
                if(s.isDestroyed()){
                    iterator.remove();
                }
            }
        }
    
        //生成随机的Sprite
        private void createRandomSprites(int canvasWidth){
            Sprite sprite = null;
            int speed = 2;
            //callTime表示createRandomSprites方法被调用的次数
            int callTime = Math.round(frame / 30);
            if((callTime + 1) % 25 == 0){
                //发送道具奖品
                if((callTime + 1) % 50 == 0){
                    //发送炸弹
                    sprite = new BombAward(bitmaps.get(7));
                }
                else{
                    //发送双子弹
                    sprite = new BulletAward(bitmaps.get(8));
                }
            }
            else{
                //发送敌机
                int[] nums = {0,0,0,0,0,1,0,0,1,0,0,0,0,1,1,1,1,1,1,2};
                int index = (int)Math.floor(nums.length*Math.random());
                int type = nums[index];
                if(type == 0){
                    //小敌机
                    sprite = new SmallEnemyPlane(bitmaps.get(4));
                }
                else if(type == 1){
                    //中敌机
                    sprite = new MiddleEnemyPlane(bitmaps.get(5));
                }
                else if(type == 2){
                    //大敌机
                    sprite = new BigEnemyPlane(bitmaps.get(6));
                }
                if(type != 2){
                    if(Math.random() < 0.33){
                        speed = 4;
                    }
                }
            }
    
            if(sprite != null){
                float spriteWidth = sprite.getWidth();
                float spriteHeight = sprite.getHeight();
                float x = (float)((canvasWidth - spriteWidth)*Math.random());
                float y = -spriteHeight;
                sprite.setX(x);
                sprite.setY(y);
                if(sprite instanceof AutoSprite){
                    AutoSprite autoSprite = (AutoSprite)sprite;
                    autoSprite.setSpeed(speed);
                }
                addSprite(sprite);
            }
        }
    
        /*-------------------------------touch------------------------------------*/
    
        @Override
        public boolean onTouchEvent(MotionEvent event){
            //通过调用resolveTouchType方法,得到我们想要的事件类型
            //需要注意的是resolveTouchType方法不会返回TOUCH_SINGLE_CLICK类型
            //我们会在onDraw方法每次执行的时候,都会调用isSingleClick方法检测是否触发了单击事件
            int touchType = resolveTouchType(event);
            if(status == STATUS_GAME_STARTED){
                if(touchType == TOUCH_MOVE){
                    if(combatAircraft != null){
                        combatAircraft.centerTo(touchX, touchY);
                    }
                }else if(touchType == TOUCH_DOUBLE_CLICK){
                    if(status == STATUS_GAME_STARTED){
                        if(combatAircraft != null){
                            //双击会使得战斗机使用炸弹
                            combatAircraft.bomb(this);
                        }
                    }
                }
            }else if(status == STATUS_GAME_PAUSED){
                if(lastSingleClickTime > 0){
                    postInvalidate();
                }
            }else if(status == STATUS_GAME_OVER){
                if(lastSingleClickTime > 0){
                    postInvalidate();
                }
            }
            return true;
        }
    
        //合成我们想要的事件类型
        private int resolveTouchType(MotionEvent event){
            int touchType = -1;
            int action = event.getAction();
            touchX = event.getX();
            touchY = event.getY();
            if(action == MotionEvent.ACTION_MOVE){
                long deltaTime = System.currentTimeMillis() - touchDownTime;
                if(deltaTime > singleClickDurationTime){
                    //触点移动
                    touchType = TOUCH_MOVE;
                }
            }else if(action == MotionEvent.ACTION_DOWN){
                //触点按下
                touchDownTime = System.currentTimeMillis();
            }else if(action == MotionEvent.ACTION_UP){
                //触点弹起
                touchUpTime = System.currentTimeMillis();
                //计算触点按下到触点弹起之间的时间差
                long downUpDurationTime = touchUpTime - touchDownTime;
                //如果此次触点按下和抬起之间的时间差小于一次单击事件指定的时间差,
                //那么我们就认为发生了一次单击
                if(downUpDurationTime <= singleClickDurationTime){
                    //计算这次单击距离上次单击的时间差
                    long twoClickDurationTime = touchUpTime - lastSingleClickTime;
    
                    if(twoClickDurationTime <=  doubleClickDurationTime){
                        //如果两次单击的时间差小于一次双击事件执行的时间差,
                        //那么我们就认为发生了一次双击事件
                        touchType = TOUCH_DOUBLE_CLICK;
                        //重置变量
                        lastSingleClickTime = -1;
                        touchDownTime = -1;
                        touchUpTime = -1;
                    }else{
                        //如果这次形成了单击事件,但是没有形成双击事件,那么我们暂不触发此次形成的单击事件
                        //我们应该在doubleClickDurationTime毫秒后看一下有没有再次形成第二个单击事件
                        //如果那时形成了第二个单击事件,那么我们就与此次的单击事件合成一次双击事件
                        //否则在doubleClickDurationTime毫秒后触发此次的单击事件
                        lastSingleClickTime = touchUpTime;
                    }
                }
            }
            return touchType;
        }
    
        //在onDraw方法中调用该方法,在每一帧都检查是不是发生了单击事件
        private boolean isSingleClick(){
            boolean singleClick = false;
            //我们检查一下是不是上次的单击事件在经过了doubleClickDurationTime毫秒后满足触发单击事件的条件
            if(lastSingleClickTime > 0){
                //计算当前时刻距离上次发生单击事件的时间差
                long deltaTime = System.currentTimeMillis() - lastSingleClickTime;
    
                if(deltaTime >= doubleClickDurationTime){
                    //如果时间差超过了一次双击事件所需要的时间差,
                    //那么就在此刻延迟触发之前本该发生的单击事件
                    singleClick = true;
                    //重置变量
                    lastSingleClickTime = -1;
                    touchDownTime = -1;
                    touchUpTime = -1;
                }
            }
            return singleClick;
        }
    
        private void onSingleClick(float x, float y){
            if(status == STATUS_GAME_STARTED){
                if(isClickPause(x, y)){
                    //单击了暂停按钮
                    pause();
                }
            }else if(status == STATUS_GAME_PAUSED){
                if(isClickContinueButton(x, y)){
                    //单击了“继续”按钮
                    resume();
                }
            }else if(status == STATUS_GAME_OVER){
                if(isClickRestartButton(x, y)){
                    //单击了“重新开始”按钮
                    restart();
                }
            }
        }
    
        //是否单击了左上角的暂停按钮
        private boolean isClickPause(float x, float y){
            RectF pauseRecF = getPauseBitmapDstRecF();
            return pauseRecF.contains(x, y);
        }
    
        //是否单击了暂停状态下的“继续”那妞
        private boolean isClickContinueButton(float x, float y){
            return continueRect.contains((int)x, (int)y);
        }
    
        //是否单击了GAME OVER状态下的“重新开始”按钮
        private boolean isClickRestartButton(float x, float y){
            return continueRect.contains((int)x, (int)y);
        }
    
        private RectF getPauseBitmapDstRecF(){
            Bitmap pauseBitmap = status == STATUS_GAME_STARTED ? bitmaps.get(9) : bitmaps.get(10);
            RectF recF = new RectF();
            recF.left = 15 * density;
            recF.top = 15 * density;
            recF.right = recF.left + pauseBitmap.getWidth();
            recF.bottom = recF.top + pauseBitmap.getHeight();
            return recF;
        }
    
        /*-------------------------------destroy------------------------------------*/
        
        private void destroyNotRecyleBitmaps(){
            //将游戏设置为销毁状态
            status = STATUS_GAME_DESTROYED;
    
            //重置frame
            frame = 0;
    
            //重置得分
            score = 0;
    
            //销毁战斗机
            if(combatAircraft != null){
                combatAircraft.destroy();
            }
            combatAircraft = null;
    
            //销毁敌机、子弹、奖励、爆炸
            for(Sprite s : sprites){
                s.destroy();
            }
            sprites.clear();
        }
    
        public void destroy(){
            destroyNotRecyleBitmaps();
    
            //释放Bitmap资源
            for(Bitmap bitmap : bitmaps){
                bitmap.recycle();
            }
            bitmaps.clear();
        }
    
        /*-------------------------------public methods-----------------------------------*/
    
        //向Sprites中添加Sprite
        public void addSprite(Sprite sprite){
            spritesNeedAdded.add(sprite);
        }
    
        //添加得分
        public void addScore(int value){
            score += value;
        }
    
        public int getStatus(){
            return status;
        }
    
        public float getDensity(){
            return density;
        }
    
        public Bitmap getYellowBulletBitmap(){
            return bitmaps.get(2);
        }
    
        public Bitmap getBlueBulletBitmap(){
            return bitmaps.get(3);
        }
    
        public Bitmap getExplosionBitmap(){
            return bitmaps.get(1);
        }
    
        //获取处于活动状态的敌机
        public List<EnemyPlane> getAliveEnemyPlanes(){
            List<EnemyPlane> enemyPlanes = new ArrayList<EnemyPlane>();
            for(Sprite s : sprites){
                if(!s.isDestroyed() && s instanceof EnemyPlane){
                    EnemyPlane sprite = (EnemyPlane)s;
                    enemyPlanes.add(sprite);
                }
            }
            return enemyPlanes;
        }
    
        //获得处于活动状态的炸弹奖励
        public List<BombAward> getAliveBombAwards(){
            List<BombAward> bombAwards = new ArrayList<BombAward>();
            for(Sprite s : sprites){
                if(!s.isDestroyed() && s instanceof BombAward){
                    BombAward bombAward = (BombAward)s;
                    bombAwards.add(bombAward);
                }
            }
            return bombAwards;
        }
    
        //获取处于活动状态的子弹奖励
        public List<BulletAward> getAliveBulletAwards(){
            List<BulletAward> bulletAwards = new ArrayList<BulletAward>();
            for(Sprite s : sprites){
                if(!s.isDestroyed() && s instanceof BulletAward){
                    BulletAward bulletAward = (BulletAward)s;
                    bulletAwards.add(bulletAward);
                }
            }
            return bulletAwards;
        }
    
        //获取处于活动状态的子弹
        public List<Bullet> getAliveBullets(){
            List<Bullet> bullets = new ArrayList<Bullet>();
            for(Sprite s : sprites){
                if(!s.isDestroyed() && s instanceof Bullet){
                    Bullet bullet = (Bullet)s;
                    bullets.add(bullet);
                }
            }
            return bullets;
        }
    }

    以上代码是通过Java代码去完成飞机大战中的各种逻辑。

  2. 以下是飞机大战的部分截图:

这是一个很简单的Android小游戏,有需要的朋友,可以找我私聊!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值