基于Linux系统和ncurses库的贪吃蛇小游戏

news2025/1/12 21:00:07

目录

前言

 一、地图,蛇身,食物设计

二、蛇和食物的初始化

食物

三、添加和删除蛇身节点

四、main函数和蛇运行方向线程

五、地图刷新线程

最终源码


前言

        ncurses库是什么我并没有深入了解,本文的重点也不是ncurses的使用,至于为什么要使用ncurses库,我们先来看一个问题:我们要通过方向键来控制蛇的走向,用c的库函数scanf和getchar都需要我们输入字符后再按回车,这显然不合理,而ncurses库就能解决这些问题,我们不需要知道他是怎么实现的,我们只需要会用就行了,因为这节主要是巩固链表的知识。最后有源码展示

最终效果:


 一、地图,蛇身,食物设计

蛇活动的范围为 20x20 ,地图的左右边界用 "|" 符号表示,上下边界用 "--" ,蛇身用 "[]"表示,食物用 "##" ,第0行和第0列是边界,除边界外每行每列都由两个空格表示,这样看起来比较方正,就是说每个单元都是两个空格的大小,所以蛇身和食物就设计成两个符号。 上下边界有42个 "-" ,因为 "|" 只占一个空格的位置,上下边界除去第一个和最后一个,其他两个为一组,刚好有20组,这个设计简直完美。

void gamemap()
{
	int hang,lie;

    /*要不断刷新地图来更新蛇和食物的位置,这里把光标归零,覆盖地图*/
	move(0,0);
    
    /*一共有22行,0和21行是边界*/
	for(hang=0;hang<=21;hang++)
	{
        /*打印上边界*/
		if(hang==0)
		{
            /*因为一个'-'和一个'|'对齐了,所以只需要打印21个"--"*/
			for(lie=0;lie<=20;lie++)
			{
				printw("--");
			}
            /*换行*/
			printw("\n");
		}
        /*打印蛇的活动范围,1到20行*/
		if(hang>=1 && hang<=20)
		{
            /*打印列,这里和上面不一样,要打印22列,因为打印上下边界的时候其中一组"--"被拆开来
            当成边界*/
			for(lie=0;lie<=21;lie++)
			{
                /*第0列和第21列打印左右边界*/
				if(lie==0 || lie==21)
				{
					printw("|");
				}
                /*判断蛇身是否存在,传入循环到这里的地图坐标,存在打印蛇身*/
				else if(hassnakenode(hang, lie))
				{
					printw("[]");
				}
                /*判断食物是否存在,传入循环到这里的地图坐标,存在打印食物*/
				else if(hasfood(hang, lie))
				{
					printw("##");
				}
                /*打印空格,两个空格为一个单元*/
				else
				{
					printw("  ");
				}
			}
            /*换行*/
			printw("\n");
		}
        /*最后一行*/
		if(hang == 21)
		{
			for(lie=0;lie<=20;lie++)
			{
				printw("--");
			}
			printw("\n");
            /*可以自行添加打印信息,我这里打印食物的位置*/
			printw("By sakabu,food.hang=%d,food.lie=%d\n",food.hang,food.lie);
		}
	}
}

/*判断蛇身是否存在*/
int hassnakenode(int i, int j)
{
	struct snake *p;
	p = head;

    /*遍历蛇的链表,如果蛇身的坐标等于传进来的地图坐标,返回1*/
	while(p != NULL)
	{
		if(p->hang==i && p->lie==j)
	        {
                	return 1;
       		}
		p = p->next;//遍历
	}
	return 0;
}

/*判断食物是否存在*/
int hasfood(int i, int j)
{
    /*如果传进来的地图坐标等于食物的坐标,返回1*/
	if(food.hang==i && food.lie==j)
	{
		return 1;
	}
	return 0;
}

在地图上显示蛇身和食物其实就是不断循环读取蛇身和食物的位置,不断更新覆盖,最终实现动画的效果,理解这一点剩下的就好办了。


二、蛇和食物的初始化

初始化一个结构体,里面只有三个元素

struct snake
{
    int hang;//行
    int lie;//列
    struct snake *next;//下个节点
};

食物

我们需要让食物被吃掉后随机生成,用到 rand() 函数,它会随机生成0~32767之间的数,但我们要控制食物生成范围为1~20,所以食物的初始化代码如下

