Unity 设计模式 之 行为型模式 -【状态模式】【观察者模式】【备忘录模式】

Unity 设计模式 之 行为型模式 -【状态模式】【观察者模式】【备忘录模式】

目录

Unity 设计模式 之 行为型模式 -【状态模式】【观察者模式】【备忘录模式】

一、简单介绍

二、状态模式(State Pattern)

1、什么时候使用状态模式

2、使用状态模式的好处

3、使用状态模式时的注意事项

三、在 Unity 中使用 状态模式

1、定义状态接口

2、具体状态类

3、上下文类

4、使用场景中的角色

5、在 Unity 中运行

6、示例分析

四、观察者模式(Observer Pattern)

1、什么时候使用观察者模式

2、使用观察者模式的好处

3、使用时的注意事项

五、在 Unity 中使用 观察者模式

1、定义观察者接口

2、定义被观察者类

3、实现观察者类

4、在场景中使用观察者模式

5、运行示例

6、示例分析

六、备忘录模式(Memento Pattern)

1、什么时候使用备忘录模式

2、使用备忘录模式的好处

3、使用时的注意事项

七、在 Unity 中使用 备忘录模式

1、定义备忘录类

2、定义发起人类

3、定义管理员类

4、在 Unity 中测试

5、运行示例

6、示例分析


一、简单介绍

设计模式 是指在软件开发中为解决常见问题而总结出的一套 可复用的解决方案。这些模式是经过长期实践证明有效的 编程经验总结,并可以在不同的项目中复用。设计模式并不是代码片段,而是对常见问题的 抽象解决方案,它提供了代码结构和模块间交互的一种设计思路,帮助开发者解决特定的设计问题。

设计模式的特点:

  1. 通用性:设计模式针对的是软件开发中常见的设计问题,适用于各种软件工程项目。
  2. 可复用性:设计模式可以在不同项目和环境下被重复使用,提高代码的可维护性和扩展性。
  3. 可扩展性:设计模式有助于让代码结构更加灵活,易于扩展和修改。
  4. 模块化:通过设计模式,可以减少代码的耦合性,增强模块间的独立性。
  5. 提高沟通效率:设计模式为开发者提供了一种通用的设计语言,使得团队成员能够快速理解并讨论设计方案。

二、状态模式(State Pattern)

状态模式(State Pattern) 是一种行为型设计模式,它允许一个对象在其内部状态发生改变时改变其行为。状态模式将与状态相关的行为封装在独立的状态类中,系统在运行时根据状态的变化来切换不同的行为。

通过状态模式,状态转换和行为执行得到了很好的分离,符合面向对象设计的单一职责原则,即每个类只负责一项具体的职责。

状态模式的结构

  1. 上下文类(Context Class):维护当前状态的引用,负责在运行时根据状态的变化调用不同的状态类的行为。
  2. 状态接口(State Interface):定义状态类的共同行为,这通常是一个抽象类或接口。
  3. 具体状态类(Concrete State Class):实现状态接口,提供每个状态下具体的行为。

1、什么时候使用状态模式

  1. 对象的行为依赖于状态变化时,并且行为会随着状态的不同而发生变化。

  2. 对象的状态变化频繁,而且状态之间的切换逻辑复杂,状态模式可以很好地管理这些状态。

  3. 状态转换具有规律性,每个状态都有固定的行为模式或规则,状态模式可以简化这种逻辑。

  4. 需要避免使用大量条件判断来实现不同状态下的行为时,状态模式是一个更优雅的解决方案。

2、使用状态模式的好处

  1. 简化条件判断逻辑:通过将状态转换逻辑封装到各个具体状态类中,状态模式避免了大量的 if-elseswitch 语句,代码更清晰。

  2. 增强系统的扩展性:状态模式将状态相关的行为封装到独立的类中,因此添加新的状态或修改现有状态的行为不会影响其他代码,符合开闭原则

  3. 使状态行为与上下文解耦:上下文类仅负责状态的切换,而具体的行为由状态类来实现,保持了上下文类的简洁。

  4. 提高代码可维护性:状态类各自独立,便于测试、调试和维护。

