Python笔记15-实战小游戏飞机大战(中)

文章目录

  • 创建第一个敌机
  • 创建一群敌机
  • 创建多行敌机
  • 让敌机移动
  • 射杀敌机
  • 生成新的敌机群
  • 结束游戏
  • 有敌机到达屏幕底端
  • 游戏结束

在上一篇基础上继续 本示例源码地址 点击下载

创建第一个敌机

在屏幕上放置外星人与放置飞船类似。每个外星人的行为都由Alien 类控制,我们将像创建Ship 类那样创建这个类。出于简化考虑,也将使用位图来表示外星人。你可以自己寻找表示外星人的图像 这里使用如下图像
在这里插入图片描述

创建Alien 类
除位置不同外,这个类的大部分代码与Ship 类相似。每个外星人最初都位于屏幕左上角附近。将每个外星人的左边距都设置为外星
人的宽度
Alien 类不需要一个在屏幕上绘制外星人的方法,因为我们将使用一个Pygame编组方法,自动在屏幕上绘制编组中的所有元素

import pygame
from pygame.sprite import Sprite
class Alien(Sprite):
    """表示单个外星人的类。"""
    def __init__(self, ai_game):
        """初始化外星人并设置其起始位置。"""
        super().__init__()
        self.screen = ai_game.screen
        # 加载外星人图像并设置其rect属性
        self.image = pygame.image.load('images/enemy2.png')
        self.rect = self.image.get_rect()
        # 每个外星人最初都在屏幕左上角附近。
        self.rect.x = self.rect.width
        self.rect.y = self.rect.height
        # 存储外星人的精确水平位置。
        self.x = float(self.rect.x)

我们最终会创建一群外星人,涉及的工作量不少,因此将新建一个名为_create_fleet() 的辅助方法。放在AlienInvasion 类的方法
init() 末尾

    def _create_fleet(self):
        """创建外星人群。"""
        # 创建一个外星人。
        alien = Alien(self)
        self.aliens.add(alien)
def __init__(self):
    """初始化游戏并创建游戏资源。"""
    pygame.init()
    self.settings = Settings()
    #全屏模式代码
    # self.screen = pygame.display.set_mode((0, 0), pygame.FULLSCREEN)
    # self.settings.screen_width = self.screen.get_rect().width
    # self.settings.screen_height = self.screen.get_rect().height
    #非全屏模式
    self.screen = pygame.display.set_mode((self.settings.screen_width, self.settings.screen_height))
    pygame.display.set_caption("Alien Invasion")
    self.ship = Ship(self)
    self.bullets = pygame.sprite.Group()
    self.aliens = pygame.sprite.Group()
    self._create_fleet()

要让外星人现身,需要在_update_screen() 中对外星人编组调用方法draw() ,对编组调用draw() 时,Pygame将把编组中的每个元素绘制到属性rect 指定的位置。方法draw() 接受一个参数,这个参数指定了要将编组中的元素绘制到哪个surface上

   def _update_screen(self):
        """更新屏幕上的图像,并切换到新屏幕。"""
        self.screen.fill(self.settings.bg_color)
        self.ship.blitme()
        for bullet in self.bullets.sprites():
            bullet.draw_bullet()
        self.aliens.draw(self.screen)
        pygame.display.flip()

运行程序看到了敌人的飞船
在这里插入图片描述

创建一群敌机

为确定一行可容纳多少个外星人,来看看可用的水平空间有多大。屏幕宽度存储在settings.screen_width 中,但需要在屏幕两边都留下一定的边距(将其设置为外星人的宽度)。因为有两个边距,所以可用于放置外星人的水平空间为屏幕宽度减去外星人宽度的两倍:

available_space_x = settings.screen_width – (2 * alien_width)
number_aliens_x = available_space_x // (2 * alien_width)

