大二必做项目贪吃蛇超详解之中篇游戏设计与分析

贪吃蛇系列文章

上篇win32
中篇设计与分析

文章目录

  • 贪吃蛇系列文章
  • 1. 地图
    • 1. 1 `<locale.h>`本地化
    • 1. 2 类项
    • 1. 3 setlocale函数
    • 1. 4 宽字符的打印
    • 1. 5 地图坐标
  • 2. 蛇身和食物
  • 3. 数据结构设计
  • 4. 游戏流程设计
  • 5. 核心逻辑实现分析
    • 5. 1 游戏主逻辑
    • 5. 2 GameStart
      • 5. 2. 1 SetInit
      • 5. 2. 2 Welcome
      • 5. 2. 3 InitMap
      • 5. 2. 4 InfoPrint
      • 5. 2. 5 SnakeInit
      • 5. 2. 6 CreatFood


1. 地图

我们最终的贪吃蛇界面是这个样子,可以发现这和之前写的C语言项目的最大不同就在于文字不是依次排列的,那我们的地图应该如何布置呢?
1
2
3
这里回顾一下控制台窗口的一些知识,如果想在控制台的窗口中指定位置输出信息,我们得知道该位置的坐标,所以首先介绍一下控制台窗口的坐标知识。
控制台窗口的坐标如下所示,横向的是X轴,从左向右依次增长,纵向是Y轴,从上到下依次增长
1
在游戏地图上,我们打印墙体使用宽字符□,打印蛇使用宽字符●,打印食物使用宽字符★(这些字符都可以在输入法中打出来)
普通的字符是占一个字节的,这类宽字符是占用2个字节
这里再简单的讲一下C语言的国际化特性相关的知识,过去C语言并不适合非英语国家(地区)使用,因为C语言最初假定字符都是单字节的。但是这些假定并不是在世界的任何地方都适用。

C语言字符默认是采用ASCI编码的,ASCI字符集采用的是单字节编码,且只使用了单字节中的低7位,最高位是没有使用的,可表示为0xxxxxxx;可以看到,ASCII字符集共包含128个字符,在英语国家中,128个字符是基本够用的,但是,在其他国家语言中,比如,在法语中,字母上方有注音符号,它就无法用 ASCI码表示。于是,一些欧洲国家就决定,利用字节中闲置的最高位编入新的符号。比如,法语中的é的编码为130(二进制10000010)。这样一来,这些欧洲国家使用的编码体系,可以表示最多256个符号。但是,这里又出现了新的问题。不同的国家有不同的字母,因此,哪怕它们都使用256个符号的编码方式,代表的字母却不一样。比如,130在法语编码中代表了é,在希伯来语编码中却代表了字母Gimel(汉),在俄语编码中又会代表另一个符号。但是不管怎样,所有这些编码方式中,0–127表示的符号是一样的,不一样的只是128–255的这一段。至于亚洲国家的文字,使用的符号就更多了,汉字就多达10万左右。一个字节只能表示256种符号,肯定是不够的,就必须使用多个字节表达一个符号。比如,简体中文常见的编码方式是 GB2312,使用两个字节表示一个汉字,所以理论上最多可以表示 256x256=65536 个符号。

后来为了使C语言适应国际化,C语言的标准中不断加入了国际化的支持。比如:加入了宽字符的类型wchar_t宽字符的输入和输出函数,加入了<locale.h>头文件,其中提供了允许程序员针对特定地区(通常是国家或者说某种特定语言的地理区域)调整程序行为的函数。

1. 1 <locale.h>本地化

<locale.h>提供的函数用于控制C标准库中对于不同的地区会产生不一样行为的部分。
在标准中,依赖地区的部分有以下几项:

数字量的格式
货币量的格式
字符集
日期和时间的表示形式

1. 2 类项

通过修改地区,程序可以改变它的行为来适应世界的不同区域。但地区的改变可能会影响库的许多部分,其中一部分可能是我们不希望修改的。所以C语言支持针对不同的类项进行修改,下面的一个宏指定一个类项:
LC_COLLATE:影响字符串比较函数 strcoll()strxfrm()
LC_CTYPE:影响字符处理函数的行为。
LC_MONETARY:影响货币格式
LC_NUMERIC:影响 printf()的数字格式。
LC_TIME:影响时间格式 strftime()wcsftime()
LC_ALL:针对所有类项修改,将以上所有类别设置为给定的语言环境。