/*定义食物为全局变量*/
struct snake food;

void initfood()
{
    /*rand余20后结果为0~19,再加1就是我们想要的范围*/
	int x = (rand()%20)+1;
	int y = (rand()%20)+1;

	food.hang = x;
	food.lie = y;
}

每当食物被蛇“吃掉”,后,调用initfood函数就可以随机生成下一个食物。

这里我们设计蛇的时候有点反常识,head节点是最开始生成的节点,后面生成的节点的最后一个叫做tail节点,所以我们的蛇移动的时候,都是在tail节点后添加节点,然后删除head节点,如果吃了食物就不删除。所以tail节点是蛇的头,head节点是蛇的尾巴!!!这一点一定要事先搞清楚,不然后续对链表的操作就会让人云里雾里。(你当然可以自己更改,只不过我这个程序链表头是蛇的尾巴,链表尾是蛇的头)

/*定义两个全局变量*/
struct snake *head = NULL;
struct snake *tail = NULL;

/*设定蛇的初始运动方向为向右走,这个程序设计成蛇从左上角出生*/
int dir = RIGHT;

void initsnake()
{
	struct snake *p;

    /*蛇死掉之后会调用这个函数,以防最后一次不是向右走的,这里再初始化一次*/
	dir = RIGHT;

    /*死掉之后遍历链表,把内存全部释放*/
	while(head != NULL)
	{
		p = head;
		head = head->next;
		free(p);
	}

    /*运行到这里就是一条新蛇了,蛇尾默认坐标为(2,2)*/
	head = (struct snake *)malloc(sizeof(struct snake));
	head->hang = 2;
	head->lie = 2;
	head->next = NULL;

    /*蛇头初始时指向蛇尾*/
	tail = head;

    /*蛇的初始长度为3*/
	addnode();
	addnode();
}

初始蛇: 

三、添加和删除蛇身节点

 大概思路是这样的,用户输入小键盘的上下左右键,ncurses库有关于这四个按键的宏定义

#define KEY_DOWN         0402

#define KEY_UP               0403

#define KEY_LEFT           0404

#define KEY_RIGHT         0405

 添加节点的时候,是在tail节点添加下一个新节点,也就是蛇的头部,那自然要判断按键的输入,根据输入的按键来设置新节点的行列坐标,可以看到上一个代码定义了dir这个全局变量,初始值为RIGHT,我们先来看添加节点的代码。

void addnode()
{
    /*为新节点分配内存空间*/
	struct snake *new = (struct snake *)malloc(sizeof(struct snake));

    /*初始化new节点的next*/
	new->next = NULL;

    /*条件选择*/
	switch(dir)
	{
        /*蛇运动方向为向上,就让新节点的行坐标相比蛇头-1,列不动*/
		case UP:
			new->hang = tail->hang-1;
			new->lie = tail->lie;
			break;
        /*同里*/
		case DOWN:
			new->hang = tail->hang+1;
            new->lie = tail->lie;
			break;
		case LEFT:
            new->hang = tail->hang;
            new->lie = tail->lie-1;
			break;
		case RIGHT:
            new->hang = tail->hang;
            new->lie = tail->lie+1;
			break;
	}

    /*蛇头的next为新节点,“建立链接”*/
	tail->next = new;
    /*蛇头为新节点*/
	tail = new;
}

可以看到,添加节点是根据蛇的运动方向来改变新节点的坐标的,接下来看看删除节点。

/*如果没吃食物,就需要不断的添加节点,同时删除节点*/
void delnode()
{
	struct snake *p;
	
    /*保存蛇尾的地址*/
    p = head;

    /*蛇尾指向他的next*/
	head = head->next;

    /*释放蛇尾的内存*/
	free(p);
}

没吃食物,就添加new节点(新蛇头,根据运动方向);删除head节点(蛇尾)。


四、main函数和蛇运行方向线程

我们现在来看看main函数,理所当然的,我们需要不断刷新地图,来更新蛇的位置,实现动画效果;与此同时,我们还需要不断检测用户的按键输入,这显然需要两个while循环,但正常的裸机程序,它跑不了两个while,你要说把他们放进同一个while里,可能会出现响应太慢的结果或者其他问题,这时候就需要引入线程编程,在我之前的博客里有讲Linux线程编程,不需要实现同步等复杂的关系,我们只需要使用最简单的创建线程,一个线程运行刷新地图,另一个线程运行监测按键。