重写_create_fleet() 使其创建一行敌机

 def _create_fleet(self):
        """创建外星人群。"""
        # 创建一个外星人并计算一行可容纳多少个外星人。
        # 外星人的间距为外星人宽度。
        alien = Alien(self)
        alien_width = alien.rect.width
        available_space_x = self.settings.screen_width - (2 * alien_width)
        number_aliens_x = available_space_x // (2 * alien_width)
        # 创建第一行外星人。
        for alien_number in range(number_aliens_x):
            # 创建一个外星人并将其加入当前行。
            alien = Alien(self)
            alien.x = alien_width + 2 * alien_width * alien_number
            alien.rect.x = alien.x
            self.aliens.add(alien)

运行可以看到创建了一行敌机
在这里插入图片描述

创建多行敌机

要创建外星人群,需要计算屏幕可容纳多少行,并将创建一行外星人的循环重复执行相应的次数。为计算可容纳的行数,要先计算可用的垂直空间:用屏幕高度减去第一行外星人的上边距(外星人高度)、飞船的高度以及外星人群最初与飞船之间的距离(外星人高度的两倍):
available_space_y = settings.screen_height – (3 * alien_height) – ship_height
每行下方都要留出一定的空白区域,不妨将其设置为外星人的高度。为计算可容纳的行数,将可用的垂直空间除以外星人高度的两倍。我们使用整除,因为行数只能是整数。

number_rows = available_space_y // (2 * alien_height)

重构_create_fleet 方法

    def _create_fleet(self):
        alien = Alien(self)
        alien_width, alien_height = alien.rect.size
        available_space_x = self.settings.screen_width - (2 * alien_width)
        number_aliens_x = available_space_x // (2 * alien_width)
        # 计算屏幕可容纳多少行外星人。
        ship_height = self.ship.rect.height
        available_space_y = (self.settings.screen_height -
                               (3 * alien_height) - ship_height)
        number_rows = available_space_y // (2 * alien_height)
        print("number_rows=",number_rows)
        # 创建外星人群。
        for row_number in range(number_rows):
            for alien_number in range(number_aliens_x):
                self._create_alien(alien_number, row_number)

    def _create_alien(self, alien_number, row_number):
        """创建一个外星人,并将其放在当前行。"""
        alien = Alien(self)
        alien_width, alien_height = alien.rect.size
        alien.x = alien_width + 2 * alien_width * alien_number
        alien.rect.x = alien.x
        alien.rect.y = alien.rect.height + 2 * alien.rect.height * row_number
        self.aliens.add(alien)

为了效果敌机图片小一点,运行效果如下
在这里插入图片描述

让敌机移动

下面来让外星人群在屏幕上向右移动,撞到屏幕边缘后下移一定的量,再沿相反的方向移动。我们将不断移动所有的外星人,直到外星人被全部消灭,或者有外星人撞上飞船或抵达屏幕底端。下面先让外星人向右移动。
settings.py添加一个控制外星人速度的设置

def __init__(self):
--snip--
# 外星人设置
self.alien_speed = 1.0

使用这个设置来实现update()

 def update(self):
        """向右移动外星人。"""
        self.x += self.settings.alien_speed
        self.rect.x = self.x

主while 循环中调用更新每个外星人位置的方法

    def run_game(self):
        """开始游戏的主循环"""
        while True:
            #检测事件
            self._check_events()
            #更新飞船位置
            self.ship.update()
            #更新子弹
            """
            对编组调用update() 时,编组自动对其中的每个精灵调
            用update() 。因此代码行bullets.update() 将为编组bullets
            中的每颗子弹调用bullet.update()"""
            self._update_bullets()

            self._update_aliens()
            # 每次循环时都重绘屏幕。

            self._update_screen()
            time.sleep(0.05) #设置每50ms刷新一次

    def _update_aliens(self):
        """更新外星人群中所有外星人的位置。"""
        self.aliens.update()

