《用MATLAB玩转游戏开发:从零开始打造你的数字乐园》基础篇(2D图形交互)-《打砖块:向量反射与实时物理模拟》MATLAB教程

《用MATLAB玩转游戏开发:从零开始打造你的数字乐园》基础篇(2D图形交互)-《打砖块:向量反射与实时物理模拟》MATLAB教程 🎮

文章目录

  • 《用MATLAB玩转游戏开发:从零开始打造你的数字乐园》基础篇(2D图形交互)-《打砖块:向量反射与实时物理模拟》MATLAB教程 🎮
    • 引言:从游戏到物理模拟
    • 1. 设计思路与游戏概述 🧠
      • 1.1 物理模型关键点 ⚛️
    • 2. 核心原理详解 🔍
      • 2.1 向量反射原理 📐
      • 2.2 碰撞检测原理 🔄
      • 2.3 游戏循环结构 🔁
    • 3. 完整流程图 📊
    • 4. 分步实现教程 🛠️
      • 4.1 初始化设置
      • 4.2 创建游戏对象
      • 4.3 游戏主循环
      • 4.4 碰撞检测函数
    • 5. 完整可运行代码 🏆
    • 6. 游戏操作说明 🎮
    • 7. 扩展思路 💡

在这里插入图片描述

引言:从游戏到物理模拟

还记得小时候玩过的打砖块游戏吗?🎯 一个小球在屏幕上弹来弹去,击碎各种砖块,同时玩家控制一块挡板防止小球掉落。这看似简单的游戏背后,其实蕴含着丰富的物理和数学原理!今天,我们就用MATLAB来实现这个经典游戏,并深入探讨其中的向量反射和实时物理模拟技术。

通过本教程,你将学到:

  • 向量运算在游戏物理中的应用
  • 实时碰撞检测的实现方法
  • MATLAB动画和交互式图形编程
  • 游戏状态管理和逻辑控制

准备好你的MATLAB环境(2016b版本),让我们开始这场有趣的编程之旅吧!🚀

1. 设计思路与游戏概述 🧠

打砖块游戏(Breakout/Arkanoid)是经典的街机游戏,包含以下核心元素:

  • 一个由玩家控制的挡板(paddle)
  • 一个运动的(ball)
  • 由砖块组成的(bricks)
  • 边界(walls)

游戏目标是用挡板反弹球,消除所有砖块。每次球碰到砖块,砖块消失,玩家得分。

1.1 物理模型关键点 ⚛️

  1. 碰撞检测:判断球与边界、挡板、砖块的接触
  2. 反射计算:球碰到物体后的运动方向变化
  3. 实时渲染:游戏画面的流畅更新
  4. 游戏逻辑:得分、生命、胜负判断

2. 核心原理详解 🔍

2.1 向量反射原理 📐

球碰到平面后的反射遵循"入射角=反射角"原则。数学上可以用向量运算表示:

给定:

  • 入射向量 v = [vx, vy]
  • 法向量 n = [nx, ny] (垂直于反射平面)

反射向量 r 的计算公式为:
r = v − 2 ( v ⋅ n ) n r = v - 2(v \cdot n)n r=v2(vn)n

在MATLAB中实现:

function reflectedVec = reflectVector(inVec, normalVec)% 归一化法向量normalVec = normalVec / norm(normalVec);% 计算反射向量reflectedVec = inVec - 2 * dot(inVec, normalVec) * normalVec;
end

2.2 碰撞检测原理 🔄

我们需要检测球与以下物体的碰撞:

  1. 边界:比较球心坐标与边界位置
  2. 挡板:判断球是否在挡板的矩形区域内
  3. 砖块:类似挡板,但需要考虑砖块是否已被消除

2.3 游戏循环结构 🔁

标准游戏循环包含以下步骤:

  1. 初始化:创建游戏对象和变量
  2. 输入处理:读取玩家操作
  3. 状态更新:计算物理和游戏逻辑
  4. 渲染:绘制游戏画面
  5. 循环控制:控制帧率和退出条件

3. 完整流程图 📊

开始
初始化游戏
绘制初始场景
游戏进行中?
处理玩家输入
更新球位置
检测碰撞
处理碰撞响应
更新游戏状态
绘制新帧
显示游戏结果
结束

4. 分步实现教程 🛠️

4.1 初始化设置