#include <pthread.h>

int main()
{
    /*创建两个线程标识符*/
	pthread_t t1;
	pthread_t t2;

    /*初始化ncurses界面的函数,后续会给出,先不看*/
	initcurses();

    /*初始化蛇和食物*/
	initsnake();
	initfood(); 
    
    /*初始化地图*/
	gamemap();

    /*创建两个线程,第二个参数默认为NULL,第四个参数表示传入运行程序的参数,我们不需要传入参数*/
	pthread_create(&t1, NULL, refreshmap, NULL);
	pthread_create(&t2, NULL, changedirection, NULL);

    /*死循环防止主线程退出*/
	while(1);

	return 0;
}

void initcurses()
{
    /*都是ncurses库的函数,具体用法我也不太清楚*/
	initscr();
	keypad(stdscr,1);
	noecho();
}

后面几个函数只不过互相调用比较多,但还是很好理解的。我们先看 changedirection 函数,这个函数的功能就是捕获用户输入按键,来改变之前定义的全局变量 dir 。

#define UP     1
#define DOWN  -1
#define LEFT   2
#define RIGHT -2

/*定义全局变量存放捕获的按键值*/
int key;

void* changedirection() 
{
	while(1)
	{
        /*捕获按键值*/
		key = getch();
		switch(key)
		{
			case KEY_DOWN:
                /*重点看turn函数*/
				turn(DOWN);
				break;
			case KEY_UP:
				turn(UP);
				break;
			case KEY_LEFT:
				turn(LEFT);
				break;
			case KEY_RIGHT:
				turn(RIGHT);
				break;
		}
	}
	return NULL;	
}

void turn(int direction)
{
    /*dir默认为RIGHT(-2),如果传进来的绝对值不和之前保存的dir的值一样,才允许改变方向*/
	if(abs(dir) != abs(direction))
	{
        /*将改变的方向赋值给dir,这个dir就是我们添加新节点判断的dir*/
		dir = direction;
	}
}

为什么要用turn函数,为什么相反的方向要定义成一正一负?在之前调试的过程中,发现蛇往右运行,我按下左键后直接掉头了,这显然不合理,所以要限制蛇的运行方向,只能向左转或向右转,结合宏定义和turn函数的判断就能理解这个函数的用途。abs是取绝对值函数。


五、地图刷新线程

void* refreshmap()
{
	while(1)
	{
        /*蛇移动的函数,后续有*/
		movesnake();
        /*不断刷新地图*/
		gamemap();
        /*ncurses库的函数,刷新界面*/
		refresh();
        /*控制刷新速度,单位为1us*/
		usleep(100000);
	}
	return NULL;
}

void movesnake()
{
    /*根据当前运行方向更新蛇头位置*/
	addnode();
    /*把蛇头的坐标传入判断食物的函数*/
	if(hasfood(tail->hang, tail->lie))
	{
        /*如果蛇头的坐标和食物重合,刷新食物*/
		initfood();
	}
	else
	{
        /*没吃到食物才进入这里,删除蛇尾*/
		delnode();
	}

    /*如果蛇撞到边界或撞到自己*/
	if(ifsnakedie())
	{
        /*刷新蛇*/
		initsnake();
	}
}

/*判断蛇是否死亡*/
int ifsnakedie()
{
	struct snake *p;
	p = head;

    /*判断蛇头和边界的坐标是否重合*/
	if(tail->hang==0 || tail->lie==0 || tail->hang==21 || tail->lie==22)
    {
        /*撞墙死亡*/
        initsnake();
    }
    /*遍历蛇身*/
	while(p->next != NULL)
	{
        /*如果蛇身和蛇头重合*/
		if(p->hang==tail->hang && p->lie==tail->lie)
		{
            /*撞自己而死*/
			return 1;
		}
		p = p->next;
	}
	return 0;
}

地图刷新还是比较复杂,函数调用较多,但每个函数实现的功能都不难理解,自己捋一遍思路会更清晰。


最终源码

#include <curses.h>
#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>

#define UP     1
#define DOWN  -1
#define LEFT   2
#define RIGHT -2

void initcurses()
{
	initscr();
	keypad(stdscr,1);
	noecho();
}

struct snake
{
	int hang;
	int lie;
	struct snake *next;
};