对编组调用方法update() ,这将自动对每个外星人调用方法update() 。如果现在运行这个游戏,你将看到外星人群向右移动,并在屏幕右边缘消失。
创建让外星人撞到屏幕右边缘后向下移动、再向左移动的设置。实现这种行为的代码如下:

  # 外星人设置
        self.alien_speed = 1.0
        self.fleet_drop_speed = 10
        # fleet_direction为1表示向右移,为-1表示向左移。
        self.fleet_direction = 1

要编写一个方法来检查外星人是否撞到了屏幕边缘,还需修改update() 让每个外星人都沿正确的方向移动

def check_edges(self):
    """如果外星人位于屏幕边缘,就返回True。"""
    screen_rect = self.screen.get_rect()
    if self.rect.right >= screen_rect.right or self.rect.left <= 0:
        return True
def update(self):
    """向左或向右移动外星人。"""
    self.x += (self.settings.alien_speed *
    self.settings.fleet_direction)
    self.rect.x = self.x

有外星人到达屏幕边缘时,需要将整群外行星下移,并改变它们的移动方向,为此,需要在AlienInvasion 中添加一些代码,因为
要在这里检查是否有外星人到达了左边缘或右边缘。我们编写方法_check_fleet_edges() 和_change_fleet_direction() ,并且修改_update_aliens() 。这些新方法将放在_create_alien() 后面

    def _check_fleet_edges(self):
        """有外星人到达边缘时采取相应的措施。"""
        for alien in self.aliens.sprites():
            if alien.check_edges():
                    self._change_fleet_direction()
                    break
   def _change_fleet_direction(self):
        """将整群外星人下移,并改变它们的方向。"""
        for alien in self.aliens.sprites():
            alien.rect.y += self.settings.fleet_drop_speed
        self.settings.fleet_direction *= -1

   def _update_aliens(self):
        """
        检查是否有外星人位于屏幕边缘,
        并更新整群外星人的位置。
        """
        self._check_fleet_edges()
        self.aliens.update()

运行之后 外星人自动移动

射杀敌机

我们创建了飞船和外星人群,但子弹击中外星人时将穿过外星人,因为还没有检查碰撞。在游戏编程中,碰撞 指的是游戏元素重叠在一起。要让子弹能够击落外星人,我们将使用sprite.groupcollide() 检测两个编组的成员之间的碰撞。
子弹击中外星人时,我们需要马上知道,以便碰撞发生后让子弹立即消失。为此,我们将在更新子弹的位置后立即检测碰撞。
函数sprite.groupcollide() 将一个编组中每个元素的rect 同另一个编组中每个元素的rect 进行比较。在这里,是将每颗子弹的rect 同每个外星人的rect 进行比较,并返回一个字典,其中包含发生了碰撞的子弹和外星人。在这个字典中,每个键都是一颗子弹,而关联的值是被该子弹击中的外星人。在方法_update_bullets() 末尾,添加如下检查子弹和外星人碰撞的代码:

def _update_bullets(self):
	"""更新子弹的位置,并删除消失的子弹。"""
	--snip--
	# 检查是否有子弹击中了外星人。
	# 如果是,就删除相应的子弹和外星人。
	collisions = pygame.sprite.groupcollide(self.bullets, self.aliens, True, True)

这些新增的代码将self.bullets 中所有的子弹都与self.aliens中所有的外星人进行比较,看它们是否重叠在一起。每当有子弹和外星人的rect 重叠时,groupcollide() 就在它返回的字典中添加一个键值对。两个实参True 让Pygame删除发生碰撞的子弹和外星人。(要模拟能够飞行到屏幕顶端、消灭击中的每个外星人的高能子弹,可将第一个布尔实参设置为False ,并保留第二个布尔参数为True 。这样被击中的外星人将消失,但所有的子弹都始终有效,直到抵达屏幕顶端后消失。)

此时运行这个游戏,被击中的外星人将消失。如图13-5所示,有些外星人被射杀了
在这里插入图片描述

生成新的敌机群