3、使用状态模式时的注意事项

  1. 状态类的数量增加:状态模式的一个潜在问题是状态类的数量可能会随着状态的增多而增加,导致类过多。此时需要评估是否有必要使用状态模式,还是可以通过其他方式优化。

  2. 上下文和状态类的依赖:虽然状态模式将状态行为独立出来,但状态类与上下文之间的相互引用可能导致依赖复杂性,需要控制好状态类的职责。

  3. 状态转换规则复杂时:如果状态之间的转换规则非常复杂,可能需要将转换逻辑从具体状态类中抽离,避免具体状态类中的代码变得过于庞大。

三、在 Unity 中使用 状态模式

在 Unity 中,状态模式可以很好地用于角色的状态管理,例如角色的运动状态、攻击状态、死亡状态等。我们可以通过状态模式将这些行为封装在不同的状态类中,使得角色能够根据当前状态动态切换行为。

下面我们使用状态模式,实现一个角色可以在站立、行走、奔跑和跳跃状态之间切换的示例,基于 Unity 的 3D 环境。

参考类图如下:

1、定义状态接口

我们首先定义一个接口 ICharacterState,每个具体的状态类都需要实现该接口。

public interface ICharacterState
{
    void HandleInput(Character character);
    void UpdateState(Character character);
}

2、具体状态类

根据不同的角色状态(站立、行走、奔跑、跳跃),我们为每个状态创建一个具体的类。每个状态类都实现了 ICharacterState 接口。

using UnityEngine;

// 站立状态类
public class StandState : ICharacterState
{
    public void HandleInput(Character character)
    {
        if (Input.GetKeyDown(KeyCode.W))
        {
            character.SetState(new WalkState());
        }
    }

    public void UpdateState(Character character)
    {
        Debug.Log("角色正在站立");
    }
}

// 行走状态类
public class WalkState : ICharacterState
{
    public void HandleInput(Character character)
    {
        if (Input.GetKeyDown(KeyCode.R))
        {
            character.SetState(new RunState());
        }
        else if (Input.GetKeyDown(KeyCode.Space))
        {
            character.SetState(new JumpState());
        }
    }

    public void UpdateState(Character character)
    {
        character.transform.Translate(Vector3.forward * 2f * Time.deltaTime);
        Debug.Log("角色正在行走");
    }
}

// 奔跑状态类
public class RunState : ICharacterState
{
    public void HandleInput(Character character)
    {
        if (Input.GetKeyDown(KeyCode.Space))
        {
            character.SetState(new JumpState());
        }
    }

    public void UpdateState(Character character)
    {
        character.transform.Translate(Vector3.forward * 5f * Time.deltaTime);
        Debug.Log("角色正在奔跑");
    }
}

// 跳跃状态类
public class JumpState : ICharacterState
{
    public void HandleInput(Character character)
    {
        // 空中不允许其他输入
    }

    public void UpdateState(Character character)
    {
        character.transform.Translate(Vector3.up * 5f * Time.deltaTime);
        Debug.Log("角色正在跳跃");
        // 跳跃结束后返回站立状态
        character.SetState(new StandState());
    }
}

3、上下文类

接下来定义上下文类 Character,它维护当前的状态,并通过 SetState() 方法进行状态的切换。

using UnityEngine;

public class Character : MonoBehaviour
{
    private ICharacterState currentState;

    void Start()
    {
        // 初始化状态为站立
        currentState = new StandState();
    }

    void Update()
    {
        // 处理输入并更新状态
        currentState.HandleInput(this);
        currentState.UpdateState(this);
    }

    // 切换状态
    public void SetState(ICharacterState newState)
    {
        currentState = newState;
    }
}

4、使用场景中的角色

在 Unity 场景中,我们可以将此状态系统应用到一个角色对象上,比如一个 3D 模型。使用状态模式,角色将根据输入切换行为,比如从站立到行走,再到奔跑或跳跃。

创建一个新的脚本 GameController 来管理和初始化角色。

using UnityEngine;