struct snake *head = NULL;
struct snake *tail = NULL;
int key;
int dir = RIGHT;

struct snake food;

void initfood()
{
	int x = (rand()%20)+1;
	int y = (rand()%20)+1;

	food.hang = x;
	food.lie = y;
}

int hassnakenode(int i, int j)
{
	struct snake *p;
	p = head;

	while(p != NULL)
	{
		if(p->hang==i && p->lie==j)
	        {
                	return 1;
       		}
		p = p->next;
	}
	return 0;
}

int hasfood(int i, int j)
{
	if(food.hang==i && food.lie==j)
	{
		return 1;
	}
	return 0;
}

void gamemap()
{
	int hang,lie;

	move(0,0);

	for(hang=0;hang<=21;hang++)
	{
		if(hang==0)
		{
			for(lie=0;lie<=20;lie++)
			{
				printw("--");
			}
			printw("\n");
		}
		if(hang>=1 && hang<=20)
		{
			for(lie=0;lie<=21;lie++)
			{
				if(lie==0 || lie==21)
				{
					printw("|");
				}
				else if(hassnakenode(hang, lie))
				{
					printw("[]");
				}
				else if(hasfood(hang, lie))
				{
					printw("##");
				}
				else
				{
					printw("  ");
				}
			}
			printw("\n");
		}
		if(hang == 21)
		{
			for(lie=0;lie<=20;lie++)
			{
				printw("--");
			}
			printw("\n");
			printw("By sakabu,food.hang=%d,food.lie=%d\n",food.hang,food.lie);
		}
	}
}

void addnode()
{
	struct snake *new = (struct snake *)malloc(sizeof(struct snake));

	new->next = NULL;

	switch(dir)
	{
		case UP:
			new->hang = tail->hang-1;
			new->lie = tail->lie;
			break;
		case DOWN:
			new->hang = tail->hang+1;
                        new->lie = tail->lie;
			break;
		case LEFT:
                        new->hang = tail->hang;
                        new->lie = tail->lie-1;
			break;
		case RIGHT:
                        new->hang = tail->hang;
                        new->lie = tail->lie+1;
			break;
	}

	tail->next = new;
	tail = new;
}

void initsnake()
{
	struct snake *p;

	dir = RIGHT;

	while(head != NULL)
	{
		p = head;
		head = head->next;
		free(p);
	}

	initfood();
	head = (struct snake *)malloc(sizeof(struct snake));
	head->hang = 2;
	head->lie = 2;
	head->next = NULL;

	tail = head;

	addnode();
	addnode();
//	addnode();
}

void delnode()
{
	struct snake *p;
	p = head;

	head = head->next;

	free(p);
}

int ifsnakedie()
{
	struct snake *p;
	p = head;

	if(tail->hang==0 || tail->lie==0 || tail->hang==21 || tail->lie==22)
        {
                initsnake();
        }
	while(p->next != NULL)
	{
		if(p->hang==tail->hang && p->lie==tail->lie)
		{
			return 1;
		}
		p = p->next;
	}
	return 0;
}

void movesnake()
{
	addnode();
	if(hasfood(tail->hang, tail->lie))
	{
		initfood();
	}
	else
	{
		delnode();
	}

	if(ifsnakedie())
	{
		initsnake();
	}
}

void* refreshmap()
{
	while(1)
	{
		movesnake();
		gamemap();
		refresh();
		usleep(100000);
	}
	return NULL;
}

void turn(int direction)
{
	if(abs(dir) != abs(direction))
	{
		dir = direction;
	}
}

void* changedirection() 
{
	while(1)
	{
		key = getch();
		switch(key)
		{
			case KEY_DOWN:
				turn(DOWN);
				break;
			case KEY_UP:
				turn(UP);
				break;
			case KEY_LEFT:
				turn(LEFT);
				break;
			case KEY_RIGHT:
				turn(RIGHT);
				break;
		}
	}
	return NULL;	
}

int main()
{
	pthread_t t1;
	pthread_t t2;

	initcurses();

	initsnake();

	gamemap();

	pthread_create(&t1, NULL, refreshmap, NULL);
	pthread_create(&t2, NULL, changedirection, NULL);

	while(1);

//	getch();
//	endwin();

	return 0;
}

代码有300多行,来看看最终运行效果:

可以看到运行的时候一开始有乱码,我看别人加了noecho这个函数后就不会有了,我这里要重新按一下全屏键才能正常显示,我也不知道什么原因,不过不影响最终效果,实际上还有个小bug,就是食物可能会生成会被蛇挡住,这里就留给你们自己去改了,撞自己、撞墙也能正常死亡,有时候会报错,有时候会直接复活,优化的地方还有很多。

本文就到这里,这个贪吃蛇小游戏将很多知识点都串联起来了,我是根据最终代码来讲思路的,肯定不如一步一步写一步一步调试效果来的好,但如果你看完这篇文章,能完全理解所有代码,相信你的编程水平也会上一层楼,自己去运行试试,自己设计一些关卡,把上述的bug解决一下,也是一种锻炼。 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

BUUCTF PWN wp--pwn1_sctf_2016

第一步 checksec&#xff0c;并检查该题的保护机制&#xff0c;32位 Arch: i386-32-little 这表示程序的架构是32位的i386&#xff08;即x86&#xff09;&#xff0c;并且使用小端序&#xff08;little-endian&#xff09;存储方式。这意味着程序是为32位系统设计的。RELRO: Pa…

期权末日轮行情即将来临!注意两个操作更好盈利!

今天带你了解期权末日轮行情即将来临&#xff01;注意两个操作更好盈利&#xff01;期权末日轮&#xff0c;就是指在期权合约到期前的最后几天&#xff0c;比如50ETF期权品种的到期日是每个月第四个星期的星期三&#xff0c;那么在最后一个星期就有可能发生末日轮行情了。 末日…

Linux安装Navicat Premium

一、安装Navicat Premium17 1、下载安装包 https://www.navicat.com.cn/download/navicat-premium#linux 2、赋执行权限 //假设安装包在/etc/navicat目录下 cd /etc/navicat chmod x navicat17-premium-cs-x86_64.AppImage 3、启动应用程序 ./navicat17-premium-cs-x86_64…

python.exe -m pip install --upgrade pip报错解决

引言 在执行命令的时候&#xff0c;提示可以更新版本 [notice] A new release of pip is available: 24.1.2 -> 24.2 [notice] To update, run: python.exe -m pip install --upgrade pip按照提示&#xff0c;直接使用 python.exe -m pip install --upgrade pip 命令进行更…

Cocos Creator2D游戏开发(15)---预制体和按钮的绑定以及冷却效果的实现

场景: 植物大战僵尸中,种植植物前,要判断状态,只有在阳光充足时才能点击 图片资源: 预制体的创建,创建一个空节点命名: CardTemplate,将三张豌豆射手相关的图片拖入 如下图 其中card_mask图片中的透明度改为150; 在assets中创建prefab文件夹,将CardTemplate节点直接拖入pre…

openGauss——体系结构

一、体系结构概览 二、驱动程序 客户端程序使用驱动程序&#xff0c;向openGauss的后端管理线程GaussMaster发起连接请求。openGauss目前支持以下四种驱动程序&#xff1a; JDBC&#xff0c;用于Java连接ODBC&#xff0c;开放数据库互连Libpq&#xff0c;C语言程序接口Psycop…

SpringMvc 以配置类的形式代替xml文件