微软开发文档对类项的介绍

1. 3 setlocale函数

char*setlocale(int category,const char* locale);

setlocale 函数用于修改当前地区,可以针对一个类项修改,也可以针对所有类项。
setlocale 的第一个参数可以是前面说明的类项中的一个,那么每次只会影响一个类项,
如果第一个参数是LC_ALL,就会影响所有的类项。
C标准给第二个参数仅定义了2种可能取值:"C"(正常模式)和""(空字符串,本地模式)。
在任意程序执行开始,都会隐藏式执行调用:

setlocale(LC ALL,"C");

当地区设置为"C"时,设置为C语言默认的模式,这时库函数按正常方式执行。
当程序运行起来后如果想改变地区,就需要调用setlocale函数。用""作为第2个参数,调用setlocale函数就可以切换到本地模式,这种模式下程序会适应本地环境。
比如:切换到我们的本地模式(汉字是宽字符)后就支持宽字符的输出等

setlocale(LC_ALL,"");//切换到本地环境

setlocale 的返回值是一个字符串指针,表示已经设置好的格式。如果调用失败,则返回空指针NULL
setlocale也可以用来查询当前地区,第二个参数设为NULL就可以了。

#include <locale.h>
#include<stdio.h>
int main()
{char* loc;loc = setlocale(LC_ALL, NULL);printf("默认的本地信息:%s\n", loc);loc = setlocale(LC_ALL, "");printf("设置后的本地信息: %s\n", loc);return 0;
}

1

1. 4 宽字符的打印

那如果想在屏幕上打印宽字符,怎么打印呢?
宽字符的字面量必须加上前缀L,否则C语言会把字面量当作窄字符类型处理。
前缀L在单引号前面,表示宽字符,宽字符的打印使用 wprintf,对应 wprintf()的占位符为 %lc
前缀L在双引号前面,表示宽字符串,对应 wprintf()的占位符为 %ls

#include <stdio.h>
#include<locale.h>
int main() {setlocale(LC_ALL, "");wchar_t ch1 = L'●';wchar_t ch2 = L'微';	//汉字也是宽字符wchar_t ch3 = L'软';	wchar_t ch4 = L'★';printf("ab\n");wprintf(L"%lc\n", ch1);	//不要忘记带Lwprintf(L"%lc\n", ch2);wprintf(L"%lc\n", ch3);wprintf(L"%lc\n", ch4);return 0;
}

2
从输出的结果来看,我们发现一个普通字符占一个字符的位置,但是打印一个汉字字符或者宽字符,占用2个字符的位置,那么我们如果要在贪吃蛇中使用宽字符,就得计算好坐标,让X坐标一直为偶数,不然会出现一些问题。
3

1. 5 地图坐标

我们以实现一个棋盘27行,58列的棋盘分析,再围绕地图画出墙,
如下:
4

2. 蛇身和食物

初始化状态:假设蛇的长度是5,蛇身的每个节点是●,在固定的一个坐标处,比如(24,5)处开始出现蛇,连续5个节点。
注意:蛇的每个节点的x坐标必须是2个倍数,否则可能会出现蛇的一个节点有一半儿出现在墙体中,另外一般在墙外的现象,坐标不好对齐。
关于食物,就是在墙体内随机生成一个坐标(x坐标必须是2的倍数),坐标不能和蛇的身体重合,然后打印★。
2

3. 数据结构设计

在游戏运行的过程中,蛇每次吃一个食物,蛇的身体就会变长一节,那么使用链表存储蛇的信息就比较方便了,蛇的每一节其实就是链表的每个节点。每个节点只要记录好蛇身节点在地图上的坐标就行所以蛇节点结构如下:

typedef struct SnakeNode
{int x;int y;struct SnakeNode* next;
}SnakeNode,* pSnakeNode;