function breakoutGame()% 清除工作区和图形窗口clc; clear; close all;% 游戏参数设置gameParams = struct(...'paddleWidth', 100, ...    % 挡板宽度'paddleHeight', 15, ...    % 挡板高度'ballRadius', 10, ...      % 球半径'brickRows', 5, ...        % 砖块行数'brickCols', 10, ...       % 砖块列数'brickWidth', 60, ...      % 砖块宽度'brickHeight', 20, ...     % 砖块高度'brickOffsetTop', 50, ...  % 砖块顶部偏移'brickPadding', 5, ...     % 砖块间距'ballSpeed', 8, ...        % 球初始速度'lives', 3, ...            % 初始生命数'score', 0 ...             % 初始得分);% 创建图形窗口fig = figure('Name', 'MATLAB打砖块', ...'NumberTitle', 'off', ...'Position', [100, 100, 800, 600], ...'KeyPressFcn', @keyDown, ...'KeyReleaseFcn', @keyUp, ...'WindowButtonDownFcn', @mouseClick);% 创建坐标轴ax = axes('Parent', fig, ...'Position', [0.05, 0.05, 0.9, 0.9], ...'XLim', [0, 800], ...'YLim', [0, 600], ...'Color', [0.1, 0.1, 0.3], ...'XTick', [], ...'YTick', []);hold(ax, 'on');axis equal;

4.2 创建游戏对象

    % 创建挡板paddle = rectangle('Parent', ax, ...'Position', [350, 30, gameParams.paddleWidth, gameParams.paddleHeight], ...'FaceColor', [0.8, 0.2, 0.2], ...'EdgeColor', 'none', ...'Curvature', [0.2, 0.2]);% 创建球theta = rand * 2 * pi;  % 随机初始角度ballVel = [gameParams.ballSpeed * cos(theta), gameParams.ballSpeed * sin(theta)];ball = rectangle('Parent', ax, ...'Position', [400, 200, gameParams.ballRadius*2, gameParams.ballRadius*2], ...'FaceColor', [0.9, 0.9, 0.1], ...'EdgeColor', 'none', ...'Curvature', [1, 1]);% 创建砖块bricks = gobjects(gameParams.brickRows, gameParams.brickCols);brickColors = hsv(gameParams.brickRows);  % 每行不同颜色for r = 1:gameParams.brickRowsfor c = 1:gameParams.brickColsbrickX = (c-1) * (gameParams.brickWidth + gameParams.brickPadding);brickY = 550 - (r-1) * (gameParams.brickHeight + gameParams.brickPadding);bricks(r,c) = rectangle('Parent', ax, ...'Position', [brickX, brickY, gameParams.brickWidth, gameParams.brickHeight], ...'FaceColor', brickColors(r,:), ...'EdgeColor', 'w');endend% 创建文本显示scoreText = text(ax, 20, 580, sprintf('得分: %d', gameParams.score), ...'Color', 'w', 'FontSize', 12);livesText = text(ax, 700, 580, sprintf('生命: %d', gameParams.lives), ...'Color', 'w', 'FontSize', 12);startText = text(ax, 400, 300, '点击开始游戏', ...'Color', 'w', 'FontSize', 24, ...'HorizontalAlignment', 'center');