public class GameController : MonoBehaviour
{
    public GameObject characterPrefab;
    private Character character;

    void Start()
    {
        // 初始化角色对象
        GameObject characterObject = Instantiate(characterPrefab);
        character = characterObject.AddComponent<Character>();
    }

    
}

5、在 Unity 中运行

  1. 创建一个空的 Unity 场景,放置一个 3D 角色(立方体、模型等)作为 characterPrefab
  2. 添加 CharacterGameController 脚本到 Unity 场景。
  3. 在游戏运行时,角色会根据键盘输入(W 切换到行走状态,R 切换到奔跑状态,空格键切换到跳跃状态)来切换不同的行为。

6、示例分析

  1. 初始状态为站立:当游戏开始时,角色默认处于站立状态。此时控制台输出“角色正在站立”,并且没有移动。
  2. 切换到行走状态:当玩家按下 W 键时,角色进入行走状态,角色开始向前移动,控制台输出“角色正在行走”。
  3. 切换到奔跑状态:按下 R 键,角色从行走状态切换到奔跑状态,移动速度加快,控制台输出“角色正在奔跑”。
  4. 切换到跳跃状态:按下空格键,角色进入跳跃状态,角色向上移动,跳跃结束后角色自动返回站立状态。

状态模式在 Unity 中非常适用于处理角色的状态转换。例如,角色的运动状态、游戏中的敌人 AI 状态等。通过将每个状态的行为封装成独立的类,我们可以灵活地进行状态切换并保持代码结构的清晰与可扩展性。

  • 简化了复杂的状态管理:避免使用大量的 if-elseswitch-case 语句,减少了冗余代码。
  • 高可扩展性:当需要添加新的状态时,只需要新建一个状态类即可,不会影响现有的代码。
  • 增强了代码的可维护性:各个状态之间相互独立,符合单一职责原则,每个状态类只负责处理特定状态下的行为。

四、观察者模式(Observer Pattern)

观察者模式(Observer Pattern)是一种行为型设计模式,它定义了对象之间的一对多依赖关系。当一个对象的状态发生变化时,它的所有依赖者(观察者)都会收到通知并自动更新。这种模式用于事件处理系统,当某个对象发生改变时,依赖该对象的其他对象能够及时响应。

在观察者模式中,主要有两个角色:

  1. 被观察者(Subject):状态发生变化的对象,它维护了一组观察者列表,并在状态发生变化时通知它们。
  2. 观察者(Observer):依赖被观察者的对象,它注册到被观察者上,并在被观察者状态变化时得到通知。

1、什么时候使用观察者模式

  • 事件系统:需要实现基于事件驱动的系统,许多 UI 系统使用观察者模式来处理按钮点击、值变化等事件。
  • 数据同步:当某个对象的状态改变时,依赖它的其他对象需要同步更新。
  • 系统中存在一对多的依赖关系:例如,一个对象的变化需要通知多个对象做出反应,典型的例子包括股票价格变化通知、新闻订阅系统等。
  • 通知机制:在某些情况下,多个模块需要接收到某个变化的信息(如游戏事件、社交媒体通知等)。

2、使用观察者模式的好处

  1. 解耦合:观察者和被观察者之间的耦合度低。被观察者只知道观察者实现了某些接口,而无需了解它们的具体实现细节。
  2. 灵活性:可以动态添加或移除观察者,系统的扩展性好。
  3. 提高代码的可维护性:将状态的变化与相应的反应分开,使代码更易于维护和修改。
  4. 响应式更新:观察者模式允许系统中的多个对象自动响应某个对象的状态变化,无需显式调用每个依赖对象。

3、使用时的注意事项

  1. 性能问题:当观察者数量过多时,每次状态改变都需要通知所有观察者,这可能会引起性能问题。
  2. 避免循环依赖:如果观察者在更新过程中再次触发了被观察者的通知,可能会导致循环调用或死锁。
  3. 顺序问题:多个观察者对同一事件做出响应时,要注意观察者之间的顺序依赖,可能会导致某些观察者未按预期更新。
  4. 内存泄漏问题:要确保观察者可以正确地从被观察者中移除,以避免内存泄漏问题。

