Android 实现扫雷小游戏实例代码

Android 实现扫雷小游戏实例

最近学习Android 应用编程,抽空做个小应用,大家熟悉的扫雷应用,练手用,

以下是实现代码:

MainActivity 类

public class MainActivity extends Activity implements OnClickListener,
    OnLongClickListener {
  // 最外层布局
  LinearLayout textviews;
  LinearLayout buttons;

  int[][] map = new int[10][10];

  // 用来隐藏所有Button
  List<Button> buttonList = new ArrayList<Button>();

  // -1

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    textviews = (LinearLayout) findViewById(R.id.textviews);
    buttons = (LinearLayout) findViewById(R.id.buttons);
    initData();
    initView();
  }

  Set<Integer> random地雷;

  private void initData() {
    // 10个地雷 显示* 数组中是-1
    // 90个 雷的边上是数字,其他是空白 0 1-8
    // 100个数字 从里面随机取走10个
    // 初始化
    for (int i = 0; i < 10; i++) {
      for (int j = 0; j < 10; j++) {
        map[i][j] = 0;
      }
    }
    // 抽取100个数 99
    random地雷 = getRandom();
    // 丢入map
    for (Integer integer : random地雷) {
      int hang = integer / 10;// 98
      int lie = integer % 10;
      // 所有的地雷用-1代替
      map[hang][lie] = -1;
    }
    // 为所有的空白地点去设置数值
    for (int i = 0; i < 10; i++) {
      for (int j = 0; j < 10; j++) {
        if (map[i][j] == -1)
          continue; // 继续下次循环
        int sum = 0;
        // 左上角
        if (i != 0 && j != 0) {// 防止下标越界
          if (map[i - 1][j - 1] == -1)
            sum++;
        }
        // 上面
        if (j != 0) {
          if (map[i][j - 1] == -1)
            sum++;
        }
        // 右上角
        if (j != 0 && i != 9) {
          if (map[i + 1][j - 1] == -1)
            sum++;
        }
        // 左边
        if (i != 0) {
          if (map[i - 1][j] == -1)
            sum++;
        }
        // 右边
        if (i != 9) {
          if (map[i + 1][j] == -1)
            sum++;
        }
        // 左下角
        if (j != 9 && i != 0) {
          if (map[i - 1][j + 1] == -1)
            sum++;
        }
        if (j != 9) {
          if (map[i][j + 1] == -1)
            sum++;
        }
        if (j != 9 && i != 9) {
          if (map[i + 1][j + 1] == -1)
            sum++;
        }
        map[i][j] = sum;
      }
    }

    // 打印看看
    for (int i = 0; i < 10; i++) {
      for (int j = 0; j < 10; j++) {
        System.out.print(map[i][j] + " ");
      }
      System.out.println();
    }
  }

  private Set<Integer> getRandom() {
    // 没有重复的
    Set<Integer> set = new HashSet<Integer>();
    while (set.size() != 10) {
      int random = (int) (Math.random() * 100);
      set.add(random);
    }
    return set;
  }

  // 创建视图
  private void initView() {
    int width = getResources().getDisplayMetrics().widthPixels / 10;
    LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(width,
        width);
    for (int i = 0; i < 10; i++) {
      // 每一条的布局
      LinearLayout tvs = new LinearLayout(this);
      tvs.setOrientation(LinearLayout.HORIZONTAL);
      LinearLayout btns = new LinearLayout(this);
      btns.setOrientation(LinearLayout.HORIZONTAL);
      // 添加内层的100个按钮和文本
      for (int j = 0; j < 10; j++) {
        // 底层的TextView
        TextView tv = new TextView(this);
        tv.setBackgroundResource(R.drawable.textview_bg);
        tv.setLayoutParams(params);
        tv.setGravity(Gravity.CENTER);
        if (map[i][j] == -1)
          tv.setText("*");
        else if (map[i][j] != 0)
          tv.setText(map[i][j] + "");
        tvs.addView(tv);
        // 底层的Button
        Button btn = new Button(this);
        btn.setBackgroundResource(R.drawable.button);
        btn.setLayoutParams(params);
        btn.setTag(i * 10 + j);
        btn.setOnClickListener(this);
        btn.setOnLongClickListener(this);
        buttonList.add(btn);
        btns.addView(btn);
      }
      textviews.addView(tvs);
      buttons.addView(btns);
    }
  }

  @Override
  public void onClick(View v) {
    int id = (Integer) v.getTag();
    int hang = id / 10;
    int lie = id % 10;
    // 隐藏按钮,显示底下的数据
    v.setVisibility(View.INVISIBLE);
    isOver(hang, lie);
    // 判断点击的是否是一个数字
    if (map[hang][lie] == 0) {
      // 开始递归
      显示周围所有的空白(hang, lie);
    }
    if (isWin()) {
      Toast.makeText(this, "游戏胜利", Toast.LENGTH_SHORT).show();
    }
  }

  // 显示周围所有的button
  public void 显示周围所有的空白(int i, int j) {
    // 检测周围的元素,如果为0 继续调用自身,并且显示
    // 不是,就显示button
    // 从左上角开始
    // 左上角
    // 先显示自己
    buttonList.get(i * 10 + j).setVisibility(View.INVISIBLE);
    if (i != 0 && j != 0) {// 防止下标越界
      if (map[i - 1][j - 1] == 0) {
        if (判断是否需要递归(i - 1, j - 1))
          显示周围所有的空白(i - 1, j - 1);
      } else {
        隐藏button(i - 1, j - 1);
      }
    }
    // 上面
    if (j != 0) {
      if (map[i][j - 1] == 0) {
        if (判断是否需要递归(i, j - 1))
          显示周围所有的空白(i, j - 1);
      } else {
        隐藏button(i, j - 1);
      }
    }
    // 右上角
    if (j != 0 && i != 9) {
      if (map[i + 1][j - 1] == 0) {
        if (判断是否需要递归(i + 1, j - 1))
          显示周围所有的空白(i + 1, j - 1);
      } else {
        隐藏button(i + 1, j - 1);
      }
    }
    // 左边
    if (i != 0) {
      if (map[i - 1][j] == 0) {
        if (判断是否需要递归(i - 1, j))
          显示周围所有的空白(i - 1, j);
      } else {
        隐藏button(i - 1, j);
      }
    }
    // 右边
    if (i != 9) {
      if (map[i + 1][j] == 0) {
        if (判断是否需要递归(i + 1, j))
          显示周围所有的空白(i + 1, j);
      } else {
        隐藏button(i + 1, j);
      }
    }
    // 左下角
    if (j != 9 && i != 0) {
      if (map[i - 1][j + 1] == 0) {
        if (判断是否需要递归(i - 1, j + 1))
          显示周围所有的空白(i - 1, j + 1);
      } else {
        隐藏button(i - 1, j + 1);
      }
    }
    if (j != 9) {
      if (map[i][j + 1] == 0) {
        if (判断是否需要递归(i, j + 1))
          显示周围所有的空白(i, j + 1);
      } else {
        隐藏button(i, j + 1);
      }
    }
    if (j != 9 && i != 9) {
      if (map[i + 1][j + 1] == 0) {
        if (判断是否需要递归(i + 1, j + 1))
          显示周围所有的空白(i + 1, j + 1);
      } else {
        隐藏button(i + 1, j + 1);
      }
    }

  }

  private void 隐藏button(int i, int j) {
    int 位置 = i * 10 + j;
    buttonList.get(位置).setVisibility(View.INVISIBLE);
  }

  boolean 判断是否需要递归(int hang, int lie) {
    // 判断是否是现实的
    int 位置 = hang * 10 + lie;
    if (buttonList.get(位置).getVisibility() == View.INVISIBLE)
      return false;
    else
      return true;
  }

  private boolean isOver(int hang, int lie) {
    // OVER
    if (map[hang][lie] == -1) {
      Toast.makeText(this, "GameOver", Toast.LENGTH_SHORT).show();
      for (int i = 0; i < buttonList.size(); i++) {
        buttonList.get(i).setVisibility(View.INVISIBLE);
      }
      return true;
    }
    return false;
  }

  // 最多10个旗子
  List<Integer> 旗子 = new ArrayList<Integer>();

  @Override
  public boolean onLongClick(View v) {
    // 插旗子
    // 1. 有了旗子 就取消
    // 2. 没有就插旗
    Button btn = (Button) v;
    int tag = (Integer) v.getTag();
    if (旗子.contains(tag)) {
      // 如果使用drawableTop 对应的java代码的方法
      // setCompoundDrawablesWithIntrinsicBounds
      btn.setText("");
      // int ArrayList.remove(int);//下标
      旗子.remove((Integer) tag);
    } else {
      // 没有插旗就需要插旗,判断数量是否超过了上限
      if (旗子.size() != 10) {
        旗子.add(tag);
        btn.setText("∉ " + 旗子.size());
        btn.setTextColor(Color.RED);
      } else {
        Toast.makeText(this, "没有旗子了", Toast.LENGTH_SHORT).show();
      }

    }
    // 消耗事件,
    return true;
  }

  // 是否胜利
  public boolean isWin() {
    int sum = 0;
    for (int i = 0; i < buttonList.size(); i++) {
      if (buttonList.get(i).getVisibility() == View.INVISIBLE)
        sum++;
    }
    if (sum == 90)
      return true;
    return false;
  }
}

xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent"
  android:gravity="center"
  android:orientation="vertical" >
  <FrameLayout
    android:id="@+id/framelayout"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" >

    <LinearLayout
      android:id="@+id/textviews"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:orientation="vertical" />

    <LinearLayout
      android:id="@+id/buttons"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:orientation="vertical" >
    </LinearLayout>
  </FrameLayout>

</LinearLayout>

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android按钮单击事件的四种常用写法总结

    很多学习Android程序设计的人都会发现每个人对代码的写法都有不同的偏好,比较明显的就是对控件响应事件的写法的不同.因此本文就把这些写法总结一下,比较下各种写法的优劣,希望对大家灵活地选择编码方式可以有一定的参考借鉴价值. xml文件代码如下: <Button android:id="@+id/button1" android:layout_width="wrap_content" android:layout_height="wrap_conte

  • Android基础之使用Fragment控制切换多个页面

    今天讲解一下Fragment的控制,主要是切换View和页面替换等操作.还有就是如何获取Fragment的管理对象,以及与Activity的通信方式.1.管理Fragment要在activity中管理fragment,需要使用FragmentManager. 通过调用activity的getFragmentManager()取得它的实例. •可以通过FragmentManager做一些事情, 包括: 使用findFragmentById()(用于在activity layout中提供一个UI的f

  • Android的Activity跳转动画各种效果整理

    大家使用Android的原生UI都知道,Android的Activity跳转就是很生硬的切换界面.其实Android的Activity跳转可以设置各种动画.下面给大家看看效果:  实现非常简单,用overridePendingtransition(int inId, int outId)即可实现.inId是下一界面进入效果的xml文件的id,outId是当前界面退出效果的xml文件id. 效果是用xml文件写的,首先要在res文件夹下建立anim文件夹,然后把动画效果xml文件放到里面去. 下面

  • Android开发之SQLite的使用方法

    前言 SQLite是一种轻量级的小型数据库,虽然比较小,但是功能相对比较完善,一些常见的数据库基本功能也具有,在现在的嵌入式系统中使用该数据库的比较多,因为它占用系统资源很少.Android系统中也不例外,也是采用SQLite,本节中就学习下在andorid中怎样使用该数据库来存放数据,并且对SQLite完成简单的新建,更新,查询,删除等操作. 实验说明: Android中使用SQLite数据库时,需要用adb来辅助调试,如果想在windows下的cmd命令行中使用adb,必须先配置环境变量,我

  • 解析android中ProgressBar的用法

    范例说明Android的Widget,有许多是为了与User交互而特别设计的,但也有部分是作为程序提示.显示程序运行状态的Widget.现在介绍的范例,与前一章介绍过的ProgressDialog对话框的应用目的相似,但由于前章介绍的ProgressDialog是继承自Android.app.ProgressDialog所设计的互动对话窗口,在应用时,必须新建ProgressDialog对象,在运行时会弹出"对话框"作为提醒,此时应用程序后台失去焦点,直到进程结束后,才会将控制权交给应

  • android TextView设置中文字体加粗实现方法

    英文设置加粗可以在xml里面设置: 复制代码 代码如下: <SPAN style="FONT-SIZE: 18px">android:textStyle="bold"</SPAN> 英文还可以直接在String文件里面直接这样填写: 复制代码 代码如下: <string name="styled_text">Plain, <b>bold</b>, <i>italic</

  • android Handler详细使用方法实例

    开发环境为android4.1.Handler使用例1这个例子是最简单的介绍handler使用的,是将handler绑定到它所建立的线程中.本次实验完成的功能是:单击Start按钮,程序会开始启动线程,并且线程程序完成后延时1s会继续启动该线程,每次线程的run函数中完成对界面输出nUpdateThread...文字,不停的运行下去,当单击End按钮时,该线程就会停止,如果继续单击Start,则文字又开始输出了. 软件界面如下: 实验主要部分代码和注释: MainActivity.java: 复

  • android PopupWindow 和 Activity弹出窗口实现方式

    本人小菜一个.目前只见过两种弹出框的实现方式,第一种是最常见的PopupWindow,第二种也就是Activity的方式是前几天才见识过.感觉很霸气哦.没想到,activity也可以做伪窗口. 先贴上最常见的方法,主要讲activity的方法. 一.弹出PopupWindow 复制代码 代码如下: /** * 弹出menu菜单 */ public void menu_press(){ if(!menu_display){ //获取LayoutInflater实例 inflater = (Layo

  • android listview优化几种写法详细介绍

    这篇文章只是总结下getView里面优化视图的几种写法,就像孔乙己写茴香豆的茴字的几种写法一样,高手勿喷,勿笑,只是拿出来分享,有错误的地方欢迎大家指正,谢谢. listview Aviewthatshowsitemsinaverticallyscrollinglist. 一个显示一个垂直的滚动子项的列表视图在android开发中,使用listview的地方很多,用它来展现数据,成一个垂直的视图.使用listview是一个标准的适配器模式,用数据--,界面--xml以及适配器--adapter,

  • Android中实现可滑动的Tab的3种方式

    1. 第一种,使用 TabHost + ViewPager 实现该方法会有一个Bug,当设置tabHost.setCurrentTab()为0时,ViewPager不显示(准确的说是加载),只有点击其他任意一个tab后才会加载. 有解的同学吼一声~~~~~~~ Activity: 复制代码 代码如下: package com.swordy.demo.android.fragment; import java.util.Random; import android.os.Bundle;import

  • Android应用开发SharedPreferences存储数据的使用方法

    SharedPreferences是Android中最容易理解的数据存储技术,实际上SharedPreferences处理的就是一个key-value(键值对).SharedPreferences常用来存储一些轻量级的数据. 复制代码 代码如下: //实例化SharedPreferences对象(第一步) SharedPreferences mySharedPreferences= getSharedPreferences("test", Activity.MODE_PRIVATE);

  • Android SQLite数据库增删改查操作的使用详解

    一.使用嵌入式关系型SQLite数据库存储数据 在Android平台上,集成了一个嵌入式关系型数据库--SQLite,SQLite3支持NULL.INTEGER.REAL(浮点数字). TEXT(字符串文本)和BLOB(二进制对象)数据类型,虽然它支持的类型只有五种,但实际上sqlite3也接受varchar(n). char(n).decimal(p,s) 等数据类型,只不过在运算或保存时会转成对应的五种数据类型. SQLite最大的特点是你可以把各种类型的数据保存到任何字段中,而不用关心字段

  • Android中判断网络连接是否可用及监控网络状态

    获取网络信息需要在AndroidManifest.xml文件中加入相应的权限. <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> 1)判断是否有网络连接 复制代码 代码如下: public boolean isNetworkConnected(Context context) { if (context != null) { ConnectivityManager mConn

随机推荐