4.3 游戏主循环

    % 游戏状态变量gameState = struct(...'isRunning', false, ...    % 游戏是否进行中'paddleDir', 0, ...        % 挡板移动方向 (-1:左, 0:停止, 1:右)'paddleSpeed', 15, ...     % 挡板移动速度'activeBricks', true(gameParams.brickRows, gameParams.brickCols) ... % 活跃砖块);% 键盘控制回调函数function keyDown(~, event)switch event.Keycase 'leftarrow'gameState.paddleDir = -1;case 'rightarrow'gameState.paddleDir = 1;case 'escape'gameState.isRunning = false;endendfunction keyUp(~, event)switch event.Keycase {'leftarrow', 'rightarrow'}gameState.paddleDir = 0;endend% 鼠标点击开始游戏function mouseClick(~, ~)if ~gameState.isRunninggameState.isRunning = true;delete(startText);startText = [];endend% 主游戏循环while ishandle(fig)if gameState.isRunning% 更新挡板位置paddlePos = get(paddle, 'Position');newX = paddlePos(1) + gameState.paddleSpeed * gameState.paddleDir;% 限制挡板不超出边界newX = max(0, min(newX, 800 - gameParams.paddleWidth));set(paddle, 'Position', [newX, paddlePos(2), paddlePos(3), paddlePos(4)]);% 更新球位置ballPos = get(ball, 'Position');newBallX = ballPos(1) + ballVel(1);newBallY = ballPos(2) + ballVel(2);% 检测碰撞[ballVel, gameParams, gameState] = checkCollisions(...[newBallX, newBallY], ballVel, gameParams, gameState, paddle, bricks);% 更新球位置set(ball, 'Position', [newBallX, newBallY, ballPos(3), ballPos(4)]);% 更新文本显示set(scoreText, 'String', sprintf('得分: %d', gameParams.score));set(livesText, 'String', sprintf('生命: %d', gameParams.lives));% 检查游戏结束条件if newBallY < 0  % 球落到底部gameParams.lives = gameParams.lives - 1;if gameParams.lives <= 0gameState.isRunning = false;text(ax, 400, 300, '游戏结束!', ...'Color', 'r', 'FontSize', 36, ...'HorizontalAlignment', 'center');else% 重置球位置set(ball, 'Position', [400, 200, gameParams.ballRadius*2, gameParams.ballRadius*2]);theta = rand * 2 * pi;ballVel = [gameParams.ballSpeed * cos(theta), gameParams.ballSpeed * sin(theta)];pause(1);endend% 检查胜利条件if ~any(gameState.activeBricks(:))gameState.isRunning = false;text(ax, 400, 300, '恭喜通关!', ...'Color', 'g', 'FontSize', 36, ...'HorizontalAlignment', 'center');endend% 控制帧率pause(0.02);end

4.4 碰撞检测函数

function [newVel, gameParams, gameState] = checkCollisions(ballPos, ballVel, gameParams, gameState, paddle, bricks)% 获取球参数ballX = ballPos(1) + gameParams.ballRadius;ballY = ballPos(2) + gameParams.ballRadius;% 边界碰撞检测if ballX <= gameParams.ballRadius || ballX >= 800 - gameParams.ballRadiusballVel(1) = -ballVel(1);  % 水平反转endif ballY >= 600 - gameParams.ballRadiusballVel(2) = -ballVel(2);  % 垂直反转end% 挡板碰撞检测paddlePos = get(paddle, 'Position');if ballY <= paddlePos(2) + paddlePos(4) + gameParams.ballRadius && ...ballY >= paddlePos(2) && ...ballX >= paddlePos(1) - gameParams.ballRadius && ...ballX <= paddlePos(1) + paddlePos(3) + gameParams.ballRadius% 计算碰撞点相对于挡板中心的位置 (-1到1)hitPos = (ballX - (paddlePos(1) + paddlePos(3)/2)) / (paddlePos(3)/2);% 根据碰撞点调整反射角度maxAngle = pi/3;  % 最大反射角度 (60度)angle = hitPos * maxAngle;% 计算新速度向量speed = norm(ballVel);ballVel = [speed * sin(angle), speed * cos(angle)];% 增加一点速度让游戏更有挑战性ballVel = ballVel * 1.02;end% 砖块碰撞检测for r = 1:gameParams.brickRowsfor c = 1:gameParams.brickColsif gameState.activeBricks(r,c)brickPos = get(bricks(r,c), 'Position');% 检查球是否与砖块相交if ballX + gameParams.ballRadius > brickPos(1) && ...ballX - gameParams.ballRadius < brickPos(1) + brickPos(3) && ...ballY + gameParams.ballRadius > brickPos(2) && ...ballY - gameParams.ballRadius < brickPos(2) + brickPos(4)% 确定碰撞边 (简化版)if ballY < brickPos(2) || ballY > brickPos(2) + brickPos(4)ballVel(2) = -ballVel(2);  % 上下碰撞elseballVel(1) = -ballVel(1);  % 左右碰撞end% 标记砖块为不活跃并隐藏gameState.activeBricks(r,c) = false;set(bricks(r,c), 'Visible', 'off');% 增加分数gameParams.score = gameParams.score + 10;% 只需要处理一次碰撞break;endendendendnewVel = ballVel;
end

5. 完整可运行代码 🏆

将以下所有代码段按顺序组合成一个.m文件即可运行:

function breakoutGame()% 清除工作区和图形窗口clc; clear; close all;% 游戏参数设置gameParams = struct(...'paddleWidth', 100, ...    % 挡板宽度'paddleHeight', 15, ...    % 挡板高度'ballRadius', 10, ...      % 球半径'brickRows', 5, ...        % 砖块行数'brickCols', 10, ...       % 砖块列数'brickWidth', 60, ...      % 砖块宽度'brickHeight', 20, ...     % 砖块高度'brickOffsetTop', 50, ...  % 砖块顶部偏移'brickPadding', 5, ...     % 砖块间距'ballSpeed', 8, ...        % 球初始速度'lives', 3, ...            % 初始生命数'score', 0 ...             % 初始得分);% 创建图形窗口fig = figure('Name', 'MATLAB打砖块', ...'NumberTitle', 'off', ...'Position', [100, 100, 800, 600], ...'KeyPressFcn', @keyDown, ...'KeyReleaseFcn', @keyUp, ...'WindowButtonDownFcn', @mouseClick);% 创建坐标轴ax = axes('Parent', fig, ...'Position', [0.05, 0.05, 0.9, 0.9], ...'XLim', [0, 800], ...'YLim', [0, 600], ...'Color', [0.1, 0.1, 0.3], ...'XTick', [], ...'YTick', []);hold(ax, 'on');axis equal;% 创建挡板paddle = rectangle('Parent', ax, ...'Position', [350, 30, gameParams.paddleWidth, gameParams.paddleHeight], ...'FaceColor', [0.8, 0.2, 0.2], ...'EdgeColor', 'none', ...'Curvature', [0.2, 0.2]);% 创建球theta = rand * 2 * pi;  % 随机初始角度ballVel = [gameParams.ballSpeed * cos(theta), gameParams.ballSpeed * sin(theta)];ball = rectangle('Parent', ax, ...'Position', [400, 200, gameParams.ballRadius*2, gameParams.ballRadius*2], ...'FaceColor', [0.9, 0.9, 0.1], ...'EdgeColor', 'none', ...'Curvature', [1, 1]);% 创建砖块bricks = gobjects(gameParams.brickRows, gameParams.brickCols);brickColors = hsv(gameParams.brickRows);  % 每行不同颜色for r = 1:gameParams.brickRowsfor c = 1:gameParams.brickColsbrickX = (c-1) * (gameParams.brickWidth + gameParams.brickPadding);brickY = 550 - (r-1) * (gameParams.brickHeight + gameParams.brickPadding);bricks(r,c) = rectangle('Parent', ax, ...'Position', [brickX, brickY, gameParams.brickWidth, gameParams.brickHeight], ...'FaceColor', brickColors(r,:), ...'EdgeColor', 'w');endend% 创建文本显示scoreText = text(ax, 20, 580, sprintf('得分: %d', gameParams.score), ...'Color', 'w', 'FontSize', 12);livesText = text(ax, 700, 580, sprintf('生命: %d', gameParams.lives), ...'Color', 'w', 'FontSize', 12);startText = text(ax, 400, 300, '点击开始游戏', ...'Color', 'w', 'FontSize', 24, ...'HorizontalAlignment', 'center');% 游戏状态变量gameState = struct(...'isRunning', false, ...    % 游戏是否进行中'paddleDir', 0, ...        % 挡板移动方向 (-1:左, 0:停止, 1:右)'paddleSpeed', 15, ...     % 挡板移动速度'activeBricks', true(gameParams.brickRows, gameParams.brickCols) ... % 活跃砖块);% 键盘控制回调函数function keyDown(~, event)switch event.Keycase 'leftarrow'gameState.paddleDir = -1;case 'rightarrow'gameState.paddleDir = 1;case 'escape'gameState.isRunning = false;endendfunction keyUp(~, event)switch event.Keycase {'leftarrow', 'rightarrow'}gameState.paddleDir = 0;endend% 鼠标点击开始游戏function mouseClick(~, ~)if ~gameState.isRunninggameState.isRunning = true;delete(startText);startText = [];endend% 主游戏循环while ishandle(fig)if gameState.isRunning% 更新挡板位置paddlePos = get(paddle, 'Position');newX = paddlePos(1) + gameState.paddleSpeed * gameState.paddleDir;% 限制挡板不超出边界newX = max(0, min(newX, 800 - gameParams.paddleWidth));set(paddle, 'Position', [newX, paddlePos(2), paddlePos(3), paddlePos(4)]);% 更新球位置ballPos = get(ball, 'Position');newBallX = ballPos(1) + ballVel(1);newBallY = ballPos(2) + ballVel(2);% 检测碰撞[ballVel, gameParams, gameState] = checkCollisions(...[newBallX, newBallY], ballVel, gameParams, gameState, paddle, bricks);% 更新球位置set(ball, 'Position', [newBallX, newBallY, ballPos(3), ballPos(4)]);% 更新文本显示set(scoreText, 'String', sprintf('得分: %d', gameParams.score));set(livesText, 'String', sprintf('生命: %d', gameParams.lives));% 检查游戏结束条件if newBallY < 0  % 球落到底部gameParams.lives = gameParams.lives - 1;if gameParams.lives <= 0gameState.isRunning = false;text(ax, 400, 300, '游戏结束!', ...'Color', 'r', 'FontSize', 36, ...'HorizontalAlignment', 'center');else% 重置球位置set(ball, 'Position', [400, 200, gameParams.ballRadius*2, gameParams.ballRadius*2]);theta = rand * 2 * pi;ballVel = [gameParams.ballSpeed * cos(theta), gameParams.ballSpeed * sin(theta)];pause(1);endend% 检查胜利条件if ~any(gameState.activeBricks(:))gameState.isRunning = false;text(ax, 400, 300, '恭喜通关!', ...'Color', 'g', 'FontSize', 36, ...'HorizontalAlignment', 'center');endend% 控制帧率pause(0.02);end% 碰撞检测函数function [newVel, gameParams, gameState] = checkCollisions(ballPos, ballVel, gameParams, gameState, paddle, bricks)% 获取球参数ballX = ballPos(1) + gameParams.ballRadius;ballY = ballPos(2) + gameParams.ballRadius;% 边界碰撞检测if ballX <= gameParams.ballRadius || ballX >= 800 - gameParams.ballRadiusballVel(1) = -ballVel(1);  % 水平反转endif ballY >= 600 - gameParams.ballRadiusballVel(2) = -ballVel(2);  % 垂直反转end% 挡板碰撞检测paddlePos = get(paddle, 'Position');if ballY <= paddlePos(2) + paddlePos(4) + gameParams.ballRadius && ...ballY >= paddlePos(2) && ...ballX >= paddlePos(1) - gameParams.ballRadius && ...ballX <= paddlePos(1) + paddlePos(3) + gameParams.ballRadius% 计算碰撞点相对于挡板中心的位置 (-1到1)hitPos = (ballX - (paddlePos(1) + paddlePos(3)/2)) / (paddlePos(3)/2);% 根据碰撞点调整反射角度maxAngle = pi/3;  % 最大反射角度 (60度)angle = hitPos * maxAngle;% 计算新速度向量speed = norm(ballVel);ballVel = [speed * sin(angle), speed * cos(angle)];% 增加一点速度让游戏更有挑战性ballVel = ballVel * 1.02;end% 砖块碰撞检测for r = 1:gameParams.brickRowsfor c = 1:gameParams.brickColsif gameState.activeBricks(r,c)brickPos = get(bricks(r,c), 'Position');% 检查球是否与砖块相交if ballX + gameParams.ballRadius > brickPos(1) && ...ballX - gameParams.ballRadius < brickPos(1) + brickPos(3) && ...ballY + gameParams.ballRadius > brickPos(2) && ...ballY - gameParams.ballRadius < brickPos(2) + brickPos(4)% 确定碰撞边 (简化版)if ballY < brickPos(2) || ballY > brickPos(2) + brickPos(4)ballVel(2) = -ballVel(2);  % 上下碰撞elseballVel(1) = -ballVel(1);  % 左右碰撞end% 标记砖块为不活跃并隐藏gameState.activeBricks(r,c) = false;set(bricks(r,c), 'Visible', 'off');% 增加分数gameParams.score = gameParams.score + 10;% 只需要处理一次碰撞break;endendendendnewVel = ballVel;end
end

