首页 > 其他分享 >SDL贪吃蛇代码

SDL贪吃蛇代码

时间:2024-09-02 22:22:40浏览次数:13  
标签:body direction food 代码 贪吃蛇 snake SDL NULL

#include <stdio.h>
#include <SDL.h>
#include <stdlib.h>
#include <SDL_image.h>
#undef main

// 定义屏幕宽度和高度及蛇的块大小
#define SCREEN_WIDTH 800
#define SCREEN_HEIGHT 600
#define SNAKE_BLOCK 20

// SDL窗口和渲染器全局变量
SDL_Window* gWindow = NULL;
SDL_Renderer* gRenderer = NULL;

// 定义一个点结构体,用于表示蛇的身体部分和食物的位置
typedef struct {
    int x;
    int y;
} Point;

// 定义一个蛇结构体,包含蛇的身体部分数组、长度和方向
typedef struct {
    Point* body;     // 蛇的身体数组
    int length;      // 蛇的长度
    int direction;   // 蛇的方向(0=右, 1=下, 2=左, 3=上)
} Snake;

// 游戏中的蛇
Snake* gSnake;
// 食物的位置
Point gFood;

// 初始化蛇的函数
void init_snake(Snake* snake) {
    snake->length = 3;          // 蛇初始长度为3
    snake->direction = 0;       // 初始方向向右
    snake->body = (Point*)malloc(snake->length * sizeof(Point)); // 分配内存给蛇的身体数组
    for (int i = 0; i < snake->length; i++) {
        snake->body[i].x = 200 - i * SNAKE_BLOCK; // 设置蛇的初始位置
        snake->body[i].y = 200;
    }
}

// 初始化食物位置的函数
void init_food(Point* food) {
    food->x = rand() % (SCREEN_WIDTH / SNAKE_BLOCK) * SNAKE_BLOCK; // 随机生成食物的X坐标
    food->y = rand() % (SCREEN_HEIGHT / SNAKE_BLOCK) * SNAKE_BLOCK; // 随机生成食物的Y坐标
}

// 绘制蛇的函数
void draw_snake(Snake* snake, SDL_Renderer* renderer) {
    // 加载蛇的头、身体和尾巴图像
    SDL_Texture* snakeHeadTexture = IMG_LoadTexture(renderer, "..//image//head2.png");
    SDL_Texture* snakeBodyTexture = IMG_LoadTexture(renderer, "..//image//body2.png");
    SDL_Texture* snakeTailTexture = IMG_LoadTexture(renderer, "..//image//tail2.png");

    if (snakeHeadTexture == NULL || snakeBodyTexture == NULL || snakeTailTexture == NULL) {
        printf("无法加载图像: %s\n", SDL_GetError()); // 错误处理
        return;
    }

    SDL_Rect block;
    for (int i = 0; i < snake->length; i++) {
        block.x = snake->body[i].x;
        block.y = snake->body[i].y;
        block.w = SNAKE_BLOCK;
        block.h = SNAKE_BLOCK;

        // 绘制蛇的头部,并根据方向旋转图像
        if (i == 0) {
            SDL_RenderCopyEx(renderer, snakeHeadTexture, NULL, &block,
                snake->direction * 90, NULL, SDL_FLIP_NONE);
        }
        // 绘制蛇的尾部,并根据前后方向调整角度
        else if (i == snake->length - 1) {
            int prevDirection = (snake->direction + 2) % 4; // 上一个方向
            int angle = prevDirection * 90;

            // 根据前后的方向调整角度
            if (prevDirection == 0 && snake->direction == 3) { // 右转上
                angle = 270;
            }
            else if (prevDirection == 3 && snake->direction == 0) { // 上转右
                angle = 90;
            }
            else if (prevDirection == 1 && snake->direction == 2) { // 下转左
                angle = 270;
            }
            else if (prevDirection == 2 && snake->direction == 1) { // 左转下
                angle = 90;
            }

            SDL_RenderCopyEx(renderer, snakeTailTexture, NULL, &block, angle, NULL, SDL_FLIP_NONE);
        }
        // 绘制蛇的身体部分
        else {
            SDL_RenderCopy(renderer, snakeBodyTexture, NULL, &block);
        }
    }

    // 释放图像纹理
    SDL_DestroyTexture(snakeHeadTexture);
    SDL_DestroyTexture(snakeBodyTexture);
    SDL_DestroyTexture(snakeTailTexture);
}

