qt的血泪教训——地图存储与绘制

文章目录

  • 前言
  • [一. 项目背景](#一. 项目背景)
  • [二. 代码实现](#二. 代码实现)
    • [1. 指针数组创建](#1. 指针数组创建)
    • [2. gui坐标和数组的对应问题](#2. gui坐标和数组的对应问题)

前言

本文并不是严格的教程,只是笔者考试一败涂地后的碎碎念

一. 项目背景

从文件读入数据, 文件共有M + 1行, 数据格式如下:

text 复制代码
m n
a(0,0) a(0,1) ... a(0,n-1)
...
a(m-1,0) a(m-1,1) ... a(m-1,n-1)

m为地图的行数, n为地图的列数,

a(x,y)表示第x行第y列的数值:

  • =0, 用绿色表示
  • =1, 用红色表示

二. 代码实现

1. 指针数组创建

c++ 复制代码
class MainWindow : public QWidget
{
	...
	int **map;
	int m, n;
};
void MainWindow::load()
{
	map = new int*[m]; // m行
	for(int i = 0; i < m; ++i){
		//注意, 这里是对刚刚创建好的每一行进行操作, 所以循环到m
		map[i] = new int[n];
	}
}

2. gui坐标和数组的对应问题

然后就是本文的核心了, 也是最容易弄错的地方. 由于存地图的时候先存行数后存列数, 所以每个坐标(x, y)表示的是x行y列, 然而在绘图的时候用的是xy坐标, 也就是说, 横坐标对应的是列数, 纵坐标对应的是行数, 和存储正好相反. 因此paint函数应该如下:

c++ 复制代码
void MainWindow::paintEvent(QPaintEvent *event)
{
	QPainter painter(this);
	const int FILE_SIZE = 60;
	for(int y = 0; y < m; ++y){//先行,先y
		for(int x = 0; x < n; ++x){//后列,后x
			switch(map[x][y]){
			case 0:
			painter.fillRect(x * FILE_SIZE, y * FILE_SIZE, FILE_SIZE, FILE_SIZE, Qt::green);
				break;
			case 1:
			painter.fillRect(x * FILE_SIZE, y * FILE_SIZE, FILE_SIZE, FILE_SIZE, Qt::red);
				break;
			}	
		}
	}
}

否则绘制出来的图形会是轴对称的,长宽颠倒.

相关推荐
khalil102022 分钟前
代码随想录算法训练营Day-34动态规划03 | 01背包问题 二维、01背包问题 一维、416. 分割等和子集
数据结构·c++·算法·leetcode·动态规划·背包问题·01背包
前进吧-程序员30 分钟前
C++ 内存到底分配在哪?
java·jvm·c++
雾岛听蓝1 小时前
Qt操作指南:状态栏、浮动窗口与对话框使用
开发语言·经验分享·笔记·qt
兩尛1 小时前
c++面试常问1
jvm·c++·面试
点云侠1 小时前
隧道中线提取的优化方法
c++·算法·最小二乘法
汉克老师1 小时前
GESP2023年6月认证C++三级( 第二部分判断题(1-10))
c++·数组·位运算·进制·gesp三级·gesp3级
minji...1 小时前
Linux 线程同步与互斥(五) 日志,线程池
linux·运维·服务器·开发语言·c++·算法
6Hzlia1 小时前
【Hot 100 刷题计划】 LeetCode 108. 将有序数组转换为二叉搜索树 | C++ 分治法详解
c++·算法·leetcode
兩尛2 小时前
c++面试常问2
开发语言·c++·面试
re林檎2 小时前
八大排序算法(C++实现)
c++·算法·排序算法