要管理整条贪吃蛇,我们再封装一个Snake的结构来维护整条贪吃蛇:
那么Snake中应该有哪些数据呢?

  1. 作为一个链表,头结点是必须要保存下来的
  2. 贪吃蛇可以改变方向,那么贪吃蛇的方向也应该要存储下来
  3. 如果我们想判断贪吃蛇是否死亡,并在游戏结束时告诉玩家是如何死亡的,可以把游戏状态也存储起来
  4. 在游戏中当然少不了分数
  5. 每次吃食物的分数,这个会随着速度的改变而改变,所以也要存储起来
  6. 食物的位置,这个贪吃蛇每次只会在场上生成一个食物,将食物的信息放在Snake中,可以方便开发
  7. 睡眠时间,这个实际上是速度,我们在游戏运行函数中再介绍

另外可以发现,方向只有四个,可以一一列举出来,所以我们可以使用枚举

enum DERCTION	//方向
{UP = 1, DOWN,LEFT,RIGHT
};

状态实际上也是有限的:正常,撞墙,撞到自己,玩家自行退出,也可以一一列举:

enum STATUS
{NORMAL,KILL_BY_WALL,KILL_BY_SELF,ESC
};

那么Snake结构体就可以写成这样,在变量名称前加上_方便与外部变量区分

typedef struct Snake
{pSnakeNode _Head;	//头enum DERCTION _Dir;	//方向enum STATUS _Sta;	//状态pSnakeNode _Food;	//食物int _FoodAdd;		//食物加的分数int _Score;			//当前分数int _SleepTime;		//睡眠时间
}Snake,*pSnake;

4. 游戏流程设计

5
那么至此,前期准备基本完成,接下来我们开始完成游戏的核心逻辑

5. 核心逻辑实现分析

5. 1 游戏主逻辑

程序开始就设置程序支持本地模式,然后进入游戏的主逻辑。
主逻辑分为3个过程

游戏开始(GameStart)完成游戏的初始化
游戏运行(GameRun)完成游戏运行逻辑的实现
游戏结束(GameOver)完成游戏结束的工作

注意:setlocale(LC_ALL, "");不需要放在上面的逻辑中,因为上面的逻辑会随着游戏的再来一把反复执行,而这个代码并不需要反复运行。

<test.h>

#include"game.h"void game()
{char input = 'y';		//用于判断是否再来一把do{Snake s = { 0 };	//做出一条蛇,将其中的内容都置为空srand((unsigned int)time(NULL));	//食物的生成需要随机数,我们在这里设置一下//开始游戏GameStart(&s); //进行游戏GameRun(&s);//结束游戏GameOver(&s); //这个代码用于解决一个bug,在后面介绍//这是AI给出的解决办法,就不多介绍了,<conio.h>是这两个函数需要的头文件//这个while循环是用来读取蛇运行的时候按下的VK虚拟键的循环,//把在蛇运行的时候按下的VK键的键值全面读走(包括上键的键值)while (_kbhit())	//_kbhit()检测是否有按键被按下{//使用 _getch() 获取按下的键_getch();}//如果是主动退出的,就不需要询问是否再来一把了if (s._Sta == ESC){input = 'n';getchar();	//这个getchar用于在release版本下阻止程序直接退出}//在结束之后,询问是否要再来一把else{SetPos(15, 15);printf("要再来一把吗?(Y/y)");input = getchar();while (getchar() != '\n');	//清理'\n'}} while (input=='y'||input=='Y');SetPos(10, 27);		//程序退出时,会有一个xxx程序已正常退出的提示,我们让它不要破坏游戏地图
}int main()
{setlocale(LC_ALL, "");//设置能输出长字符game();
}

在游戏过程中我们会用到非常多次SetPos来设置光标位置,至于这些位置的具体坐标可以自行不断尝试来找到较好的位置,博客中的是我个人觉得比较好的。

5. 2 GameStart

这个部分要完成的任务:

控制台窗口大小的设置
控制台窗口名字的设置
鼠标光标的隐藏
打印欢迎界面
创建地图
初始化蛇
创建第一个食物

我们将其中的每一个任务分别封装成一个函数:

void GameStart(pSnake ps)
{//设置控制台大小,隐藏光标SetInit();//打印欢迎界面Welcome();//布置地图InitMap();//打印介绍信息InfoPrint();//放蛇SnakeInit(ps);//放食物CreatFood(ps);//getchar();	//可以用来停止代码执行,方便调试,项目完成后要注释掉
}

5. 2. 1 SetInit