五、在 Unity 中使用 观察者模式

为了演示如何在 Unity 中使用 观察者模式,我们将实现一个示例:当玩家接触到一个触发器(Trigger)时,游戏会通知观察者更新,例如改变颜色、显示文本等。这个示例将演示如何使用观察者模式管理多个对象对玩家触发事件做出反应。

参考类图如下:

1、定义观察者接口

首先,我们定义一个观察者接口,所有的观察者类都需要实现这个接口。

public interface IObserver
{
    void OnNotify();
}

2、定义被观察者类

然后我们定义一个被观察者类。在这个示例中,被观察者是一个触发器,当玩家接触触发器时,它会通知所有观察者。

using System.Collections.Generic;
using UnityEngine;

public class TriggerSubject : MonoBehaviour
{
    private List<IObserver> observers = new List<IObserver>();

    // 注册观察者
    public void RegisterObserver(IObserver observer)
    {
        observers.Add(observer);
    }

    // 移除观察者
    public void RemoveObserver(IObserver observer)
    {
        observers.Remove(observer);
    }

    // 通知所有观察者
    public void NotifyObservers()
    {
        foreach (IObserver observer in observers)
        {
            observer.OnNotify();
        }
    }

    // Unity 的触发器事件,当玩家接触触发器时调用
    private void OnTriggerEnter(Collider other)
    {
        if (other.CompareTag("Player"))
        {
            NotifyObservers();
        }
    }
}

3、实现观察者类

接下来我们创建几个不同的观察者类,每个观察者类会响应触发器的通知。在这个例子中,我们将创建两个观察者:

  • 一个会改变颜色
  • 一个会显示文本

3.1 观察者 1:改变颜色的观察者

using UnityEngine;

public class ColorObserver : MonoBehaviour, IObserver
{
    public Renderer objectRenderer;

    public void OnNotify()
    {
        // 随机改变对象的颜色
        objectRenderer.material.color = new Color(Random.value, Random.value, Random.value);
        Debug.Log("ColorObserver: Color changed!");
    }
}

3.2 观察者 2:显示文本的观察者

using UnityEngine;
using UnityEngine.UI;

public class TextObserver : MonoBehaviour, IObserver
{
    public Text messageText;

    public void OnNotify()
    {
        // 显示通知文本
        messageText.text = "Player triggered the event!";
        Debug.Log("TextObserver: Text updated!");
    }
}

4、在场景中使用观察者模式

现在我们在 Unity 场景中设置以下内容:

  1. 创建一个空的 GameObject,命名为 Trigger,并将 TriggerSubject 脚本附加到该对象上。同时,在该对象上添加一个 BoxCollider,并勾选 Is Trigger

  2. 创建两个 3D 物体(如立方体或球体),并附加 ColorObserver 脚本到其中一个物体,记得将 objectRenderer 变量拖入到 Inspector 中。

  3. 创建一个 UI 文本,附加 TextObserver 脚本,并将 messageText 变量拖入 Inspector。

  4. 在游戏的 Start() 函数中,注册观察者。

using UnityEngine;

public class GameManager : MonoBehaviour
{
    public TriggerSubject triggerSubject;
    public ColorObserver colorObserver;
    public TextObserver textObserver;

    void Start()
    {
        // 注册观察者
        triggerSubject.RegisterObserver(colorObserver);
        triggerSubject.RegisterObserver(textObserver);
    }

    void OnDestroy()
    {
        // 在销毁时移除观察者,避免内存泄漏
        triggerSubject.RemoveObserver(colorObserver);
        triggerSubject.RemoveObserver(textObserver);
    }
}

5、运行示例

  1. GameManager 脚本挂载到 Unity 场景中的一个空对象上。
  2. GameManager 中的 Inspector 窗口,将 triggerSubjectcolorObservertextObserver 分别拖入相应的字段。
  3. 运行游戏,当玩家接触 Trigger 对象时,注册的观察者将会被通知:
    • 物体的颜色会发生变化。
    • UI 文本会显示 "Player triggered the event!"。