来看看我这个菜鸡玩了一局的效果,屏幕前的你也可以试试看看能的多少分~
在这里插入图片描述

6. 游戏操作说明 🎮

  • 左右箭头键:移动挡板
  • ESC键:退出游戏
  • 鼠标点击:开始游戏

7. 扩展思路 💡

如果你想进一步提升这个游戏,可以考虑:

  1. 增加音效:使用audioplayer添加碰撞音效
  2. 多种砖块:不同颜色砖块需要多次击中才能消除
  3. 特殊道具:球碰到某些砖块会掉落道具,如加长挡板、额外生命等
  4. 关卡设计:设计不同布局的砖块排列
  5. 粒子效果:砖块消除时添加爆炸效果

希望你喜欢这个MATLAB打砖块游戏教程!通过这个项目,你不仅学会了向量反射的原理,还掌握了实时物理模拟和游戏开发的基本技巧。Happy coding! 🚀

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

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

相关文章

Redisson 看门狗机制

何为看门狗 看门狗机制的主要作用是自动续期锁&#xff0c;确保在节点完成任务之前&#xff0c;锁不会过期。具体来说&#xff0c;当一个节点获取到锁后&#xff0c;看门狗会定期检查该锁的过期时间&#xff0c;并在必要时延长锁的过期时间&#xff0c;确保节点可以顺利完成任…