外星人无穷无尽:一群外星人被消灭后,又会出现另一群外星人。要在一群外星人被消灭后再显示一群外星人,首先需要检查编组aliens 是否为空。如果是,就调用_create_fleet() 。我们将在_update_bullets() 末尾执行这项任务

def _update_bullets(self):
	--snip--
	if not self.aliens:
	# 删除现有的子弹并新建一群外星人。
	self.bullets.empty()
	self._create_fleet()

如果现在尝试在游戏中射杀外星人,可能会发现子弹的速度不太合适(有点快或有点慢),游戏感不好。当前,可通过修改设置让这
款游戏更有意思、更好玩。要修改子弹的速度,可调整settings.py中bullet_speed 的值。在我的系统中,我把bullet_speed 的值调整到1.5,让子弹的速度快些

结束游戏

如果玩家没能在足够短的时间内将整群外星人消灭干净,导致有外星人撞到了飞船或抵达屏幕底端,飞船将被摧毁。与此同时,限制玩家可使用的飞船数,在玩家用光所有的飞船后游戏将结束。
首先检查外星人和飞船之间的碰撞,以便在外星人撞上飞船时做出合适的响应。为此,在AlienInvasion 中更新每个外星人的位置后,立即检测外星人和飞船之间的碰撞:

def _update_aliens(self):
	--snip--
	self.aliens.update()
	# 检测外星人和飞船之间的碰撞。
	if pygame.sprite.spritecollideany(self.ship, self.aliens):
	print("Ship hit!!!")

下面来编写一个用于跟踪游戏统计信息的新类GameStats ,并将其保存为文件game_stats.py,当外星人与飞船发生碰撞时该做些什么。们不销毁Ship 实例并创建新的,而是通过跟踪游戏的统计信息来记录飞船被撞了多少次

class GameStats:
    """跟踪游戏的统计信息。"""
    def __init__(self, ai_game):
        """初始化统计信息。"""
        self.settings = ai_game.settings
        self.reset_stats()
    def reset_stats(self):
        """初始化在游戏运行期间可能变化的统计信息。"""
        self.ships_left = self.settings.ship_limit

在游戏运行期间,只创建一个GameStats 实例,但每当玩家开始新游戏时,需要重置一些统计信息。为此,在方法reset_stats()中初始化大部分统计信息,而不是在__init__() 中直接初始化。我们在__init__() 中调用这个方法,这样创建GameStats 实例时将妥善地设置这些统计信息,在玩家开始新游戏时也能调用reset_stats() 。
一开始玩家拥有的飞船数存储在settings.py的ship_limit中:

self.ship_limit = 3

还需对alien_invasion.py做些修改,以创建一个GameStats 实例。

from game_stats import GameStats

从Python标准库的模块time 中导入函数sleep() ,以便在飞船被外星人撞到后让游戏暂停片刻。我们还导入了GameStats 。

def __init__(self):
	--snip--
	self.screen = pygame.display.set_mode(	(self.settings.screen_width, self.settings.screen_height))
	pygame.display.set_caption("Alien Invasion")
	# 创建一个用于存储游戏统计信息的实例。
	self.stats = GameStats(self)
	self.ship = Ship(self)
	--snip--

有外星人撞到飞船时,将余下的飞船数减1,创建一群新的外星人,并将飞船重新放到屏幕底端的中央。另外,让游戏暂停片刻,让玩家在新外星人群出现前注意到发生了碰撞并将重新创建外星人群。实现这些功能的大部分代码放到新方法_ship_hit() 中

    def _update_aliens(self):
        """
        检查是否有外星人位于屏幕边缘,
        并更新整群外星人的位置。
        """
        self._check_fleet_edges()
        self.aliens.update()
        if pygame.sprite.spritecollideany(self.ship, self.aliens):
            self._ship_hit()

def _ship_hit(self):
    """响应飞船被外星人撞到。"""
    # 将ships_left减1。
    self.stats.ships_left -= 1
    # 清空余下的外星人和子弹。
    self.aliens.empty()
    self.bullets.empty()
    # 创建一群新的外星人,并将飞船放到屏幕底端的中央。
    self._create_fleet()
    self.ship.center_ship()
    # 暂停。
    time.sleep(0.5)

