Qt/C++基于重力模拟的像素点水平堆叠效果

server/2024/11/28 22:15:26/

本文将深入解析一个基于 Qt/C++ 的像素点模拟程序。程序通过 重力作用,将随机分布的像素点下落并水平堆叠,同时支持窗口动态拉伸后重新计算像素点分布。


程序功能概述

  • 随机生成像素点:程序在初始化时随机生成一定数量的像素点,每个像素点有随机的初始位置和速度。
  • 重力模拟:像素点在重力作用下不断下落,并与地面发生弹性碰撞。
  • 水平堆叠:像素点在下落后静止并水平堆叠,模拟堆积效果。
  • 碰撞检测:像素点之间检测是否发生碰撞并进行弹性处理。
  • 窗口自适应:当窗口大小变化时,自动重新分布像素点。

核心实现细节

1. 像素点结构体

像素点通过 Pixel 结构体定义,包含以下属性:

  • xy:像素点的二维位置。
  • vxvy:像素点的水平和垂直速度。
  • isStatic:标记像素点是否已经静止。

定义如下:

struct Pixel {double x, y;       // 像素点位置double vx, vy;     // 像素点速度bool isStatic;     // 是否已经静止
};

2. 重力与碰撞模拟

重力计算

每个像素点的垂直速度受重力影响,逐帧增加重力值:

p.vy += gravity;

像素点的速度更新后,其位置随之更新:

p.x += p.vx;
p.y += p.vy;
边界检测

为了防止像素点越过窗口边界,加入边界检测与反弹逻辑:

if (p.x < radius || p.x > width() - radius) {p.vx *= -restitution; // 水平方向的反弹p.x = clamp(p.x, (double)radius, (double)(width() - radius));
}

在这里:

  • radius 是像素点的半径,防止越界。
  • restitution 控制反弹的弹性,值为 0-11 表示完全弹性,0 表示无弹性。
碰撞地面

当像素点接触地面时,垂直速度反弹,同时逐渐水平减速:

if (p.y > groundLevel) {p.vy *= -restitution; // 垂直方向反弹p.y = groundLevel;    // 防止越过地面p.vx *= damping;p.vy *= damping;
}
  • groundLevel 表示地面高度。
  • damping 表示速度阻尼系数,用于减小速度,模拟摩擦力。

3. 静止与堆叠机制

像素点在速度足够小的情况下会被标记为静止:

if (std::abs(p.vx) < staticThreshold && std::abs(p.vy) < staticThreshold) {p.vx = 0;p.vy = 0;p.isStatic = true;
}

静止后,像素点不再更新速度和位置,从而节省计算资源。

水平堆叠的网格机制

为了实现堆叠效果,使用了一个二维网格 grid 来记录像素点的静止位置:

std::vector<std::vector<bool>> grid(gridWidth, std::vector<bool>(gridHeight, false));

每个网格单元表示一个像素点的静止状态。当像素点静止后,更新对应的网格状态:

int gridX = p.x / (2 * radius);
int gridY = groundLevel / (2 * radius);
if (gridX >= 0 && gridX < gridWidth && gridY >= 0 && gridY < gridHeight) {grid[gridX][gridY] = true;
}

通过检测 grid 的状态,确保新下落的像素点不会与静止点重叠。


4. 碰撞处理

当两个像素点之间的距离小于其直径时,认为发生了碰撞。处理逻辑如下:

double distance = std::sqrt(dx * dx + dy * dy);
if (distance < 2 * radius && distance > 1e-6) { // 避免除以零double overlap = 2 * radius - distance;double nx = dx / distance;double ny = dy / distance;p.x += nx * overlap / 2;p.y += ny * overlap / 2;other.x -= nx * overlap / 2;other.y -= ny * overlap / 2;// 水平速度调整p.vx *= damping;other.vx *= damping;
}

这里:

  • overlap 表示像素点之间的重叠距离。
  • nxny 是碰撞方向的单位向量。
  • 调整每个像素点的位置以消除重叠,并减小速度。

5. 窗口自适应

窗口大小改变事件

当窗口大小改变时,重新初始化像素点,以适应新的窗口尺寸。通过重载 resizeEvent 实现:

void GravityWidget::resizeEvent(QResizeEvent *event) {initializePixels(); // 重新初始化像素点QWidget::resizeEvent(event);
}
像素点重新初始化

重新生成随机分布的像素点:

void GravityWidget::initializePixels() {pixels.clear();for (int i = 0; i < 200; ++i) {Pixel p;p.x = std::rand() % width();p.y = std::rand() % height();p.vx = 0;p.vy = 0;p.isStatic = false;pixels.push_back(p);}
}

6. 图形绘制

通过重载 paintEvent 实现像素点的绘制:

void GravityWidget::paintEvent(QPaintEvent *) {QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);painter.setBrush(Qt::blue);for (const auto &p : pixels) {painter.drawEllipse(QPointF(p.x, p.y), 3, 3); // 绘制小圆点}
}

运行效果

  1. 重力下落:随机生成的像素点在重力作用下逐渐下落。
  2. 弹性碰撞:像素点彼此碰撞时会弹开,最终逐渐稳定。
  3. 水平堆叠:像素点在地面静止后自然堆叠,形成类似水波的排列效果。
  4. 窗口自适应:拉伸或缩小窗口时,像素点会重新生成并分布。


http://www.ppmy.cn/server/145734.html

相关文章

学习HTML第三十三天

学习文章目录 一.fieldset 与 legend 的使用&#xff08;了解&#xff09;二.表单总结三.框架标签 一.fieldset 与 legend 的使用&#xff08;了解&#xff09; fieldset 可以为表单控件分组、 legend 标签是分组的标题 二.表单总结 form表单&#xff1a; action 属性&#…

记录下在html文件中如何直接使用npm依赖,以threejs为例

参考&#xff1a; https://www.cnblogs.com/shayloyuki/p/17191489.html 共三种方式 我的代码截图 方式一&#xff1a; threejsDemo_script.html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name&…

双十一线上服务调用链路追踪SkyWalking实战分析

序言 随着电商行业的飞速发展&#xff0c;双十一购物节已成为全球最大的购物狂欢节之一。在双十一期间&#xff0c;电商平台需要处理海量的用户请求和订单&#xff0c;这对系统的稳定性和性能提出了极高的要求。为了确保系统在高并发环境下的稳定运行&#xff0c;对线上服务的…

Electron文件写入、读取(作用:公共全局变量,本地存储)

Electron文件写入、读取&#xff08;作用&#xff1a;公共全局变量&#xff0c;本地存储&#xff09; 使用Node.js的fs模块 也可以直接使用Node.js的fs模块来实现本地文件的读写操作 // electron/main.jsconst fs require(fs);// 写入文件localhost.txt fs.writeFileSync(lo…

python实现TCP服务端,支持对所有客户端的数据收发,支持终端自定义命令操作,提供clear命令一键断开所有的客户端连接

前言 python实现TCP服务端&#xff0c;支持对所有客户端的数据收发&#xff0c;支持终端自定义命令操作&#xff0c;提供clear命令一键断开所有的客户端连接 简单易懂&#xff0c;直接上码 源码 import socket import threadingclass TCPServer:# 修改此处ip 端口def __ini…

Linux 命令总结

01. ls 指令 功能&#xff1a;对于目录&#xff0c;该命令列出该目录下的所有子目录与文件。对于文件&#xff0c;将列出文件名以及其他信息。 常用选项&#xff1a; -a 列出目录下的所有文件&#xff0c;包括以 . 开头的隐含文件。 -i 输出文件的 i 节点的索引信息。 如 ls –…

不需要双手离开键盘 vscode

目标是“不需要双手离开键盘”&#xff01; ctrl shift O 打开函数导航窗格 ctrl enter 行中换行 alt ↑/↓上下移行 shift alt ↑/↓上下复制 ctrl ←/→ 按代码块移动 ctrl delete / backspace按代码块删除 ctrl l 选择单行 shift delete 删除整行 ctrl C/V 复制/…

On-Chip-Network之routing

在确定网络拓扑之后&#xff0c;路由算法用来决定消息将通过网络的哪条路径到达目的地。 路由算法的目标是将流量均匀地分布在由网络拓扑提供的路径上&#xff0c;以avoid hotspots and minimize contention&#xff0c;从而减少网络延迟和提高吞吐量。所有这些性能目标必须在严…