void SetInit()
{system("mode con cols=100 lines=30");system("title 贪吃蛇");HANDLE Houtput = GetStdHandle(STD_OUTPUT_HANDLE);CONSOLE_CURSOR_INFO stdoutinfo;GetConsoleCursorInfo(Houtput, &stdoutinfo);stdoutinfo.bVisible = false;SetConsoleCursorInfo(Houtput, &stdoutinfo);
}

这个函数就是上一篇文章的主要内容,这里就不再赘述了。

5. 2. 2 Welcome

void Welcome()
{//打印欢迎界面SetPos(40, 15);wprintf(L"欢迎来到贪吃蛇");SetPos(40, 25);system("pause");	//这个代码相当于打印一个"请按任意键继续...",和 getchar();system("cls");		//清空屏幕SetPos(25, 12);wprintf(L"按↑↓←→控制方向,F1加速,F2减速,速度越快,分数越高");	//打印汉字也可以使用 printfSetPos(25, 13);wprintf(L"空格键暂停,ESC退出");SetPos(40, 25);system("pause");system("cls");
}

这个函数就是游戏最开始的两个界面。

5. 2. 3 InitMap

我们在这个函数中会用许多次宽字符,为了方便使用,我们可以在头文件中进行宏定义

#define WALL L'□'
#define SNAKE_BODY L'●'
#define FOOD L'★'

这样,比如我们要打印墙体,我们就可以直接:

wprintf(L"%lc",WALL);

参考代码:

void InitMap()
{for (int i = 0; i < 60; i += 2)wprintf(L"%lc",WALL);	//打印第一行SetPos(0, 28 - 1);for (int i = 0; i < 60; i += 2)wprintf(L"%lc", WALL);	//打印最下面一行for (int i = 1; i < 28; i++){SetPos(0, i);wprintf(L"%lc", WALL);	//打印左边一列}for (int i = 1; i < 28; i++){SetPos(60 - 2, i);wprintf(L"%lc", WALL);	//打印右边一列}
}

5. 2. 4 InfoPrint

可以在这里打印上自己的名字做个防伪认证:)

void InfoPrint()
{//打印提示信息SetPos(64, 15);printf("不能穿墙,不能咬到自己\n");SetPos(64, 16);printf("用 ↑. ↓. ←. → 分别控制蛇的移动.");SetPos(64, 17);printf("F1 为加速,F2 为减速\n");SetPos(64, 18);printf("ESC :退出游戏.space:暂停游戏.");SetPos(64, 20);printf("CSDN:fhvyxyci");
}

分数和每次吃食物的得分由于要刷新,就不在初始化的时候打印了。

5. 2. 5 SnakeInit

void SnakeInit(pSnake ps);

初始化蛇的步骤:

  1. 头插出一个有5个节点的链表
  2. 把这5个节点打印出来
  3. 初始化结构体其他数据
void SnakeInit(pSnake ps)
{pSnakeNode cur = NULL;for (int i = 0; i < 5; i++)	//初始设置长度为5{//创建一个节点,这一步也封装成 BuyNodepSnakeNode cur = (pSnakeNode)malloc(sizeof(SnakeNode));if (!cur){perror("SnakeInit()::malloc()");exit(1);}cur->x = X_INIT + 2 * i;	//X_INIT和Y_INIT是宏定义,方便修改初始坐标cur->y = Y_INIT;cur->next = NULL;//头插if (!ps->_Head){ps->_Head = cur;}else{cur->next = ps->_Head;ps->_Head = cur;}}//打印蛇cur = ps->_Head;while (cur){SetPos(cur->x, cur->y);wprintf(L"%lc", SNAKE_BODY);cur = cur->next;}//初始化数据ps->_Dir = RIGHT;		//初始方向为右ps->_FoodAdd = 10;		//初始食物的分数为10ps->_Score = 0;			//初始分数ps->_SleepTime = 200;	//_SleppTime与速度有关ps->_Sta = NORMAL;		//初始状态是正常
}

5. 2. 6 CreatFood

这个函数不是只在初始化的时候调用,写的时候可能要注意一下。

