c语言对对碰游戏源码

简介: c语言对对碰游戏源码
//画素材的x和y都是反的,因为x表示行,但是画出来x表示列,y同
// 来自公众号:c语言与cpp编程
#include <graphics.h>
#include <fstream>
#include <strstream>
#include <iomanip>
#include <cstdlib>
#include <ctime>
#pragma comment(lib, "Winmm.lib")
using namespace std;
/*******************************定义枚举类型*****************************/
enum color{blank, red, yellow, blue, green, white, orange, purple, shizijia, zhadan, qicai, alarm};
/*******************************定义全局变量*****************************/
const int MaxT = 12;  // 时间上限
const int T = 10 * MaxT;// 时间速度
const int V = 300;    // 停留时间
clock_t start, now;   // 控制时间
color gem[9][8];    // 地图
            // IMAGE对象
IMAGE img[12], music_img[2], exit_img, jindutiao;
int Score, Time;    // 成绩 时间
bool Flag, Music = true;// 是否加载 音乐
/**********************************函数声明*****************************/
void load(void);  // 加载IMAGE对象
void Blank(void); // 清空
bool soso(void);  // 搜索空格
bool baidu(void); // 搜索解法
void New(void);   // 更新
void print(void); // 输出
bool judge(void); // 判断是否可以消除
void fall(void);  // 下落
void draw(void);  // 消除
bool quit(void);  // 是否重新开始
void play(void);  // 游戏过程
/**********************************定义函数*****************************/
void load(void)
{
  char c[20];
  int i;
  // 加载IMAGE对象
  for (i = 0; i < 12; i++)
  {
    ostrstream strout(c, 50);
    strout <<"图片\\" <<i <<".jpg" <<ends;
    loadimage(&img[i], c);
  }
  loadimage(&music_img[0], "图片\\音乐关.jpg");
  loadimage(&music_img[1], "图片\\音乐开.jpg");
  loadimage(&exit_img, "图片\\退出.jpg");
  loadimage(&jindutiao, "图片\\进度条.jpg");
  // 加载音乐
  mciSendString("open 背景音乐.mp3 alias mymusic", NULL, 0, NULL);
  // 随机种子
  srand(unsigned(time(NULL)));
  // 打开文件
  ifstream fin("存档.dat");
  if (!fin)
    throw -1; // 如果打开失败则抛出异常
  // 读存档
  fin >>Flag >>Music;
  if (Flag)
  {
    HWND wnd = GetHWnd();
    SetWindowText(wnd, "对对碰");
    if (MessageBox(wnd, "是否继续上次游戏?", "游戏开始", MB_YESNO | MB_ICONQUESTION) == IDYES)
    {
      fin >>Score >>Time;
      for (i = 0; i < 9; i++)
        for (int j = 0; j < 8; j++)
        {
          int t;
          fin >>t;
          gem[i][j] = color(t);
        }
    }
    else
      Flag = false;
  }
  fin.close();
}
void Blank(void)
{
  for (int i = 1; i < 9; i++)
    for (int j = 0; j < 8; j++)
      gem[i][j] = blank;
  print();
  fall();
}
bool soso(void)
{
  for (int i = 1; i < 9; i++)
    for (int j = 0; j < 8; j++)
      if (gem[i][j] == blank)
        return true;
  return false;
}
bool baidu(void)
{
  int i, j;
  color t;
  bool flag = false;
  // 如果有一个道具则返回真
  for (i = 1; i < 9; i++)
    for (j = 0; j < 8; j++)
      if (gem[i][j] >= shizijia)
        return true;
  // 搜索解法
  for (i = 1; i < 9; i++)
    for (j = 0; j < 7; j++)
    {
      t = gem[i][j];
      gem[i][j] = gem[i][j + 1];
      gem[i][j + 1] = t;
      if (judge())
        flag = true;
      t = gem[i][j];
      gem[i][j] = gem[i][j + 1];
      gem[i][j + 1] = t;
      if (flag)
        return true;
    }
  for (i = 1; i < 7; i++)
    for (j = 0; j < 8; j++)
    {
      t = gem[i][j];
      gem[i][j] = gem[i + 1][j];
      gem[i + 1][j] = t;
      if (judge())
        flag = true;
      t = gem[i][j];
      gem[i][j] = gem[i + 1][j];
      gem[i + 1][j] = t;
      if (flag)
        return true;
    }
  return false;
}
void New(void)
{
  static int old_score = -1, old_time = T;
  bool flag = false;
  if (Score != old_score) // 更新分数
  {
    if (Score > 999999999)
    {
      if (MessageBox(GetHWnd(), "恭喜你达到了最高分!\n是否重新开始?", "游戏结束", MB_YESNO | MB_ICONQUESTION) == IDYES)
        Score = 0;
      else
        exit(0);
    }
    char s[15];
    ostrstream strout(s, 15);
    strout <<'$' <<setiosflags(ios::right) <<setw(9) <<Score <<ends;
    setfont(30, 0, "微软雅黑");
    setcolor(RGB(255, 128, 0));
    outtextxy(480, 60, s);
    old_score = Score;
    flag = true;
  }
  if (Time != old_time) // 更新时间
  {
    if (Time > old_time)
      putimage(540, 140, &jindutiao);
    setcolor(BLACK);
    for (int i = 0; i < T - Time; i++)
    {
      line(540, 140 + 2 * i, 580, 140 + 2 * i);
      line(540, 140 + 2 * i + 1, 580, 140 + 2 * i + 1);
    }
    old_time = Time;
    flag = true;
  }
  if (flag)       // 写存档
  {
    ofstream fout("存档.dat");
    fout <<Flag   <<'\t'
       <<Music  <<'\t'
       <<Score  <<'\t'
       <<Time   <<endl;
    for (int i = 0; i < 9; i++)
    {
      for (int j = 0; j < 8; j++)
        fout <<int(gem[i][j]) <<'\t';
      fout <<endl;
    }
    fout.close();
  }
}
void print(void)
{
  int i, j;
  for (i = 1; i < 9; i++)
    for (j = 0; j < 8; j++)
      putimage(60 * j, 60 * (i - 1), &img[int(gem[i][j])]);
  New();
}
bool judge(void)
{
  int i, j;
  for (i = 1; i < 9; i++)
    for (j = 0; j < 6; j++)
      if (gem[i][j] == gem[i][j + 1] && gem[i][j] == gem[i][j + 2])
        return true;
  for (i = 1; i < 7; i++)
    for (j = 0; j < 8; j++)
      if (gem[i][j] == gem[i + 1][j] && gem[i][j] == gem[i + 2][j])
        return true;
  return false;
}
void fall(void)
{
  {
    int i, j, a[8] = {0};
    bool sign = false;
    for (j = 0; j < 8; j++)
      for (i = 8; i >= 1; i--)
        if (gem[i][j] == blank)
        {
          a[j] = i;
          sign = true;
          break;
        }
    if (sign == false)
      return;
    IMAGE im[8];
    for (j = 0; j < 8; j++)
      if (a[j] > 1)
        getimage(&im[j], 60 * j, 0, 60, 60 * (a[j] - 1));
    for (i = 0; i < 60 ; i += 2)
      for (j = 0; j < 8; j++)
        if (a[j] > 0)
        {
          if (a[j] > 1)
            putimage(60 * j, i + 2, &im[j]);
          putimage(60 * j, i + 2 - 60, &img[int(gem[0][j])]);
          Sleep(1);
        }
    for (j = 0; j < 8; j++)
      if (a[j] > 0)
      {
        for (i = a[j]; i > 0; i--)
          gem[i][j] = gem[i - 1][j];
        if (rand() % 50 == 0)
          gem[0][j] = color(rand() % 4 + 8);
        else
          gem[0][j] = color(rand() % 7 + 1);
      }
  } // 加对大括号使递归时撤销内存空间
  if (soso())
    fall();
  if (judge())
    draw();
}
void draw(void)
{
  {
    int i, j;
    bool a[9][8] = {false};
    for (i = 1; i < 9; i++)
      for (j = 0; j < 6; j++)
        if (gem[i][j] == gem[i][j + 1] && gem[i][j] == gem[i][j + 2])
          a[i][j] = a[i][j + 1] = a[i][j + 2] = true;
    for (i = 1; i < 7; i++)
      for (j = 0; j < 8; j++)
        if (gem[i][j] == gem[i + 1][j] && gem[i][j] == gem[i + 2][j])
          a[i][j] = a[i + 1][j] = a[i + 2][j] = true;
    for (i = 1; i < 9; i++)
      for (j = 0; j < 8; j++)
        if (a[i][j])
        {
          gem[i][j] = blank;
          Score += 10;
        }
    Sleep(V);
    Time += T / MaxT;
    if (Time > T)
      Time = T;
    print();
  } // 加对大括号使递归时撤销内存空间
  fall();
}
bool quit(void)
{
  char str[50];
  ostrstream strout(str, 50);
  strout <<"得分:" <<Score <<"\n重新开始吗?" <<ends;
  if (MessageBox(GetHWnd(), str, "游戏结束", MB_YESNO | MB_ICONQUESTION) == IDYES)
    return true;
  return false;
}
void play(void)
{
  MOUSEMSG m;
  int i, j, x, y, x1, y1;
  char fx;
  color t;
  bool sign;
  if (Flag == false)
  {
    do
    {
      for (i = 0; i < 9; i++)
        for (j = 0; j < 8; j++)
          gem[i][j] = color(rand() % 7 + 1);
    }while (judge() || !baidu());
    Score = 0;
    Time = T;
  }
  setbkcolor(BLACK);
  cleardevice();
  setfont(30, 0, "微软雅黑");
  setcolor(RGB(255, 128, 0));
  outtextxy(480, 60, "$       0");
  putimage(480, 400, &music_img[int(Music)]);
  putimage(560, 400, &exit_img);
  putimage(540, 140, &jindutiao);
  print();
  now = start = clock();
  Flag = true;
  do
  {
    if (soso())
      fall();
    fx = 0;
    sign = true;
    while (true)
    {
      if (MouseHit())
      {
        m = GetMouseMsg();
        x1 = m.x;
        y1 = m.y;
        if (m.uMsg == WM_LBUTTONDOWN)
        {
          if (x1 < 480)
          {
            x = y1 / 60 + 1;
            y = x1 / 60;
            switch (gem[x][y])
            {
            case shizijia:
              for (i = 1; i < 9; i++)
              {
                gem[i][y] = blank;
                Score += 10;
              }
              for (j = 0; j < 8; j++)
              {
                gem[x][j] = blank;
                Score += 10;
              }
              Score -= 20;
              Time += T / MaxT;
              if (Time > T)
                Time = T;
              print();
              Sleep(V);
              fall();
              if (!baidu())
                Blank();
              break;
            case zhadan:
              if (x > 1)
              {
                gem[x - 1][y] = blank;
                Score += 10;
                if (y > 0)
                {
                  gem[x - 1][y - 1] =blank;
                  Score += 10;
                }
              }
              if (y > 0)
              {
                gem[x][y - 1] = blank;
                if (x < 8)
                {
                  gem[x + 1][y - 1] =blank;
                  Score += 10;
                }
              }
              if (x < 8)
              {
                gem[x + 1][y] = blank;
                if (y < 7)
                {
                  gem[x + 1][y + 1] =blank;
                  Score += 10;
                }
              }
              if (y < 7)
              {
                gem[x][y + 1] = blank;
                if (x > 1)
                {
                  gem[x - 1][y + 1] =blank;
                  Score += 10;
                }             
              }
              gem[x][y] = blank;
              Time += T / MaxT;
              if (Time > T)
                Time = T;
              print();
              Sleep(V);
              fall();
              if (!baidu())
                Blank();
              break;
            case qicai:
              t = color(rand() % 7 + 1);
              putimage(60 * y, 60 * (x - 1), &img[int(t)]);
              Sleep(V);
              for (i = 1; i < 9; i++)
                for (j = 0; j < 8; j++)
                  if (gem[i][j] == t)
                  {
                    gem[i][j] = blank;
                    putimage(60 * j, 60 * (i - 1), &img[0]);
                    Score += 10;
                  }
              gem[x][y] = blank;
              Time += T / MaxT;
              if (Time > T)
                Time = T;
              print();
              fall();
              if (!baidu())
                Blank();
              break;
            case alarm:
              Time = T;
              gem[x][y] = blank;
              print();
              fall();
              if (!baidu())
                Blank();
              break;
            }
            break;
          }
          else if (x1 > 480 && x1 < 560 && y1 > 400 && y1 < 480)
          {
            if (Music)
              mciSendString("stop mymusic", NULL, 0, NULL);
            else
              mciSendString("play mymusic from 0 repeat", NULL, 0, NULL);
            Music = !Music;
            putimage(480, 400, &music_img[int(Music)]);
          }
          else if (x1 > 560 && x1 < 640 && y1 > 400 && y1 < 480)
            exit(0);
        }
      }
      now = clock();
      if (now - start >= CLOCKS_PER_SEC * MaxT / T)
      {
        start = now;
        Time--;
        New();
        if (Time <= 0)
          return;
      }
    }
    while (m.mkLButton && m.y < 480)
    {
      m = GetMouseMsg();
      x = m.x;
      y = m.y;
      if (x - x1 > 30 || x1 - x > 30 || y - y1 > 30 || y1 - y > 30)
      {
        sign = false;
        break;
      }
      now = clock();
      if (now - start >= CLOCKS_PER_SEC * MaxT / T)
      {
        start = now;
        Time--;
        New();
        if (Time <= 0)
          return;
      }
    }
    if(sign)
    {
      putimage(x1 - x1 % 60, y1 - y1 % 60, &img[int(gem[y1 / 60 + 1][x1 / 60])], SRCPAINT);
      continue;
    }
    if (y1 - y > 30 && y1 / 60 > 0)
      fx = 'u';
    if (y - y1 > 30 && y1 / 60 < 7)
      fx = 'd';
    if (x1 - x > 30 && x1 / 60 > 0)
      fx = 'l';
    if (x - x1 > 30 && x1 / 60 < 7)
      fx = 'r';
    x = y1 / 60 + 1;
    y = x1 / 60;
    x1 = y1 = 0;
    switch (fx)
    {
      case 'u':x1 = -1; break;
      case 'd':x1 = 1;  break;
      case 'l':y1 = -1; break;
      case 'r':y1 = 1;  break;
      case '\0':continue;
    }
    for (i = 0; i < 60;)
    {
      putimage(60 * y, 60 * (x - 1), &img[0]);
      putimage(60 * y, 60 * (x - 1), &img[0]);
      i += 1;
      putimage(60 * (y + y1) - i * y1, 60 * (x - 1 + x1) - i * x1, &img[int(gem[x + x1][y + y1])]);
      putimage(60 * y + i * y1, 60 * (x - 1) + i * x1, &img[int(gem[x][y])]);
      Sleep(2);
    }
    t = gem[x][y];
    gem[x][y] = gem[x + x1][y + y1];
    gem[x + x1][y + y1] = t;
    if (judge())
    {
      draw();
      if (!baidu())
        Blank();
    }
    else
    {
      for (i = 0; i < 60;)
      {
        putimage(60 * y, 60 * (x - 1), &img[0]);
        putimage(60 * y, 60 * (x - 1), &img[0]);
        i += 1;
        putimage(60 * (y + y1) - i * y1, 60 * (x - 1 + x1) - i * x1, &img[int(gem[x + x1][y + y1])]);
        putimage(60 * y + i * y1, 60 * (x - 1) + i * x1, &img[int(gem[x][y])]);
        Sleep(2);
      }
      t = gem[x][y];
      gem[x][y] = gem[x + x1][y + y1];
      gem[x + x1][y + y1] = t;
    }
    now = clock();
    if (now - start >= CLOCKS_PER_SEC * MaxT / T)
    {
      start = now;
      Time--;
      New();
      if (Time <= 0)
        return;
    }
  }while (true);
}
/***********************************主函数******************************/
int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
  try
  {
    // 加载素材
    load();
  }
  catch (int)
  {
    ofstream fout("存档.dat");
    fout <<false <<'\t' <<true <<endl;
  }
  // 打开界面
  initgraph(640, 480);
  // 游戏过程
  do
  {
    if (Music)
      mciSendString("play mymusic from 0 repeat", NULL, 0, NULL);
    play();
    if (Music)
      mciSendString("stop mymusic", NULL, 0, NULL);
    Flag = false;
  }while (quit());
  // 关闭游戏
  ofstream fout("存档.dat");
  fout <<false <<'\t' <<Music <<endl;
  fout.close();
  mciSendString("close mymusic", NULL, 0, NULL);
  closegraph();
  return 0;
}
/***********************************THE END************************************/
相关文章
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
261 9
|
2月前
|
C语言
C语言之斗地主游戏
该代码实现了一个简单的斗地主游戏,包括头文件引入、宏定义、颜色枚举、卡牌类、卡牌类型类、卡牌组合类、玩家类、游戏主类以及辅助函数等,涵盖了从牌的生成、分配、玩家操作到游戏流程控制的完整逻辑。
92 8
|
3月前
|
C语言
扫雷游戏(用C语言实现)
扫雷游戏(用C语言实现)
142 0
|
2月前
|
存储 算法 C语言
用C语言开发游戏的实践过程,包括选择游戏类型、设计游戏框架、实现图形界面、游戏逻辑、调整游戏难度、添加音效音乐、性能优化、测试调试等内容
本文探讨了用C语言开发游戏的实践过程,包括选择游戏类型、设计游戏框架、实现图形界面、游戏逻辑、调整游戏难度、添加音效音乐、性能优化、测试调试等内容,旨在为开发者提供全面的指导和灵感。
55 2
|
2月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
112 16
|
2月前
|
搜索推荐 算法 C语言
【排序算法】八大排序(下)(c语言实现)(附源码)
本文继续学习并实现了八大排序算法中的后四种:堆排序、快速排序、归并排序和计数排序。详细介绍了每种排序算法的原理、步骤和代码实现,并通过测试数据展示了它们的性能表现。堆排序利用堆的特性进行排序,快速排序通过递归和多种划分方法实现高效排序,归并排序通过分治法将问题分解后再合并,计数排序则通过统计每个元素的出现次数实现非比较排序。最后,文章还对比了这些排序算法在处理一百万个整形数据时的运行时间,帮助读者了解不同算法的优劣。
161 7
|
2月前
|
搜索推荐 算法 C语言
【排序算法】八大排序(上)(c语言实现)(附源码)
本文介绍了四种常见的排序算法:冒泡排序、选择排序、插入排序和希尔排序。通过具体的代码实现和测试数据,详细解释了每种算法的工作原理和性能特点。冒泡排序通过不断交换相邻元素来排序,选择排序通过选择最小元素进行交换,插入排序通过逐步插入元素到已排序部分,而希尔排序则是插入排序的改进版,通过预排序使数据更接近有序,从而提高效率。文章最后总结了这四种算法的空间和时间复杂度,以及它们的稳定性。
133 8
|
2月前
|
C语言
【数据结构】二叉树(c语言)(附源码)
本文介绍了如何使用链式结构实现二叉树的基本功能,包括前序、中序、后序和层序遍历,统计节点个数和树的高度,查找节点,判断是否为完全二叉树,以及销毁二叉树。通过手动创建一棵二叉树,详细讲解了每个功能的实现方法和代码示例,帮助读者深入理解递归和数据结构的应用。
152 8
|
2月前
|
C语言 Windows
C语言课设项目之2048游戏源码
C语言课设项目之2048游戏源码,可作为课程设计项目参考,代码有详细的注释,另外编译可运行文件也已经打包,windows电脑双击即可运行效果
43 1
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
126 4