新方法center_ship() ,请将其添加到ship.py的末尾

   def center_ship(self):
        """让飞船在屏幕底端居中。"""
        self.rect.midbottom = self.screen_rect.midbottom
        self.x = float(self.rect.x)

有敌机到达屏幕底端

如果有外星人到达屏幕底端,我们将像有外星人撞到飞船那样做出响应。为检测这种情况,在alien_invasion.py中添加一个新方法:

    def _check_aliens_bottom(self):
        """检查是否有外星人到达了屏幕底端。"""
        screen_rect = self.screen.get_rect()
        for alien in self.aliens.sprites():
            if alien.rect.bottom >= screen_rect.bottom:
                # 像飞船被撞到一样处理。
                self._ship_hit()
                break

们在_update_aliens() 中调用_check_aliens_bottom()

def _update_aliens(self):
    """
    检查是否有外星人位于屏幕边缘,
    并更新整群外星人的位置。
    """
    self._check_fleet_edges()
    self.aliens.update()
    if pygame.sprite.spritecollideany(self.ship, self.aliens):
        self._ship_hit()
    self._check_aliens_bottom()

游戏结束

现在这个游戏看起来更完整了,但它永远都不会结束,只是ships_left 不断变成越来越小的负数。下面在GameStats 中添加一个作为标志的属性game_active ,以便在玩家的飞船用完后结束游戏。首先,在GameStats 类的方法__init__() 末尾设置这

def __init__(self, ai_game):
	--snip--
	# 游戏刚启动时处于活动状态。
	self.game_active = True

_ship_hit() 中添加代码,在玩家的飞船用完后将game_active 设置为False

def _ship_hit(self):
    """响应飞船被外星人撞到。"""
    # 将ships_left减1。
    if self.stats.ships_left>0:
        self.stats.ships_left -= 1
        # 清空余下的外星人和子弹。
        self.aliens.empty()
        self.bullets.empty()
        # 创建一群新的外星人,并将飞船放到屏幕底端的中央。
        self._create_fleet()
        self.ship.center_ship()
        # 暂停。
        time.sleep(0.5)
    else:
        self.stats.game_active=False

我们需要确定游戏的哪些部分在任何情况下都应运行,哪些部分仅在游戏处于活动状态时才运行:
在主循环中,在任何情况下都需要调用_check_events() ,即便游戏处于非活动状态。例如,我们需要知道玩家是否按了Q 键以退出游戏,或者是否单击了关闭窗口的按钮。我们还需要不断更新屏幕,以便在等待玩家是否选择开始新游戏时修改屏幕。其他的函数仅在游戏处于活动状态时才需要调用,因为游戏处于非活动状态时,不用更新游戏元素的位置。

    def run_game(self):
        """开始游戏的主循环"""
        while True:
            #检测事件
            self._check_events()
            if self.stats.game_active:
                #更新飞船位置
                self.ship.update()
                #更新子弹
                """
                对编组调用update() 时,编组自动对其中的每个精灵调
                用update() 。因此代码行bullets.update() 将为编组bullets
                中的每颗子弹调用bullet.update()"""
                self._update_bullets()
                self._update_aliens()
            # 每次循环时都重绘屏幕。

            self._update_screen()
            time.sleep(0.05) #设置每50ms刷新一次

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++面试宝典第25题:阶乘末尾零的个数

题目 给定一个整数n,返回n!(n的阶乘)结果尾数中零的个数。 示例 1: 输入:3 输出:0 解释:3! = 6,尾数中没有零。 示例 2: 输入:5 输出:1 解释:5! = 120,尾数中有1个零。 解析 这道题主要考察应聘者对于数学问题的分析和理解能力,以及在多个解决方案中,寻求最优…

从零学习Linux操作系统 第二十部分 mariadb数据库的管理