void CreatFood(pSnake ps)
{int x = 0, y = 0;
again:do{x = rand() % 55 + 2;	//注意范围y = rand() % 26 + 1;} while (x % 2 == 1);	//x必须是偶数pSnakeNode cur = ps->_Head;while (cur){//检查食物是否与身体重合if (cur->x == x && cur->y == y)goto again;		//当然,goto语句一般不推荐使用,你可以改造一下这里的逻辑,换成循环cur = cur->next;}//食物要有x,y坐标,那不如直接把它做成一个SnakeNode,这样还可以方便后面吃食物pSnakeNode food = (pSnakeNode)malloc(sizeof(SnakeNode));if(!food){perror("CreatFood()::malloc()");exit(1);}food->x = x;food->y = y;food->next = NULL;ps->_Food = food;//打印食物SetPos(ps->_Food->x, ps->_Food->y);wprintf(L"%c", FOOD);
}

剩下的逻辑在后面的博客中介绍。
谢谢你的阅读,喜欢的话来个点赞收藏评论关注吧!
我会持续更新更多优质文章

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/413971.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

145. 利用 Redis Bitmap实践: 用户签到统计

文章目录 一、Redis Bitmap简介二、Bitmap 的主要应用三、Go使用Redis实现签到统计用户签到查询用户签到状态统计今年累计签到天数统计当月的签到情况 总结 在现代应用程序中&#xff0c;用户签到是一个常见的功能。我们通常使用 MySQL 数据库来存储用户的签到记录。然而&#…

Go入门:gin框架极速搭建图书管理系统

Go入门:gin框架极速搭建图书管理系统 前言 本项目适合 Golang 初学者,通过简单的项目实践来加深对 Golang 的基本语法和 Web 开发的理解。 项目源码请私信,欢迎前往博主博客torna.top免费查看。 项目结构 D:. ├─ go.mod ├─ go.sum │ ├─ cmd │ └─ main │ …

Docker 容器编排之 Docker Compose

目录 1 Docker Compose 概述 1.1 主要功能 1.2 工作原理 1.3 Docker Compose 中的管理层 2 Docker Compose 的常用命令参数 2.1 服务管理 2.1.1 docker-compose up &#xff1a; 2.1.2 docker-compose down &#xff1a; 2.1.3 docker-compose start &#xff1a; 2.1.4 docker…

OpenCV绘图函数(7)从一个椭圆定义中提取出多边形的顶点坐标函数ellipse2Poly()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 近似一个椭圆弧为一个多边形线。 函数 ellipse2Poly 计算近似指定椭圆弧的多边形线的顶点。它被 ellipse 函数所使用。如果 arcStart 大于 arcEn…

景芯SoC A72实战反馈

先说结论&#xff1a; 内容非常全面&#xff0c;讲解到位&#xff0c;会有专门的工程师一对一答疑&#xff0c;整个项目跑下来提升非常大&#xff0c;绝对物超所值&#xff01; 一些细节&#xff1a; 本人微电子专业研一在读&#xff0c;有过两次简单的数字芯片流片经历&…

spring security 相关过滤器

Spring Security 提供了 30 多个过滤器。默认情况下Spring Boot 在对 SpringSecurity 进入自动化配置时&#xff0c;会创建一个名为 SpringSecurityFilerChain 的过滤器&#xff0c;并注入到Spring容器中&#xff0c;这个过滤器将负责所有的安全管理&#xff0c;包括用户认证、…

Windows系统中批量管理Windows服务器远程桌面工具——RDCMan

一、背景 在公司没有部署对应的堡垒机系统之前,做运维测试工作的人员,需要管理大量的服务器,每天需要对服务器进行必要的巡检、系统更新发布等内容,特别是有很多Windows服务器的时候,如果我们使用Windows自带的“远程桌面连接”只能一台台连接,比较繁琐。并且不能知道那台…

十二星座男女、穿越到古代会成为什麽角色 。

白羊座&#xff08; 大将军 &#xff09;&#xff1b; 金牛座&#xff08;财务大臣&#xff09;&#xff1b; 双子座&#xff08; 奸臣 &#xff09;&#xff1b; 巨蟹座&#xff08;御厨太医&#xff09;&#xff1b; 狮子座&#xff08;皇帝&#xff09;&#xff1b; …

虚幻5|按键触发学习

一&#xff0c;如图参考 1.下移 驱动阈值 越大按时间长才会触发&#xff0c;越小很快就可以触发 2.按下 当按下超出驱动阈值大小就会触发一次&#xff0c;这里的驱动阈值只能设置再0.1~1的大小 3.已松开 当按下的时候&#xff0c;先触发单次的started&#xff0c;如果按压…

精选算法编程题

一、有序数组的平方 给你一个按 非递减顺序 排序的整数数组 nums&#xff0c;返回 每个数字的平方 组成的新数组&#xff0c;要求也按 非递减顺序 排序。 示例 1&#xff1a; 输入&#xff1a;nums [-4,-1,0,3,10]输出&#xff1a;[0,1,9,16,100]解释&#xff1a;平方后&am…

中国各城市金融科技公司数目数据集(2009-2023年)

金融科技&#xff08;FinTech&#xff09;是金融与科技深度融合的产物&#xff0c;它利用大数据、云计算、人工智能、区块链等现代信息技术手段&#xff0c;对传统金融产品、业务、流程及服务模式进行革新&#xff0c;从而实现金融服务效率的提升、风险管理的优化以及客户体验的…

《深入浅出WPF》读书笔记.8路由事件

《深入浅出WPF》读书笔记.8路由事件 背景 路由事件是直接响应事件的变种。直接响应事件&#xff0c;事件触发者和事件响应者必须显示订阅。而路由事件的触发者和事件响应者之间的没有显示订阅&#xff0c;事件触发后&#xff0c;事件响应者安装事件监听器&#xff0c;当事件传…

我熟悉你的NLP焦虑,只因没有它

大家好&#xff0c;我是凡人。 最近凡人被一个NLP&#xff08;神经语言程序学[Neuro-Linguistic Programming]的英文缩写&#xff09;学习内容给震惊到了&#xff0c;熟悉NLP的同学都知道&#xff0c;NLP知识不仅庞大而且很有深度。 比如&#xff1a;机器信息就包含下图内容 肝…

集成电路学习:什么是IDE集成开发环境

IDE&#xff1a;集成开发环境 IDE&#xff0c;全称“Integrated Development Environment”&#xff0c;即集成开发环境&#xff0c;是一种用于提供程序开发环境的应用程序。它集成了代码编写、分析、编译、调试等多种功能于一体的开发软件服务套&#xff0c;为开发者提供了一个…

mybatis-plus中Swagger 模式和Kotlin 模式是什么?

在 MyBatis-Plus 中&#xff0c;Swagger 模式和 Kotlin 模式是为了支持特定技术栈和开发需求的两种配置选项。它们分别针对 API 文档生成和 Kotlin 语言提供了更好的支持和集成。 Swagger 模式 Swagger 模式主要用于生成 API 文档。在 MyBatis-Plus 中启用 Swagger 模式后&am…

C语言 | Leetcode C语言题解之第378题有序矩阵中第K小的元素

题目&#xff1a; 题解&#xff1a; bool check(int **matrix, int mid, int k, int n) {int i n - 1;int j 0;int num 0;while (i > 0 && j < n) {if (matrix[i][j] < mid) {num i 1;j;} else {i--;}}return num > k; }int kthSmallest(int **matri…

CSAPP Data Lab

CSAPP 的第一个 Lab&#xff0c;对应知识点为书中的第 2 章&#xff08;信息的表示与处理&#xff09;&#xff0c;要求使用受限制的运算符和表达式实现一些位操作。主要分为两个部分&#xff1a;整数部分和浮点数部分。其中整数部分限制较多&#xff0c;比较偏重技巧性&#x…

点餐收银小程序

一、项目概述 Hi&#xff0c;大家好&#xff0c;今天分享的项目是《点餐收银小程序》。 系统含管理员/商家/用户三种角色&#xff0c;商家能维护菜式类别、维护菜品信息&#xff0c;用户在小程序能够选择门店&#xff0c;查看门店下各个分类的菜式信息&#xff0c;并进行加购…

C语言基础(三十一)

1、线性搜索&#xff1a; #include "date.h" #include <stdio.h> #include <stdlib.h> #include <time.h> // 希尔排序 void shellSort(int arr[], int n) { for (int gap n / 2; gap > 0; gap / 2) { for (int i gap; i < n; i…

智慧党建解决方案

1. 新时代党建工作背景 报告强调了新时代党建工作的重要性&#xff0c;提出要利用互联网、大数据等新兴技术推进智慧党建&#xff0c;提高党的执政能力和领导水平。 2. 基层党组织建设挑战 基层党组织在日常工作中面临组织管理难、过程监管难、宣传教育难等问题&#xff0c;…