// 绘制食物的函数
void draw_food(Point* food, SDL_Renderer* renderer) {
    SDL_Rect block;
    block.x = food->x;
    block.y = food->y;
    block.w = SNAKE_BLOCK;
    block.h = SNAKE_BLOCK;
    SDL_SetRenderDrawColor(renderer, 255, 0, 0, 255); // 设置颜色为红色
    SDL_RenderFillRect(renderer, &block); // 在屏幕上绘制食物
}

// 移动蛇的函数
void move_snake(Snake* snake) {
    Point newHead = snake->body[0]; // 获取蛇头的位置
    switch (snake->direction) { // 根据方向更新蛇头的位置
    case 0: newHead.x += SNAKE_BLOCK; break;
    case 1: newHead.y += SNAKE_BLOCK; break;
    case 2: newHead.x -= SNAKE_BLOCK; break;
    case 3: newHead.y -= SNAKE_BLOCK; break;
    }

    // 将蛇的头部移到新的位置
    for (int i = snake->length - 1; i > 0; i--) {
        snake->body[i] = snake->body[i - 1];
    }
    snake->body[0] = newHead;
}

// 检查蛇是否吃到食物的函数
int check_collision(Snake* snake, Point* food) {
    return (snake->body[0].x == food->x && snake->body[0].y == food->y); // 如果蛇头的位置等于食物的位置,则返回真
}

// 更新游戏状态的函数
void update_game(Snake* snake, Point* food) {
    move_snake(snake); // 移动蛇
    if (check_collision(snake, food)) { // 如果蛇吃到了食物
        snake->length++; // 增加蛇的长度
        snake->body = (Point*)realloc(snake->body, snake->length * sizeof(Point)); // 重新分配内存以适应新的长度
        init_food(food); // 重新初始化食物的位置
    }
}

// 主函数
int main(int argc, char* args[]) {
    bool running = true;
    if (SDL_Init(SDL_INIT_VIDEO) < 0) {
        printf("无法初始化SDL: %s\n", SDL_GetError()); // 错误处理
        return -1;
    }

    gWindow = SDL_CreateWindow("Snake Game", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, SCREEN_WIDTH, SCREEN_HEIGHT, SDL_WINDOW_SHOWN);
    if (gWindow == NULL) {
        printf("无法创建窗口: %s\n", SDL_GetError()); // 错误处理
        return -1;
    }

    gRenderer = SDL_CreateRenderer(gWindow, -1, SDL_RENDERER_ACCELERATED);
    if (gRenderer == NULL) {
        printf("无法创建渲染器: %s\n", SDL_GetError()); // 错误处理
        return -1;
    }

    gSnake = (Snake*)malloc(sizeof(Snake));
    init_snake(gSnake);
    init_food(&gFood);

    SDL_Event e;
    bool quit = false;
    while (!quit) {
        while (SDL_PollEvent(&e) != 0) {
            if (e.type == SDL_QUIT) {
                quit = true;
            }
            else if (e.type == SDL_KEYDOWN) {
                switch (e.key.keysym.sym) {
                case SDLK_UP:
                    if (gSnake->direction != 1) gSnake->direction = 3; // 避免反向移动
                    break;
                case SDLK_DOWN:
                    if (gSnake->direction != 3) gSnake->direction = 1; // 避免反向移动
                    break;
                case SDLK_LEFT:
                    if (gSnake->direction != 0) gSnake->direction = 2; // 避免反向移动
                    break;
                case SDLK_RIGHT:
                    if (gSnake->direction != 2) gSnake->direction = 0; // 避免反向移动
                    break;
                }
            }
        }

        // 加载并绘制背景图像
        SDL_Texture* bgTexture = IMG_LoadTexture(gRenderer, "..//image//background.bmp");
        if (bgTexture == NULL) {
            printf("无法加载图像: %s\n", SDL_GetError()); // 错误处理
            return -1;
        }

        SDL_RenderClear(gRenderer);
        SDL_RenderCopy(gRenderer, bgTexture, NULL, NULL);
        SDL_DestroyTexture(bgTexture); // 渲染完成后销毁纹理

        draw_snake(gSnake, gRenderer);
        draw_food(&gFood, gRenderer);

        update_game(gSnake, &gFood);

        SDL_RenderPresent(gRenderer); // 显示新帧
        SDL_Delay(100); // 延迟以控制游戏速度
    }

    free(gSnake->body);
    free(gSnake);
    SDL_DestroyRenderer(gRenderer);
    SDL_DestroyWindow(gWindow);
    IMG_Quit(); // 退出SDL_image
    SDL_Quit();

    return 0;
}