[架构之美]linux常见故障问题解决方案(十九)

[架构之美]linux下常见故障问题解决方案 一&#xff0c;文本文件忙 问题一&#xff1a;rootwh-VMware-Virtual-Platform:/home/hail# cp /root/containerd/bin/* /usr/bin/ cp: 无法创建普通文件 ‘/usr/bin/containerd’: 文本文件忙 在Linux系统中遇到“文本文件忙”错误时…

QT实现曲线图缩放、拖拽以及框选放大

.h文件 protected: void saveAxisRange();void wheelEvent(QWheelEvent *event) override;void mousePressEvent(QMouseEvent *event) override;void mouseMoveEvent(QMouseEvent *event) override;void mouseReleaseEvent(QMouseEvent *event) override;private:QPoint m_…

【Pandas】pandas DataFrame corr

Pandas2.2 DataFrame Computations descriptive stats 方法描述DataFrame.abs()用于返回 DataFrame 中每个元素的绝对值DataFrame.all([axis, bool_only, skipna])用于判断 DataFrame 中是否所有元素在指定轴上都为 TrueDataFrame.any(*[, axis, bool_only, skipna])用于判断…

青藏高原七大河流源区径流深、蒸散发数据集(TPRED)

时间分辨率 月空间分辨率 1km - 10km共享方式 开放获取数据大小 83.27 MB数据时间范围 1998-07-01 — 2017-12-31元数据更新时间 2024-07-22 数据集摘要 通过构建耦合积雪、冻土、冰川等冰冻圈水文物理过程的WEB-DHM模型&#xff08;Water and Energy Budget-based Distribute…

window环境下,如何通过USB接口控制打印机

虽然说大多数情况下&#xff0c;我们可以非常便利的通过打印机驱动来控制打印机&#xff0c;但还是有一些特殊情况&#xff0c;导致无法通过打印机驱动来完成我们预想的任务&#xff0c;比如&#xff0c;打印机只是一个系统设备中的一部分&#xff0c;需要协调其它设备一起工作…

CDGP数据治理主观题评分标准与得分策略

1.数据模型题目评分标准 1)准确理解题目中所描述的业务逻辑和需求得[1分] 2)正确使用模型设计方法,使用信息工程、信息建模集成定义、巴克符号、陈氏符号等其中一种得[1分] 3)正确设计实体和属性,题目中涉及的实体数量为25-30个,10个以内得[2分],10-20个得[3分],25个…

工业设计破局密码:3D 可视化技术点燃产业升级引擎

3D可视化是一种将数据、信息或抽象概念以三维图形、模型和动画的形式呈现出来的技术。3D可视化技术通过构建三维数字孪生体&#xff0c;将设计思维转化为可交互的虚拟原型&#xff0c;不仅打破了传统二维设计的空间局限&#xff0c;更在效率、精度与用户体验层面开创了全新维度…

Qt中在子线程中刷新UI的方法

Qt中在子线程中刷新UI的方法 在Qt中UI界面并不是线程安全的&#xff0c;意味着在子线程中不能随意操作UI界面组件&#xff08;比如按钮、标签&#xff09;等&#xff0c;如果强行操作这些组件有可能会导致程序崩溃。那么在Qt中如何在子线程中刷新UI控件呢&#xff1f; 两种方…

为了摸鱼和吃瓜,我开发了一个网站

平时上班真的比较累&#xff0c;摸鱼和吃瓜还要跳转多个平台的话&#xff0c;就累上加累了。 所以做了一个聚合了全网主流平台热搜的网站。 目前市面上确实有很多这种网站了&#xff0c;所以目前最主要有两点和他们不同&#xff1a; 给热搜列表增加了配图&#xff0c;刷的时候…