一、对于数据库的基本介绍 1.什么是数据库 数据库就是个高级的表格软件 2.常见数据库 Mysql Oracle mongodb db2 sqlite sqlserver … 3.Mysql (SUN -----> Oracle) 4.mariadb (Mysql的一种&#xff09; 数据库中的常用名词 1.字段 &#xff1a;表格中的表头 2.表 &…

MySQL原理(一)架构组成(1)物理文件组成

目录 一、日志文件 1、错误日志 Error Log 1.1、作用&#xff1a; 1.2、开启关闭&#xff1a; 1.3、使用 2、二进制日志 Binary Log & Binary Log Index 2.1、作用&#xff1a; 2.2、开启关闭&#xff1a; 2.3、Binlog还有一些附加选项参数 &#xff08;1&#x…

嵌入式C++必知必会之基础知识-常用关键字(1)

温故而知新 Hello&#xff0c;大家好&#xff01;我是木荣。温故而知新&#xff0c;可以为师矣。作为一名攻城狮&#xff0c;扎实的基本功是解决问题及完成工作中任务的重要前提。没有良好的基本功作为铺垫&#xff0c;一味的追求知识的宽度是毫无意义&#xff0c;知其然更要知…

野火霸道V2学习笔记

STM32F103学习笔记 STM32F103学习笔记说明基础配置配置KeilMDK配置串口下载程序美化Keil界面配置VScode 理论知识STM32命名方式例子 置位与清零GPIOGPIO简介GPIO和引脚的区别引脚的分类 GPIO 框图讲解保护二极管推挽输出推挽输出的含义推挽输出的原理推挽输出的特点推挽输出的应…

力扣题目训练(3)

2024年1月27日力扣题目训练 2024年1月27日力扣题目训练290. 单词规律292. Nim 游戏303. 区域和检索 - 数组不可变91. 解码方法92. 反转链表 II41. 缺失的第一个正数 2024年1月27日力扣题目训练 2024年1月27日第三天编程训练&#xff0c;今天主要是进行一些题训练&#xff0c;包…

大数据Doris(六十一):SQL函数之Bitmap函数

文章目录 SQL函数之Bitmap函数 一、BITMAP_AND(BITMAP lhs, BITMAP rhs)

用C#实现最小二乘法(用OxyPlot绘图)

最小二乘法介绍✨ 最小二乘法&#xff08;Least Squares Method&#xff09;是一种常见的数学优化技术&#xff0c;广泛应用于数据拟合、回归分析和参数估计等领域。其目标是通过最小化残差平方和来找到一组参数&#xff0c;使得模型预测值与观测值之间的差异最小化。 最小二…

小白都会的幻兽帕鲁服务器搭建教程(详细图文)

​  简介&#xff1a;由于幻兽帕鲁游戏的火爆&#xff0c;导致其官方服务器频现游戏卡顿掉线&#xff0c;为了能够正常流畅的体验幻兽帕鲁&#xff0c;有不少人都搭建了幻兽帕鲁服务器(私服)&#xff0c;网上虽然也有很多幻兽帕鲁服务器搭建教程&#xff0c;但内容专业性有点…

qq通讯录怎么关闭?QQ好友删除了怎么恢复?

在QQ中&#xff0c;通讯录是我们管理好友和进行聊天的重要工具&#xff0c;但有时候我们可能需要一些隐私保护&#xff0c;不让一些用户通过手机通讯录添加自己。如果您正在思考qq通讯录怎么关闭以及恢复意外删除的好友&#xff0c;本文将为您详细介绍如何关闭QQ通讯录和恢复被…

Java开发秘籍:实战公众号开发指南

目录 第一章&#xff1a;公众号开发简介 1.1 什么是公众号开发 1.2 公众号开发的意义和应用场景 1.3 公众号开发的技术要求和基础知识 第二章&#xff1a;搭建公众号开发环境 2.1 配置Java开发环境 2.2 下载和安装开发工具 2.3 设置公众号开发所需的配置文件 第三章&a…

2024阿里云和腾讯云的第一战打响:搭建《幻兽帕鲁》私服游戏

为了搭建《幻兽帕鲁》游戏私服&#xff0c; 2024年阿里云 VS 腾讯云的第一场战争开始了…… 事情是这样的&#xff1a; 1月19日&#xff0c;最离谱新游 《幻兽帕鲁》突然爆火了&#xff0c;这是一款日本开发商展耗费4年开发的冒险类游戏&#xff0c;这款戏一推出就迅速俘获了…

优思学院|精益管理如何判定哪些活动是增值或非增值?

“时间就是金钱”——这句老话我们都耳熟能详。但在工作中&#xff0c;我们真正从事的、对组织增加价值的活动有多少呢&#xff1f;我们常常认为自己的每一项任务都是维持运营的关键。然而&#xff0c;当我们从精益管理的视角进行分析&#xff0c;可能会惊讶地发现&#xff0c;…

蓝牙----蓝牙连接建立_连接建立

蓝牙----蓝牙连接建立_连接建立 蓝牙连接建立过程图1.主机扫描到广播包1.1判断是否是自己关心的广播包1.2广播地址添加到扫描列表 2.主机扫描结束&#xff0c;建立连接3.主从连接成功后&#xff0c;执行连接建立后事件3.1.主机将连接句柄和设备地址添加到连接列表3.2.主机进行G…

spring boot学习第八篇:操作elastic search的索引和索引中的数据

前提参考&#xff1a;elastic search入门-CSDN博客 前提说明&#xff1a;已经安装好了elastic search 7.x版本&#xff0c;我的es版本是7.11.1 1、 pom.xml文件内容如下&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns&q…

C++ QT入门1——记事本基础功能实现(基本控件布局+信号与槽+文件类操作)

C QT入门1——记事本基础功能实现&#xff08;基本控件布局信号与槽文件类操作&#xff09; 一、UI界面的基础设置样式表通用配置通过样式表设置按钮三态以图片作为按钮背景 二、按键响应——☆信号与槽信号与槽基本概念按键QPushButton设置信号与槽自定义信号与槽自定义信号与…

【TCP】重传与超时机制

前言 在网络通信的世界里&#xff0c;传输控制协议&#xff08;TCP&#xff09;扮演着一个至关重要的角色。它确保了数据的可靠传输&#xff0c;就像邮差确保每一封信都能准确无误地送达收件人手中一样。但是&#xff0c;网络环境充满了不确定性&#xff0c;数据包可能会因为各…

云堡垒机是软件堡垒机吗?是一种产品吗?

很多小伙伴傻傻分不清楚云堡垒机和软件堡垒机&#xff1f;在问云堡垒机是软件堡垒机吗&#xff1f;是一种产品吗&#xff1f;两者有区别吗&#xff1f;今天我们大家就来一起探讨下&#xff0c;仅供参考哦&#xff01; 首先我们需要知道的是&#xff0c;云堡垒机和软件堡垒机是两…

数字美妆技术:美颜SDK和动态贴纸技术的崭新时代

数字美妆的兴起标志着人们对于自身形象的追求不再局限于现实生活&#xff0c;而是延伸到了虚拟世界。同时&#xff0c;美颜SDK的动态贴纸技术也开始进入到大家的视野之中。 一、美颜SDK&#xff1a;技术之作 通过复杂的图像处理算法&#xff0c;美颜SDK能够实时检测人脸&…

AWS 专题学习 P10 (Databases、 Data Analytics)

文章目录 专题总览1. Databases1.1 选择合适的数据库1.2 数据库类型1.3 AWS 数据库服务概述Amazon RDSAmazon AuroraAmazon ElastiCacheAmazon DynamoDBAmazon S3DocumentDBAmazon NeptuneAmazon Keyspaces (for Apache Cassandra)Amazon QLDBAmazon Timestream 2. Data & …