6、示例分析

  • 触发器 (TriggerSubject) 是被观察者,当玩家接触它时,它会调用 NotifyObservers() 方法通知所有的观察者。
  • 观察者 (ColorObserverTextObserver) 实现了 IObserver 接口,并在 OnNotify() 方法中定义了各自的行为。
  • 通过这种设计,任何新增的观察者只需要实现 IObserver 接口,并注册到 TriggerSubject 中,而不需要修改已有的代码。

通过这个示例,我们可以看到如何在 Unity 中运用 观察者模式,处理多个对象对同一事件的响应。这种模式在处理游戏事件、状态变化等场景中十分有用,尤其是在复杂的游戏系统中。

  • 松耦合:观察者和被观察者之间的关系松耦合,被观察者不需要关心观察者的具体实现。
  • 扩展性:可以轻松添加新的观察者,而无需修改被观察者的代码。

六、备忘录模式(Memento Pattern)

备忘录模式(Memento Pattern)是一种行为型设计模式,它允许在不暴露对象内部状态的情况下保存和恢复对象的状态。备忘录模式通常用于需要撤销或恢复操作的场景。模式主要由三个角色组成:

  1. 发起人(Originator):需要保存和恢复状态的对象。
  2. 备忘录(Memento):保存发起人的内部状态的对象。
  3. 管理员(Caretaker):负责管理备忘录的对象,它不对备忘录的内容进行操作,只负责保存和恢复。

1、什么时候使用备忘录模式

  • 撤销操作:当用户需要撤销某个操作,例如文本编辑器中的撤销/重做功能。
  • 游戏存档:在游戏中保存和恢复玩家的状态,例如关卡进度、角色状态等。
  • 复杂状态管理:当对象的状态比较复杂,且需要频繁保存和恢复时。

2、使用备忘录模式的好处

  1. 封装性:备忘录模式允许保存对象的状态,而无需暴露其内部结构。
  2. 简化状态管理:可以轻松地实现状态的保存和恢复,尤其是在需要实现撤销/重做功能时。
  3. 易于实现:对于需要保留状态的对象,备忘录模式提供了一种简单、直观的方式来管理对象的状态。

3、使用时的注意事项

  1. 内存使用:备忘录模式可能会消耗大量内存,特别是在需要保存大量状态时。需要合理管理备忘录的数量。
  2. 性能问题:频繁的状态保存和恢复可能会影响系统性能。
  3. 设计复杂性:虽然备忘录模式提供了良好的封装性,但实现可能会增加系统的复杂性。

七、在 Unity 中使用 备忘录模式

为了在 Unity 中演示 备忘录模式,我们将实现一个简单的示例:玩家可以在场景中移动,并能够保存和恢复其位置和生命值的状态。这个示例将展示如何使用备忘录模式管理游戏状态。

参考类图如下:

1、定义备忘录类

首先,我们定义一个备忘录类,用于保存玩家的状态信息(位置和生命值)。

[System.Serializable]
public class PlayerMemento
{
    public Vector3 position;
    public int health;

    public PlayerMemento(Vector3 position, int health)
    {
        this.position = position;
        this.health = health;
    }
}

2、定义发起人类

接下来,我们定义一个 Player 类,它代表玩家,并提供保存和恢复状态的方法。

using UnityEngine;

public class Player : MonoBehaviour
{
    public Vector3 position;
    public int health;

    void Start()
    {
        position = transform.position; // 初始化位置
        health = 100; // 初始化生命值
    }

    // 创建备忘录
    public PlayerMemento CreateMemento()
    {
        return new PlayerMemento(position, health);
    }

    // 从备忘录恢复状态
    public void RestoreMemento(PlayerMemento memento)
    {
        position = memento.position;
        health = memento.health;
        transform.position = position; // 更新实际位置
    }