操作系统学习笔记第2章 (竟成)

第 2 章 进程管理 【考纲内容】 1.进程与线程&#xff1a; (1) 进程 / 线程的基本概念&#xff1b; (2) 进程 / 线程的状态与转换&#xff1b; (3) 线程的实现&#xff1a;内核支持的线程&#xff1b;线程库支持的线程&#xff1b; (4) 进程与线程的组织与控制&#xff1b; (5)…

77.评论日记

房间要经常搞卫生&#xff0c;不然会很多灰&#xff0c;很多头发&#xff0c;很多垃圾。 当然&#xff0c;即使一直搞卫生&#xff0c;在一些看不到的角落也是会慢慢囤积垃圾。 想要把那些角落也打扫干净&#xff0c;没别的办法&#xff0c;只有把那个角落上所有的东西都移开&a…

语音合成之十二 TTS声学编解码器的演进

TTS声学编解码器的演进 1 引言&#xff1a;声码器/声学编解码器在现代TTS中的关键作用2 奠定基石&#xff1a;从早期声码器到神经合成的曙光3. HiFi-GAN: 革新高效高保真波形生成4. 新的疆域&#xff1a;面向富语义TTS的先进声学编解码器5. XCodec2.0: 统一声学与语义信息6.BiC…

大学之大:悉尼科技大学2025.5.10

悉尼科技大学&#xff1a;从技术先驱到全球创新枢纽的百年征程 一、历史沿革&#xff1a;从技工培训到世界百强名校的蜕变 1. 工业革命的技术火种&#xff08;1843-1945&#xff09; 悉尼科技大学的历史可追溯至1843年成立的悉尼机械学院&#xff08;Sydney Mechanics’ Scho…

安装阿里云的yum源并且下载软件(CentOS7版本)

目录 1. 进入root模式: 2. 进入yum.repos.d文件下 3.备份 4. 安装阿里云的yum源 5. 安装dnf 6. 安装epel-release 7. 清除缓存,并新建缓存 8. 安装智能拼音软件包 8.1安装 8.2 进入应用程序 -- 系统工具 -- 设置 8.3重启后就可以打中文啦~ (需要重新启动才能)…

Discriminative and domain invariant subspace alignment for visual tasks

用于视觉任务的判别性和域不变子空间对齐 作者&#xff1a;Samaneh Rezaei&#xff0c;Jafar Tahmoresnezhad 文章于2018年12月4日收到&#xff0c;2019年5月24日被接受&#xff0c;2019年6月3日在线发表于Iran Journal of Computer Science期刊&#xff0c;DOI: 10.1007/s42…

用jsp简单实现C语言标准化测试系统

C语言标准化测试系统 在Web编程技术的学习过程中&#xff0c;我们小组为了深入理解相关技术原理&#xff0c;提升实践能力&#xff0c;开发了一个基于动态Web工程框架的C语言标准化考试系统。现在&#xff0c;就来和大家分享一下我们的项目经历。 一、实验目的剖析 这个项目…

QMK键盘固件自定义指南 - 打造你的专属键盘体验

QMK键盘固件自定义指南 - 打造你的专属键盘体验 &#x1f680; 前言 在机械键盘的世界里&#xff0c;QMK固件让你的键盘不再只是简单的输入设备&#xff0c;而是可以按照你的意愿定制的强大工具。本文将深入浅出地介绍如何自定义QMK键盘的行为&#xff0c;从基础概念到高级应…

5.9培训

文件上传 先找文件上传的地方&#xff0c;打开代理链接BP&#xff0c;它需要一个xls文件 我们创建一个sqzr.xls bp拦截了之后&#xff0c;我们修改请求&#xff0c;把后缀改成php&#xff0c;发送请求 找到我们的静态资源所在的位置 访问http://192.168.1.100:81/static/upload…

【FAQ】HarmonyOS SDK 闭源开放能力 — PDF Kit

1.问题描述&#xff1a; 预览PDF文件&#xff0c;文档上所描述的loadDocument接口&#xff0c;可以返回文件的状态&#xff0c;并无法实现PDF的预览&#xff0c;是否有能预览PDF相关接口&#xff1f; 解决方案&#xff1a; 1、执行loadDocument进行加载PDF文件后&#xff0c…