1、配置类 1.1、创建Mvc 项目之后创建 MyWebApplicationInitializer 类 实现接口 WebApplicationInitializer public class MyWebApplicationInitializer implements WebApplicationInitializer {Overridepublic void onStartup(ServletContext servletContext) throws Serv…

大数据-101 Spark Streaming DStream转换 窗口操作状态 跟踪操作 附带多个案例

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff08;已更完&#xff09;HDFS&#xff08;已更完&#xff09;MapReduce&#xff08;已更完&am…

windows安装pytorch精简版(英伟达GPU)

1 下载anaconda 官网:Index of /anaconda/archive/ | 清华大学开源软件镜像站 | Tsinghua Open Source Mirror 选择下载Anaconda-1.4.0-Windows-x86.exe 2 创建虚拟环境 以管理员身份打开Anaconda Prompt conda env list conda creat -n yolov8 python3.8 创建过程中有提示,填…

【图文详解】idea码云环境搭建

公众号&#xff1a;墨轩学习网-----B站&#xff1a;墨轩大楼 欢迎关注&#xff01;&#xff01;&#xff01; 一、码云简介 目前开源中国的四大框架&#xff0c;即四条产品线&#xff1a;开源中国社区、众包、码云和招聘。 码云是开源中国推出的基于Git的代码托管服务&#…

鸿蒙HarmonyOS开发:创建新的Lite工程

当开始开发一个应用/服务时&#xff0c;首先需要根据工程创建向导&#xff0c;创建一个新的工程&#xff0c;工具会自动生成对应的代码和资源模板。 说明 在运行DevEco Studio工程时&#xff0c;建议每一个运行窗口有2GB以上的可用内存空间。 创建和配置新工程 DevEco Studio提…

如何应对市场变革的战略利器之敏捷企业架构实践全景指南

敏捷与企业架构融合的必然性 在全球化和数字化的双重推动下&#xff0c;市场竞争的激烈程度前所未有。企业必须迅速适应市场的变化&#xff0c;以在激烈的竞争中脱颖而出。然而&#xff0c;传统的企业架构往往侧重于长期战略规划&#xff0c;尽管它在维持企业的稳定性方面功不…

阿贝云评测:免费虚拟主机与免费云服务器的优势对比

阿贝云作为一家知名云服务提供商&#xff0c;以其稳定可靠的服务质量在业界享有盛誉。其中&#xff0c;其免费虚拟主机和免费云服务器备受用户喜爱。在这篇评测中&#xff0c;我们将对这两种服务进行详细对比。 首先&#xff0c;就免费虚拟主机而言&#xff0c;阿贝云提供的免费…

图片工具箱:一键批量加水印,守护创意,提升效率!

前言 你是否曾在处理海量图片时&#xff0c;被繁琐的步骤和漫长的等待时间折磨得苦不堪言&#xff1f;是否梦想过拥有一款神器&#xff0c;能让你的图片处理工作变得轻松愉快&#xff0c;从此告别加班的烦恼&#xff0c;迎接升职加薪的曙光&#xff1f;那么&#xff0c;让我向…

我主编的电子技术实验手册(18)——认识电感

本专栏是笔者主编教材&#xff08;图0所示&#xff09;的电子版&#xff0c;依托简易的元器件和仪表安排了30多个实验&#xff0c;主要面向经费不太充足的中高职院校。每个实验都安排了必不可少的【预习知识】&#xff0c;精心设计的【实验步骤】&#xff0c;全面丰富的【思考习…

Golang学习笔记-Golang中的锁

同步原语和锁 Golang作为一个原生支持用户态的语言&#xff0c;当提到并发进程&#xff0c;多线程的时候&#xff0c;是离不开锁的&#xff0c;锁是一种并发编程中的同步原语&#xff08;Synchronization Primitives&#xff09;&#xff0c;它能保证多个 Goroutine 在访问同一…

数据库的范式

作用是减小表的冗余。 防止插入删除更新异常。 第一、第二、第三、第四、BC范式。并且不是越高越好。 第一范式&#xff1a;1NF的定义为&#xff1a;符合1NF的关系中的每个属性都不可再分。表1所示的情况&#xff0c;就不符合1NF的要求。 …

终端Tabby介绍和使用

介绍一款开源的跨平台终端模拟器&#xff0c;支持系统&#xff1a;Windows、macOS、Linux 下载安装 下载链接&#xff1a;Release Alpha 164 Eugeny/tabby GitHub 下载适合自己环境的版本&#xff1a; 选项说明 【1】Enable analytics选项的作用主要是允许Tabby收集和分析…

【数据集】遥感影像建筑物变化检测对比实验常用数据集分享

整理了几个变化检测的对比试验中常用的变化检测数据集&#xff08;建筑物&#xff09; LEVIR-CD 下载链接: https://justchenhao.github.io/LEVIR/ 数据介绍&#xff1a; 用于建筑物变化检测数据集 分辨率&#xff1a;0.5m 尺寸&#xff1a;1024*1024 数量&#xff1a;637组&…

python-求和again(赛氪OJ)

[题目描述] 最近小理遇到了麻烦的问题&#xff0c;你能帮帮他吗&#xff1f; 题目是这样的&#xff1a;计算 SUM(n)123...10^n 。输入格式&#xff1a; 输入包含多组数据&#xff0c;每组数据一行&#xff0c;包括一个整数 n 。当 n−1 时输入终止。输出格式&#xff1a; 对于每…