    void Update()
    {
        // 移动玩家
        float moveSpeed = 5f;
        float moveHorizontal = Input.GetAxis("Horizontal");
        float moveVertical = Input.GetAxis("Vertical");
        Vector3 movement = new Vector3(moveHorizontal, 0.0f, moveVertical) * moveSpeed * Time.deltaTime;
        transform.Translate(movement);

        // 打印当前状态
        if (Input.GetKeyDown(KeyCode.Space))
        {
            Debug.Log($"Current Position: {transform.position}, Health: {health}");
        }
    }

    // 示例:改变生命值
    public void TakeDamage(int damage)
    {
        health -= damage;
        Debug.Log($"Player took damage: {damage}. Current Health: {health}");
    }
}

3、定义管理员类

然后,我们定义一个 GameManager 类,用于管理备忘录的保存和恢复。

using System.Collections.Generic;
using UnityEngine;

public class GameManager : MonoBehaviour
{
    private Player player;
    private List<PlayerMemento> mementoList = new List<PlayerMemento>();
    private int currentStateIndex = -1;

    void Start()
    {
        player = FindObjectOfType<Player>();
    }

    // 保存当前状态
    public void SaveState()
    {
        PlayerMemento memento = player.CreateMemento();
        mementoList.Add(memento);
        currentStateIndex++;
        Debug.Log($"Game state saved! Total states: {mementoList.Count}");
    }

    // 恢复到上一个状态
    public void RestorePreviousState()
    {
        if (currentStateIndex > 0)
        {
            currentStateIndex--;
            player.RestoreMemento(mementoList[currentStateIndex]);
            Debug.Log($"Restored to state {currentStateIndex}!");
        }
        else
        {
            Debug.Log("No previous state to restore!");
        }
    }

    // 恢复到下一个状态
    public void RestoreNextState()
    {
        if (currentStateIndex < mementoList.Count - 1)
        {
            currentStateIndex++;
            player.RestoreMemento(mementoList[currentStateIndex]);
            Debug.Log($"Restored to state {currentStateIndex}!");
        }
        else
        {
            Debug.Log("No next state to restore!");
        }
    }

    void Update()
    {
        if (Input.GetKeyDown(KeyCode.S)) // 按下 S 保存状态
        {
            SaveState();
        }

        if (Input.GetKeyDown(KeyCode.R)) // 按下 R 恢复上一个状态
        {
            RestorePreviousState();
        }

        if (Input.GetKeyDown(KeyCode.N)) // 按下 N 恢复下一个状态
        {
            RestoreNextState();
        }

        if (Input.GetKeyDown(KeyCode.D)) // 按下 D 造成伤害
        {
            player.TakeDamage(10);
        }
    }
}

4、在 Unity 中测试

  1. 创建一个空的 GameObject,命名为 GameManager,并附加 GameManager 脚本。
  2. 创建一个球体作为玩家对象,附加 Player 脚本。
  3. 在 Unity 编辑器中,设置场景,确保玩家对象和游戏管理器在同一场景中。

5、运行示例

在游戏运行时,玩家可以使用以下键来测试备忘录模式:

  • S:保存当前状态。
  • R:恢复到上一个保存的状态。
  • N:恢复到下一个保存的状态。
  • D:造成 10 点伤害(测试生命值变化)。

6、示例分析

  • 发起人(Player):玩家的状态可以被保存和恢复。CreateMemento() 方法创建备忘录,RestoreMemento() 方法从备忘录恢复状态。
  • 备忘录(PlayerMemento):封装了玩家的位置和生命值。
  • 管理员(GameManager):管理备忘录,允许保存和恢复状态。

通过这个示例,我们展示了如何在 Unity 中实现 备忘录模式,用于管理对象的状态保存和恢复。这种模式在处理游戏中的撤销/重做功能、存档等场景中非常有用。

  • 状态管理简单:通过备忘录模式,可以轻松保存和恢复玩家的状态。
  • 封装性强:玩家的内部状态不暴露,保持了良好的封装性。
  • 灵活性:可以方便地扩展状态保存的内容,例如增加更多的属性。

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

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

相关文章

VCNet论文阅读笔记

