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

news2024/11/15 21:31:45

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.coloradmin.cn/o/2159281.html

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

相关文章

Android RecyclerView 实现 GridView ,并实现点击效果及方向位置的显示

效果图 一、引入 implementation com.github.CymChad:BaseRecyclerViewAdapterHelper:2.9.30 二、使用步骤 1.Adapter public class UnAdapter extends BaseQuickAdapter<UnBean.ResultBean, BaseViewHolder> {private int selectedPosition RecyclerView.NO_POSITIO…

苹果电脑系统重磅更新——macOS Sequoia 15 系统 新功能一 览

有了 macoS Sequoia&#xff0c;你的工作效率将再次提升&#xff1a;快速调整桌面布局&#xff0c;一目了然地浏览网页重点&#xff0c;还可以通过无线镜像功能操控你的iPhone。 下面就来看看几项出色新功能&#xff0c;还有能够全面发挥这些功能的 App 和游戏。 macOS Sequo…

探秘 Web Bluetooth API:连接蓝牙设备的新利器

引言 随着物联网技术的快速发展&#xff0c;蓝牙设备在日常生活中扮演着越来越重要的角色。而在 Web 开发领域&#xff0c;Web Bluetooth API 的出现为我们提供了一种全新的方式来连接和控制蓝牙设备。本文将深入探讨 Web Bluetooth API 的使用方法和原理&#xff0c;帮助开发…

ElasticSearch的使用、Kibana和ES-Head可视化工具

ElasticSearch的使用、Kibana和ES-Head可视化工具 一、ElasticSearch概述1. ES2.IK分词器3. Kibana4.Head 二、安装1.ES安装2. 配置跨域和IK分词器3.Kibana安装4. Head安装 三、常用操作1. ES结构2. ES操作1. 索引的基本操作-创建索引-查看索引-修改索引-删除索引-特殊查看 2.文…

git笔记之在多个分支中复用某个分支提交的更改

git笔记之在多个分支中复用某个分支提交的更改 code review! 文章目录 git笔记之在多个分支中复用某个分支提交的更改1.实现该功能的 Bash 脚本示例2.这个脚本是否可以处理新添加的文件&#xff1f;3.该脚本使用前&#xff0c;应先使用下述脚本重置本地仓库所有分支与远程保持一…

Jenkins Pipeline 中通过勾选参数来控制是否构建 Docker 镜像

1.定义参数&#xff1a; 使用 booleanParam 定义一个布尔参数&#xff0c;示例如下 booleanParam(name: BUILD_DOCKER, description: 是否构建Docker镜像, defaultValue: false)2.使用参数&#xff1a; 在 stage 中&#xff0c;根据参数的值决定构建方式&#xff1a; stage(编…

HTTP 与 HTTPS 的三次握手与四次挥手详解

文章目录 HTTP 与 HTTPS 的三次握手与四次挥手详解一、HTTP 的三次握手与四次挥手1. HTTP 三次握手2. HTTP 四次挥手 二、HTTPS 的三次握手与四次挥手1. HTTPS&#xff08;无证书&#xff09;的三次握手与 SSL/TLS 握手2. HTTPS&#xff08;有证书&#xff09;的三次握手与 SSL…

vue-baidu-map的基本使用

前言 公司项目需求引入百度地图&#xff0c;由于给的时间比较短&#xff0c;所以就用了已经封装好了的vue-baidu-map 一、vue-baidu-map是什么&#xff1f; vue-baidu-map是基于vue.js封装的百度地图组件(官方文档) 二、使用步骤 1.下载插件 //我下载的版本 npm install …

注册讲堂 | 医疗器械组合包类产品常见问题(2)

问题一&#xff1a;组合包类产品的有效期应该如何确定&#xff1f; 组合包类产品的有效期以组件中最短有效期为最终产品有效期。 对于外购件&#xff0c;需要考虑组件购买时的剩余效期及影响效期主要因素&#xff0c;如材料老化、灭菌有效期等。 问题二&#xff1a;组合包类产…

【计算机网络强化】计网强化笔记

第一章 计算机网络体系结构 1.1 计算机网络概述 1.计算机网络由若干个节点和连接这些节点的链路组成 2. 3.计算机网络的组成 ①硬件、软件、协议 ②边缘部分和核心部分 ③通信子网和资源子网 4.电路交换、报文交换和分组交换 ①电路交换 分为三步&#xff1a;建立连接、…

基于SpringBoot+Vue的在线问诊管理系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码 精品专栏&#xff1a;Java精选实战项目…

SQL面试常见题目

SQL面试常见题目涉及多个方面&#xff0c;包括数据查询、数据操作、表的设计与优化等。以下列举一些经典的SQL面试题目&#xff0c;并附上解析答案&#xff1a; 1. 查询一张表中重复的数据 题目&#xff1a; 给定一个表 employees&#xff0c;包含 id, name, salary 列。如何…

大型语言模型(Large Language Models)的介绍

背景 大型语言模型&#xff08;Large Language Models&#xff0c;简称LLMs&#xff09;是一类先进的人工智能模型&#xff0c;它们通过深度学习技术&#xff0c;特别是神经网络&#xff0c;来理解和生成自然语言。这些模型在自然语言处理&#xff08;NLP&#xff09;领域中扮…

丢失照片/消息/文件,当发现没有备份 Android 手机数据时急救方法

当人们发现他们没有备份 Android 手机数据时&#xff0c;通常为时已晚。但是&#xff0c;我们都不想永久丢失珍贵的照片&#xff0c; 消息和其他文件。这就是为什么您应该检查 遵循 5 大免费 Android 数据恢复工具和最佳替代品 他们。 排名前五的免费 Android 数据恢复软件 1.奇…

ELK-01-elasticsearch-8.15.1安装

文章目录 前言一、下载elasticsearch二、将tar包放到服务器三、解压tar包四、更改配置文件五、添加启动用户六、用elasticserch用户启动6.1 报错6.2 解决问题16.3 解决问题26.4 再次用elasticserch用户启动6.5 windows浏览器打开 七、设置开机自动启动7.1 创建启动脚本7.2 在脚…

【C++】二、数据类型 (同C)

2.1 整形 无特殊情况&#xff0c;一般用int 2.2实型&#xff08;浮点型&#xff09; 1. 单精度 foat 2. 双精度 double 输入小数时默认double&#xff0c;定义变量时可以使用float转换为单精度 3. 可使用科学计数法表示小数&#xff08;看得懂即可&#xff09; 4. 用于浮点型用…

Excel--DATEDIF函数的用法及参数含义

DATEDIF函数的用法为: DATEDIF(start_date,end_date,unit),start_date表示的是起始时间&#xff0c;end_date表示的是结束时间。unit表示的是返回的时间代码&#xff0c;是天、月、年等。如下: Datedif函数的参数含义unit参数返回值的意义"y"两个时间段之间的整年数…

推荐使用10款源代码加密软件,保护核心源代码,减少泄密风险

在现代企业中&#xff0c;保护核心源代码的安全变得尤为重要。源代码不仅是产品的核心资产&#xff0c;也是黑客和竞争对手的目标。一旦代码泄露&#xff0c;不仅可能导致产品安全漏洞&#xff0c;还会使企业的知识产权面临威胁。为了解决这一问题&#xff0c;源代码加密软件能…

Java反序列化利用链篇 | CC1链的第二种方式-LazyMap版调用链【本系列文章的分析重点】

文章目录 CC1链的第二种方式-LazyMap版调用链LazyMap构造payloadCC1的调用链 系列篇其他文章&#xff0c;推荐顺序观看~ Java反序列化利用链篇 | JdbcRowSetImpl利用链分析Java反序列化利用链篇 | CC1链_全网最菜的分析思路【本系列文章的分析重点】Java反序列化利用链篇 | CC1…

人工智能之计算机视觉的发展历程与相关技术内容,相应的模型介绍

大家好&#xff0c;我是微学AI&#xff0c;今天给大家介绍一下人工智能之计算机视觉的发展历程与相关技术内容&#xff0c;相应的模型介绍。本文围绕计算机视觉这一领域&#xff0c;以问答的形式呈现了关键问题及详细解答。内容涵盖计算机视觉的基本概念、技术原理、应用场景等…