标签:body,direction,food,代码,贪吃蛇,snake,SDL,NULL
From: https://blog.csdn.net/2401_82456630/article/details/141831975

相关文章

  • 顺序结构存储的线性表操作【作业代码 1】
    顺序结构存储的线性表操作顺序结构存储的线性表是一种使用连续内存空间来存储元素的数据结构。在这种结构中,元素之间的相对位置通过物理存储位置直接反映出来,即元素在内存中的地址是连续的。下面,我们将基于您提供的代码片段,详细讨论顺序结构线性表的基本操作,包括初始化、查......
  • 分组密码的模式AES-CBC模式流程解析附:应用代码实现
    CBC模式:CipherBlockChainingmode(密码分组链接模式)CBC模式的加解密CBC模式中,首先将明文分组与前一个密文分组进行XOR运算,然后再进行加密。密文分组像链条一样相互连接在一起。CBC模式的加密流程图CBC模式的解密流程图将一个分组的加密过程分离出来,对ECB模式和CBC......
  • 【Django开发】django美多商城项目完整开发4.0第9篇:邮件与验证,学习目标:【附代码文档
    本教程的知识点为:项目准备项目准备配置1.修改settings/dev.py文件中的路径信息2.INSTALLED_APPS3.数据库用户部分图片1.后端接口设计:视图原型2.具体视图实现用户部分使用Celery完成发送判断帐号是否存在1.判断用户名是否存在后端接口设计:用户部分JWT什......
  • 代码随想录算法训练营,9月2日 | 242.有效的字母异位词,349. 两个数组的交集,202. 快乐数,1
    哈希表理论基础1.根据关键码的值而直接进行访问的数据结构(直白来讲其实数组就是一张哈希表,哈希表中关键码就是数组的索引下标,然后通过下标直接访问数组中的元素);2.哈希表都是用来快速判断一个元素是否出现集合里;3.哈希函数:把值对应到哈希表的函数;哈希碰撞:映射到哈希表同一个索引......
  • 喂饭教程“15行代码”教你用matlab画玫瑰花表白成功!
    本代码十分适合编程小白,大家直接复制黏贴就可以向大家喜欢的人表个白吧!代码:n=800;p=pi;[R,T]=ndgrid(linspace(0,1,n),linspace(-2,20*p,n));x=1-(.5)*((5/4)*(1-mod(3.6*T,2*p)/p).^2-.25).^2;U=2*exp(-T/(8*p));L=sin(U);J=cos(U);y=1.99*(R.^2).*(1.2*R-1).^2.*L;......
  • GCC代码的逻辑结构
    简介gcc代码的核心就是编译器cc1GCC代码的逻辑结构GCC(GNU编译器集合)的架构可以分为多个阶段,这些阶段从高层语言的源代码到机器码的转换。以下是整个过程的工作原理:1.语言特定代码定义:GCC的这一部分负责处理输入编程语言(如C、C++、Fortran等)的具体细节。功能:语言前端......
  • 《深度学习》OpenCV 图像边缘检测 算法解析及代码演示
    目录一、图像边缘检测1、什么是边缘检测2、常用的边缘检测算法        1)Sobel算子    2)Scharr算子        3)Canny边缘检测算法        4)Laplacian算子3、边缘检测流程        1)预处理        2)计算梯度     ......
  • [20240902]验证sql_idz.sh计算PLSQL代码块.txt
    [20240902]验证sql_idz.sh计算PLSQL代码块.txt--//测试验证sql_idz.sh计算PLSQL代码块是否正确.1.环境:SYS@book>@ver2==============================PORT_STRING                  :x86_64/Linux2.4.xxVERSION                     ......
  • 如何训练一个 LSTM 网络以解决特定的序列预测问题(含代码示例)
    关注我,持续分享逻辑思维&管理思维&面试题;可提供大厂面试辅导、及定制化求职/在职/管理/架构辅导;推荐专栏《10天学会使用asp.net编程AI大模型》,目前已完成所有内容。一顿烧烤不到的费用,让人能紧跟时代的浪潮。从普通网站,到公众号、小程序,再到AI大模型网站。干货满满。学成后可......
  • 【网络安全 | Java代码审计】Code-Breaking Puzzles-javacon
    未经许可,不得转载。源码:https://www.leavesongs.com/media/attachment/2018/11/23/challenge-0.0.1-SNAPSHOT.jar,下载至桌面。考察知识点:SpEL注入正文执行命令运行环境:java-jarC:\Users\86177\Desktop\challenge-0.0.1-SNAPSHOT.jar浏览器访问localhost:8080使用JD-......