VCNet论文阅读笔记 0、基本信息 信息细节英文题目VCNet and Functional Targeted Regularization For Learning Causal Effects of Continuous Treatments翻译VCNet和功能目标正则化用于学习连续处理的因果效应单位芝加哥大学年份2021论文链接[2103.07861] VCNet和功能定向正…

OpenCV_图像旋转超详细讲解

图像转置 transpose(src, dst); transpose()可以实现像素下标的x和y轴坐标进行对调&#xff1a;dst(i,j)src(j,i)&#xff0c;接口形式 transpose(InputArray src, // 输入图像OutputArray dst, // 输出 ) 图像翻转 flip(src, dst, 1); flip()函数可以实现对图像的水平翻转…

9.23 C++类中的特殊内容

仿照string类&#xff0c;实现myString //my_string.cpp #include "my_string.h" #include <iostream> #include <cstring>using namespace std;My_string::My_string():size(15){this->ptr new char[size];this->ptr[0] \0; //表示…

Qt_多元素控件

目录 1、认识多元素控件 2、QListWidget 2.1 使用QListWidget 3、QTableWidget 3.1 使用QListWidget 4、QTreeWidget 4.1 使用QTreeWidget 5、QGroupBox 5.1 使用QGroupBox 6、QTabWidget 6.1 使用QTabWidget 结语 前言&#xff1a; 在Qt中&#xff0c;控件之间…

Python办公自动化教程(001):PDF内容提取

1、Pdfplumber介绍 pdfplumber的github地址&#xff1a; https://github.com/jsvine/pdfplumber/【介绍】&#xff1a;pdfplumber 是一个用于处理 PDF 文件的 Python 第三方库&#xff0c;它提供了一种方便的方式来提取 PDF 文件中的文本、表格和其他信息。【功能】&#xff…

CICD从无到会

一 CICD是什么 CI/CD 是指持续集成&#xff08;Continuous Integration&#xff09;和持续部署&#xff08;Continuous Deployment&#xff09;或持续交付&#xff08;Continuous Delivery&#xff09; 1.1 持续集成&#xff08;Continuous Integration&#xff09; 持续集成是…

面向对象 vs 面向过程

Java 和 C 语言的区别&#xff1a;面向对象 vs 面向过程 在编程世界中&#xff0c;不同的编程语言承载着不同的编程范式。C 语言作为一门经典的面向过程编程语言&#xff0c;注重函数的调用和操作&#xff1b;而Java则是典型的面向对象编程语言&#xff0c;重视对象与类的设计…

拯救者Legion R9000X 2021R(82K8)原厂Win10与Windows11系统恢复镜像下载

LENOVO联想拯救者R9000X锐龙版2021款【82K8】预装OEM系统WIN11/10安装包&#xff0c;恢复原装出厂时开箱状态一模一样 链接&#xff1a;https://pan.baidu.com/s/15dGwacsEG0G8pOiZAHyXaQ?pwd0xgk 提取码&#xff1a;0xgk 联想原装出厂系统自带所有驱动、出厂主题壁纸、系统…

华为高级交换技术笔记 2024-2025

2024-2025 一、9/31.通信模型和封装2.以太网3.MAC地址4.以太网帧5.MAC地址表的建立 二、9/61.交换机的数据的处理2.以太网帧的分类3.广播域4.vlan技术开发背景 一、9/3 1.通信模型和封装 2.以太网 3.MAC地址 4.以太网帧 5.MAC地址表的建立 二、9/6 1.交换机的数据的处理 2.以…

Windows 配置docker和ubuntu系统

windos10 配置docke时&#xff0c;无意间发现wsl功能挺好用&#xff0c;而且是和docker 的linux容器连通的。 记录一下解决的几个问题 error during connect: Get http://%2F%2F.%2Fpipe%2Fdocker_engine/v1.40/images/json: open //./pipe/docker_engine: The system cannot …

学习记录:js算法(四十三):翻转二叉树

文章目录 翻转二叉树我的思路网上思路递归栈 总结 翻转二叉树 给你一棵二叉树的根节点 root &#xff0c;翻转这棵二叉树&#xff0c;并返回其根节点 图一&#xff1a; 图二&#xff1a; 示例 1&#xff1a;&#xff08;如图一&#xff09; 输入&#xff1a;root [4,2,7,1…

密集行人数据集 CrowdHumanvoc和yolo两种格式,yolo可以直接使用train val test已经划分好有yolov8训练200轮模型

密集行人数据集 CrowdHuman voc和yolo两种格式&#xff0c;yolo可以直接使用 train val test已经划分好 有yolov8训练200轮模型。 CrowdHuman 密集行人检测数据集 数据集描述 CrowdHuman数据集是一个专为密集行人检测设计的数据集&#xff0c;旨在解决行人密集场景下的检测挑…

2024个人简历模板免费可编辑,可能是整理最全的简历(支持Word格式下载)

提供各行业简历模板WORD可编辑格式下载&#xff0c;涵盖求职简历模板、大学生简历模板、个人简历模板、留学简历模板、英文简历模板、免费简历模板、工作简历模板、保研简历模板、暑期实习简历、寒假实习简历、校招简历等。 都是word格式&#xff0c;直接下载就能用。 网盘链…

zabbix入门单机部署

zabbix官网 1进入官网后选择右上角Download 选择你要的版本以及需要的组件&#xff0c;网页下方会自动生成需要操作的步骤 &#xff0c;跟着步骤一步一步安装即可&#xff1a; 这里跟着官网步骤一步步走下去就可以了 但是需要注意的是安装 yum install centos-release-scl源…

全面详尽的 PHP 环境搭建教程

目录 目录 PHP 环境搭建概述 在 Windows 上搭建 PHP 环境 使用集成环境 XAMPP 安装步骤 配置和测试 常用配置 手动安装 Apache、PHP 和 MySQL 安装 Apache 安装 PHP 安装 MySQL 配置 PHP 连接 MySQL 在 Linux 上搭建 PHP 环境 使用 LAMP 方案 安装 Apache 安装 …

vcruntime140_1.dll无法继续执行代码的6种解决方法

在计算机编程和软件开发中&#xff0c;我们经常会遇到各种错误和问题。其中&#xff0c;vcruntime140_1.dll无法继续执行代码是一个常见的问题。这个问题可能会导致程序崩溃&#xff0c;影响我们的工作进度。因此&#xff0c;了解这个问题的原因以及如何解决它是非常重要的。 …

Netty笔记10-Netty参数调优

文章目录 一、CONNECT_TIMEOUT_MILLISCONNECT_TIMEOUT_MILLIS设置为1秒超时CONNECT_TIMEOUT_MILLIS设置为5秒超时注意事项 二、SO_BACKLOG代码示例注意事项 三、ulimit -n(文件描述符)设置文件描述符限制在注意事项 四、TCP_NODELAY使用 TCP_NODELAY 的场景注意事项 五、SO_SND…

JavaWeb--纯小白笔记03:servlet入门---动态网页的创建

笔记&#xff1a;index.html在tomcat中为默认的名字&#xff0c;html里面的语法不严谨。改配置文件要小心&#xff0c;不然容易删掉其他 Servlet&#xff1a;服务器端小程序&#xff0c;写动态网页需要用Servlet&#xff0c;普通的java类通过继承HttpServlet&#xff0c;可以响…

【重学 MySQL】三十一、字符串函数

【重学 MySQL】三十一、字符串函数 函数名称用法描述ASCII(S)返回字符串S中的第一个字符的ASCII码值CHAR_LENGTH(s)返回字符串s的字符数&#xff0c;与CHARACTER_LENGTH(s)相同LENGTH(s)返回字符串s的字节数&#xff0c;和字符集有关CONCAT(s1,s2,…,sn)连接s1,s2,…,sn为一个字…

Docker + Win 10 学习记录

下载Docker Release notes | Docker Docs 推荐使用4.33版本&#xff0c;最新的Docker版本在win10 22H2无法安装。需要升级到win11. 查看Win10版本是否与最新版的Docker兼容 运行 win R&#xff0c; 然后输入winver 如果你的Docker版本无法在当前的win10安装&#xff0c;请更…