【原型模式】设计模式系列:高效克隆的艺术(深入解析)

news2025/1/10 1:27:21

文章目录

  • Java设计模式之原型模式详解
    • 1. 引言
    • 2. 原型模式概述
      • 2.1 定义与基本原理
      • 2.2 原型模式与其他模式的关系
      • 2.3 使用场景分析
    • 3. Java中的Cloneable接口
      • 3.1 Cloneable接口简介
      • 3.2 Object类中的clone方法
      • 3.3 实现Cloneable接口的步骤
      • 3.4 克隆方法的重写示例
    • 4. 深克隆与浅克隆
      • 4.1 深克隆的概念与实现方式
      • 4.2 浅克隆的概念与实现方式
      • 4.3 深克隆与浅克隆的区别
      • 4.4 示例代码解析
    • 5. 原型模式的应用实例
    • 6. 原型模式的优缺点
    • 7. 原型模式在实际项目中的应用
    • 8. 其他相关模式
      • 8.1 原型模式与工厂模式
      • 8.2 原型模式与建造者模式
      • 8.3 原型模式与单例模式
    • 9. 性能考量
      • 9.1 克隆操作的性能影响
      • 9.2 性能优化技巧
      • 9.3 实验数据与分析
    • 10. 高级话题
      • 10.1 自定义克隆器
      • 10.2 利用序列化实现深克隆
      • 10.3 使用框架提供的克隆功能
      • 10.4 克隆与并发控制
    • 11. 常见问题解答
    • 12. 总结与展望

Java设计模式之原型模式详解

1. 引言

设计模式简介
设计模式是一种在特定情境下解决问题的模板,它描述了一种在软件设计中反复出现的问题以及该问题的解决方案。设计模式不是完成任务的具体代码,而是用来解决常见问题的一种策略或蓝图,使得开发者能够在不同的项目中复用相同的解决方法,从而提高开发效率和软件质量。设计模式通常分为三大类:创建型模式、结构型模式和行为型模式。

原型模式的重要性
在面向对象编程中,创建对象是必不可少的一部分。然而,直接通过构造函数创建新对象可能会导致一些问题,比如:

  • 构造函数过于复杂,难以维护。
  • 对象初始化过程复杂,需要大量的配置信息。
  • 创建对象的成本较高,尤其是在资源有限的环境中。

原型模式作为一种创建型设计模式,可以有效地解决这些问题。它允许我们通过复制现有的对象来创建新的对象,而无需知道具体的创建逻辑。这种模式尤其适用于那些创建成本高或构造过程复杂的对象。此外,由于原型模式使用现有的对象作为模板,因此可以避免创建过程中的重复工作,提高程序的性能和可维护性。

2. 原型模式概述

2.1 定义与基本原理

定义:
原型模式是一种创建型设计模式,它允许一个对象通过复制已有的对象来创建新对象,而不是通过传统构造函数的方式。这种模式特别适用于创建复杂的对象,尤其是那些创建过程耗时的对象。

基本原理:
在Java中,原型模式的基本原理依赖于java.lang.Cloneable接口和Object类中的clone()方法。当一个类实现了Cloneable接口后,就可以使用clone()方法来创建一个对象的副本。clone()方法会创建一个与原对象具有相同状态的新对象,但它们是不同的对象实例。

关键组件:

  1. Prototype(原型):定义了一个用于克隆的接口。
  2. Concrete Prototype(具体原型):实现原型接口,包含业务逻辑和克隆自身的逻辑。
  3. Client(客户端):使用具体原型类的对象,并请求克隆操作。
    在这里插入图片描述

2.2 原型模式与其他模式的关系

与工厂模式的关系:

  • **相似之处:**两者都是创建型模式,都用于创建对象实例。
  • **不同之处:**工厂模式关注如何创建对象,而原型模式关注如何通过复制现有对象来创建新对象。

与建造者模式的关系:

  • **相似之处:**两种模式都解决了对象创建过程中的复杂性问题。
  • **不同之处:**建造者模式通过逐步构建对象的方式来创建复杂对象,而原型模式则通过复制一个已有对象来创建新的对象。

与单例模式的关系:

  • **相似之处:**两种模式都涉及到对象的创建过程。
  • **不同之处:**单例模式保证一个类只有一个实例,并提供一个全局访问点;而原型模式关注的是通过复制现有实例来创建新的实例。

2.3 使用场景分析

适用场景:

  1. 当创建新对象的成本较高时,可以通过复制现有的对象来节省时间和资源。
  2. 当对象的构造过程较为复杂,涉及多个步骤或依赖关系时。
  3. 当需要大量相似对象时,可以通过复制一个原型对象来快速生成新对象。
  4. 当需要创建的对象具有复杂的内部结构,且这些结构不易通过构造函数直接创建时。

不适用场景:

  1. 当对象的状态变化较大,频繁复制会导致内存消耗过大时。
  2. 当对象的创建过程简单且不耗时时,直接使用构造函数可能更为高效。

3. Java中的Cloneable接口

3.1 Cloneable接口简介

Cloneable是一个标记接口,它本身并不包含任何方法,只是表明一个类的对象支持被克隆。在Java中,如果一个类想要支持克隆功能,那么这个类必须实现Cloneable接口,否则调用clone()方法会抛出CloneNotSupportedException异常。

3.2 Object类中的clone方法

Object类中定义了一个受保护的方法clone(),该方法用于创建并返回当前对象的一个副本。默认情况下,这个方法会抛出CloneNotSupportedException异常,只有当类实现了Cloneable接口时,才能正常调用此方法。

protected native Object clone() throws CloneNotSupportedException;

3.3 实现Cloneable接口的步骤

  1. 实现Cloneable接口:在类声明中添加implements Cloneable
  2. 覆盖clone方法:在类中覆盖clone()方法,并确保正确地处理对象中的所有字段。
  3. 调用super.clone():在覆盖的clone()方法中调用super.clone()来执行实际的克隆操作。
  4. 处理非基本数据类型:对于引用类型成员变量,需要额外处理以确保深克隆或浅克隆的正确实现。

3.4 克隆方法的重写示例

假设有一个简单的Person类,包含姓名和年龄属性。

public class Person implements Cloneable {
    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    // Getters and setters...

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

4. 深克隆与浅克隆

4.1 深克隆的概念与实现方式

**概念:**深克隆是指对一个对象进行克隆时,不仅复制了对象本身,还复制了对象所引用的所有对象。也就是说,深克隆会递归地复制对象及其所有成员对象,创建一个完全独立的副本。

实现方式:

  • 使用clone()方法:如果对象中的所有成员变量也是可克隆的,则需要递归地调用每个成员变量的clone()方法。
  • 序列化反序列化:将对象序列化为字节流,再从字节流中反序列化出新的对象。

4.2 浅克隆的概念与实现方式

**概念:**浅克隆是指对一个对象进行克隆时,只复制对象本身,而不复制对象所引用的对象。也就是说,浅克隆得到的新对象与原对象共享其引用类型的成员变量。

实现方式:

  • 使用clone()方法:仅复制对象本身的字段,如果字段是引用类型,则引用的是同一个对象。

4.3 深克隆与浅克隆的区别

  • 对象独立性:深克隆创建的对象是完全独立的,而浅克隆创建的对象与其原始对象在引用类型上是共享的。
  • 性能开销:深克隆由于需要递归复制所有的引用类型成员,因此性能开销较大;而浅克隆只复制对象本身,性能开销较小。
  • 适用场景:深克隆适用于需要完全独立副本的场景,而浅克隆适用于只需要对象本身副本的场景。

4.4 示例代码解析

假设有一个Person类和一个Address类,Person类包含一个Address对象作为成员变量。

public class Address implements Cloneable {
    private String street;
    private String city;

    public Address(String street, String city) {
        this.street = street;
        this.city = city;
    }

    // Getters and setters...

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

public class Person implements Cloneable {
    private String name;
    private Address address;

    public Person(String name, Address address) {
        this.name = name;
        this.address = address;
    }

    // Getters and setters...

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Person clonedPerson = (Person) super.clone();
        clonedPerson.setAddress((Address) address.clone());
        return clonedPerson;
    }
}

在这个例子中,Person类实现了Cloneable接口,并重写了clone()方法。为了实现深克隆,Person类的clone()方法中还调用了Address类的clone()方法,这样就确保了address字段也被克隆。

5. 原型模式的应用实例

场景描述
假设我们需要开发一个游戏系统,其中包含了多种角色,每种角色都有不同的属性和技能。在游戏中,玩家可以通过选择角色并对其进行定制来创建自己的队伍。为了提高游戏的性能和减少内存占用,我们需要一种方法来快速地创建角色实例,同时避免每次创建角色时都要重新设置各种属性。

类图与设计思路
我们可以定义一个角色基类Character,它实现了Cloneable接口,并且提供了一个clone()方法来复制角色。然后,我们可以定义几个具体的子类,如Warrior, Mage, 和Archer,每个子类代表不同类型的角色。

类图如下所示:

+----------------+
| Character      |
| - name: String |
| - level: int   |
| - skills: List |
| + clone(): Character |
+----------------+
         /|\ 
       /   \ 
+---------+ +---------+ +---------+
| Warrior | | Mage    | | Archer  |
+---------+ +---------+ +---------+
| - strength: int     | | - mana: int       | | - agility: int     |
+-------------------+ +----------------+ +-------------------+

代码实现
首先定义Character基类:

import java.util.ArrayList;
import java.util.List;

public abstract class Character implements Cloneable {
    private String name;
    private int level;
    private List<String> skills;

    public Character(String name, int level, List<String> skills) {
        this.name = name;
        this.level = level;
        this.skills = new ArrayList<>(skills);
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getLevel() {
        return level;
    }

    public void setLevel(int level) {
        this.level = level;
    }

    public List<String> getSkills() {
        return skills;
    }

    public void setSkills(List<String> skills) {
        this.skills = skills;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Character clonedCharacter = (Character) super.clone();
        clonedCharacter.skills = new ArrayList<>(this.skills);
        return clonedCharacter;
    }
}

接着定义具体的子类,例如Warrior

public class Warrior extends Character {
    private int strength;

    public Warrior(String name, int level, List<String> skills, int strength) {
        super(name, level, skills);
        this.strength = strength;
    }

    public int getStrength() {
        return strength;
    }

    public void setStrength(int strength) {
        this.strength = strength;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Warrior clonedWarrior = (Warrior) super.clone();
        clonedWarrior.strength = this.strength;
        return clonedWarrior;
    }
}

运行结果分析
我们可以创建一个GameSystem类来测试角色的克隆功能:

public class GameSystem {
    public static void main(String[] args) {
        try {
            Warrior warrior = new Warrior("Conan", 10, List.of("Sword Mastery", "Battle Cry"), 100);
            Warrior clonedWarrior = (Warrior) warrior.clone();

            System.out.println("Original Warrior: " + warrior.getName() + ", Level: " + warrior.getLevel());
            System.out.println("Cloned Warrior: " + clonedWarrior.getName() + ", Level: " + clonedWarrior.getLevel());

            // 修改克隆后的对象
            clonedWarrior.setName("Barbarian");
            clonedWarrior.setLevel(11);

            System.out.println("Modified Cloned Warrior: " + clonedWarrior.getName() + ", Level: " + clonedWarrior.getLevel());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

运行结果:

Original Warrior: Conan, Level: 10
Cloned Warrior: Conan, Level: 10
Modified Cloned Warrior: Barbarian, Level: 11

这表明克隆操作成功创建了一个与原始对象独立的新对象。

6. 原型模式的优缺点

优点

  • 减少创建新对象的成本:通过复制现有对象,可以避免复杂的构造过程。
  • 提高性能:对于创建成本高的对象,原型模式可以显著提高性能。
  • 灵活性:可以通过修改原型对象来控制创建的对象的状态。

缺点

  • 克隆方法的实现:需要显式地实现Cloneable接口和clone()方法,增加了类的设计复杂度。
  • 深克隆与浅克隆问题:如果对象包含引用类型成员,需要处理深克隆与浅克隆的问题,以确保正确的对象复制。

应用时的注意事项

  • 确保所有成员变量都可以被正确克隆:特别是对于引用类型成员,需要确保它们也被正确复制。
  • 考虑性能因素:对于大型对象或包含大量引用的对象,深克隆可能会成为性能瓶颈。
  • 安全性:对于敏感数据,需要考虑是否应该通过克隆来创建新对象。

7. 原型模式在实际项目中的应用

项目背景
假设我们在开发一个报表系统,用户可以根据需求创建各种报表模板,并基于这些模板生成具体的报表。报表模板可能包含复杂的布局、样式和数据源等信息,创建一个新的报表实例需要大量的配置。

面临的问题

  • 创建报表实例的成本高:每次创建新的报表都需要设置复杂的配置信息。
  • 资源消耗大:报表可能包含大量的数据和复杂的布局,直接创建会消耗较多资源。

解决方案
采用原型模式来创建报表实例,通过复制现有的报表模板来减少创建成本。

实现细节

  • 定义报表基类:创建一个ReportTemplate类,它实现了Cloneable接口,并提供了克隆方法。
  • 具体报表模板类:为每种类型的报表定义具体的子类,例如SalesReportInventoryReport
  • 报表管理器:创建一个ReportManager类来管理报表模板,用户可以通过这个管理器获取模板并创建报表实例。
public abstract class ReportTemplate implements Cloneable {
    private String title;
    private List<String> sections;

    public ReportTemplate(String title, List<String> sections) {
        this.title = title;
        this.sections = new ArrayList<>(sections);
    }

    // Getters and setters...

    @Override
    protected Object clone() throws CloneNotSupportedException {
        ReportTemplate clonedReport = (ReportTemplate) super.clone();
        clonedReport.sections = new ArrayList<>(this.sections);
        return clonedReport;
    }
}

public class SalesReport extends ReportTemplate {
    private String salesPeriod;

    public SalesReport(String title, List<String> sections, String salesPeriod) {
        super(title, sections);
        this.salesPeriod = salesPeriod;
    }

    // Getters and setters...
    
    @Override
    protected Object clone() throws CloneNotSupportedException {
        SalesReport clonedSalesReport = (SalesReport) super.clone();
        clonedSalesReport.salesPeriod = this.salesPeriod;
        return clonedSalesReport;
    }
}

public class ReportManager {
    private Map<String, ReportTemplate> templates = new HashMap<>();

    public void registerTemplate(String name, ReportTemplate template) {
        templates.put(name, template);
    }

    public ReportTemplate getTemplate(String name) {
        return templates.get(name);
    }

    public ReportTemplate createReport(String name) throws CloneNotSupportedException {
        ReportTemplate template = getTemplate(name);
        if (template != null) {
            return (ReportTemplate) template.clone();
        }
        return null;
    }
}

结果与反馈
通过使用原型模式,我们能够快速地创建报表实例,减少了用户的等待时间,并且提高了系统的整体性能。用户反馈显示,报表创建的速度明显提升,同时也减少了服务器资源的消耗。

8. 其他相关模式

8.1 原型模式与工厂模式

  • 工厂模式用于创建对象而不需要指定具体的类。它通常用于创建一组相关或依赖对象。
  • 原型模式则用于创建对象的副本。它更适用于创建复杂的对象,尤其是当创建过程成本较高时。

对比

  • 相似性:两者都属于创建型模式,用于对象的创建。
  • 差异:工厂模式侧重于对象的创建过程,而原型模式侧重于通过复制现有对象来创建新对象。

8.2 原型模式与建造者模式

  • 建造者模式用于创建复杂的对象,通过一步一步构建的方式。
  • 原型模式则是通过复制现有的对象来创建新对象。

对比

  • 相似性:两种模式都可以用来创建复杂的对象。
  • 差异:建造者模式通过步骤来构建对象,而原型模式通过复制已有对象并修改来创建新对象。

8.3 原型模式与单例模式

  • 单例模式保证一个类只有一个实例,并提供一个全局访问点。
  • 原型模式没有这种限制,它可以创建多个实例。

对比

  • 相似性:两种模式都关注于对象的创建。
  • 差异:单例模式确保对象唯一性,而原型模式允许对象的复制。

9. 性能考量

9.1 克隆操作的性能影响

  • 浅克隆通常比较快,因为它只复制对象本身的字段。
  • 深克隆可能较慢,因为它需要递归复制所有引用类型的字段。

9.2 性能优化技巧

  • 使用缓存:存储已克隆的对象以避免重复克隆。
  • 按需克隆:只有在真正需要时才执行克隆操作。
  • 优化克隆逻辑:确保克隆逻辑尽可能高效。

9.3 实验数据与分析

  • 基准测试:使用JMH(Java Microbenchmark Harness)或其他工具进行基准测试,比较不同克隆方法的性能。
  • 分析报告:记录不同情况下克隆操作的时间消耗和其他性能指标。

10. 高级话题

10.1 自定义克隆器

  • 自定义克隆器可以为特定对象类型提供优化的克隆策略。
  • 实现:可以通过实现Cloneable接口并在类中覆盖clone()方法来实现。

10.2 利用序列化实现深克隆

  • 序列化:将对象转换为字节流,然后再反序列化为对象,以实现深克隆。
  • 实现:通过实现Serializable接口,并使用序列化/反序列化的方法实现深克隆。

10.3 使用框架提供的克隆功能

  • Spring框架:提供了BeanUtils工具类来帮助克隆对象。
  • Hibernate:通过Session的replicate()方法来复制持久化对象。

10.4 克隆与并发控制

  • 线程安全:确保克隆操作不会受到并发访问的影响。
  • 实现:可以使用synchronized关键字或者ReentrantLock来保护克隆操作。

下面是你所需的常见问题解答部分和总结与展望部分的内容。

11. 常见问题解答

如何选择使用原型模式?

  • 对象创建成本高:当创建一个新对象的成本很高时,可以选择使用原型模式。
  • 对象初始化复杂:如果对象的初始化过程很复杂,涉及多个步骤或依赖关系,可以考虑使用原型模式。
  • 需要大量相似对象:当需要创建大量的相似对象时,使用原型模式可以减少创建新对象的开销。

如何避免克隆过程中的异常?

  • 实现Cloneable接口:确保对象实现了Cloneable接口。
  • 覆盖clone方法:在类中覆盖clone()方法,并调用super.clone()
  • 处理非基本数据类型:对于引用类型成员变量,需要确保它们也可以被正确克隆,避免浅克隆导致的问题。
  • 异常处理:在clone()方法中捕获并处理CloneNotSupportedException异常。

如何处理不可克隆的对象?

  • 检查对象是否可克隆:在尝试克隆前,检查对象是否实现了Cloneable接口。
  • 使用深克隆:对于不可克隆的对象,可以考虑使用序列化方法实现深克隆。
  • 替换为可克隆对象:如果可能,可以考虑替换不可克隆的对象为可克隆的版本。

12. 总结与展望

本文要点回顾

  • 原型模式定义:原型模式是一种创建型设计模式,允许通过复制现有的对象来创建新的对象。
  • Cloneable接口:在Java中,通过实现Cloneable接口并覆盖clone()方法来支持克隆功能。
  • 深克隆与浅克隆:理解深克隆和浅克隆的区别,并知道如何实现这两种克隆方式。
  • 应用场景:原型模式适用于创建成本高、初始化复杂或需要大量相似对象的场景。
  • 其他相关模式:了解原型模式与工厂模式、建造者模式和单例模式之间的区别。
  • 性能考量:讨论克隆操作的性能影响,并提供性能优化技巧。
  • 高级话题:探讨自定义克隆器、利用序列化实现深克隆、使用框架提供的克隆功能以及克隆与并发控制等问题。

未来发展趋势

  • 现代框架的支持:随着Java和其他语言的发展,越来越多的框架和库提供了内置的克隆支持。
  • 自动克隆工具:开发人员可能会看到更多的自动克隆工具出现,这些工具可以简化克隆过程。
  • 性能优化:未来的开发趋势将继续关注性能优化,特别是对于大规模应用而言,克隆操作的性能将更加重要。
  • 并发处理:随着多核处理器的普及,克隆操作的并发处理也将变得更加重要。

本文详细介绍了23种设计模式的基础知识,帮助读者快速掌握设计模式的核心概念,并找到适合实际应用的具体模式:
【设计模式入门】设计模式全解析:23种经典模式介绍与评级指南(设计师必备)

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

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

相关文章

Django-Oscar开发独立站/外贸商城教程与问题记录

​特别说明&#xff1a; 本博客为个人开发Django-Oscar时的经验总结&#xff0c;方便后期维护&#xff01;&#xff08;第一次这么认真的记录这种大型项目&#xff0c;打个广告吧&#xff1a;本人可接单算法程序开发&#xff0c;包含深度学习和图像相关……等相关&#xff09;…

秒懂C++之stack、queue、堆

目录 前言 一.stack常用接口 二.stack模拟实现 三.例题 3.1 最小栈 题目解析&#xff1a; 算法解析&#xff1a; 代码&#xff1a; 3.2 栈的压入、弹出序列 题目解析: 算法解析&#xff1a; 代码&#xff1a; 3.3 逆波兰表达式求值 题目解析&#xff1a; 算法解析…

【Web】从TFCCTF-FUNNY浅析PHPCGI命令行注入漏洞利用

目录 背景 CVE-2012-1823 发散利用 法一&#xff1a;读文件 法二&#xff1a;数据外带 背景 CVE-2012-1823 PHP-CGI远程代码执行漏洞&#xff08;CVE-2012-1823&#xff09;分析 | 离别歌 省流&#xff1a; 命令行参数不光可以通过#!/usr/local/bin/php-cgi -d include…

目标检测 | yolov4 原理和介绍

1. 简介 YOLOv4是一种高效且准确的目标检测模型&#xff0c;它在YOLOv3的基础上引入了多项改进&#xff0c;这些改进主要集中在网络结构的优化和训练技巧的更新上。以下是YOLOv4中的一些关键技术或模块&#xff0c;它们对提高目标检测性能起到了重要作用&#xff1a; CSPDarkne…

MATLAB基础应用精讲-【数模应用】配对样本Wilcoxon检验(附MATLAB、R语言和python代码实现)

目录 知识储备 常用的统计假设检验的方法 算法原理 什么是Wilcoxon符号秩检验? 何时使用Wilcoxon符号秩检验 适用条件 SPSS-符号秩检验 一统计理论 二实例分析 三拓展知识 SAS --配对样本Wilcoxon符号秩检验 SPSSAU 配对样本Wilcoxon检验案例 1、背景 2、理论 …

【C++指南】函数重载:多态性的基石

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;《C指南》 期待您的关注 目录 引言 一、函数重载的概念 二、函数重载的原理 三、函数重载的应用场景 四、函数重载的规则 五…

springboot整合libreoffice(两种方式,使用本地和远程的libreoffice);docker中同时部署应用和libreoffice

一、 背景 因为项目中需要使用word转pdf功能&#xff0c;因为转换速度原因&#xff0c;最后选用了libreoffice&#xff0c;原因及部署请参考 linux ubuntu环境安装libreoffice&#xff0c;word转pdf 远程调用的话可选docker部署&#xff0c;请看2.3.1 二、springboot整合libr…

AT指令处理框架

<AT框架>做什么 <AT框架>介绍 ATFrame 流程 开启AT流程: ATCommandRegister(CSQ,EXEXCMD,NULL); 发送消息队列xQueueSend(ATcmdQueue,(void *) &RegcommandInfo,(TickType_t)100) AT主流程 ATCommandSendScheduler 等待开启xQueueReceive(ATcmdQueue, (voi…

详细分析python下载文件的两种方式(附Demo)

目录 前言1. 接口方式2. Nginx 前言 基本项目的构建推荐阅读&#xff1a;构建Flask前后端不分离项目&#xff08;附Python Demo&#xff09; 对于Java的基本知识&#xff0c;推荐阅读&#xff1a; java框架 零基础从入门到精通的学习路线 附开源项目面经等&#xff08;超全&…

Unknown input format pdf Pandoc can convert to PDF, but not from PDF.解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

【Lua运行环境安装及配置】

首先在百度搜索“Lua菜鸟教程” 网址&#xff1a;Lua 教程 | 菜鸟教程 (runoob.com) 找到环境安装 点击Github下载地址&#xff0c;GitHub可以在正规上网下进入网址&#xff08;页面更新不出来多试几次&#xff09; 点击.exe安装 下载完成后进行安装 可自定义磁盘 项目文件很…

线程局部变量(ThreadLocal)

线程局部变量&#xff08;ThreadLocal&#xff09; 1、什么是ThreadLocal&#xff1f;2、基本用法3、注意事项 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 1、什么是ThreadLocal&#xff1f; ThreadLocal 是 Java 中用于实现线程局部变…

IEC 61850标准协议解读 1.建模讲解

IEC61850协议标准和之前的101、104协议最大的区分就是在模型这里&#xff0c;模型可以说是一个变电站里面的智能设备的定义&#xff0c;里面定义了网络连接信息、数据上报方式、设备访问控制点等&#xff0c;所以说在不同系统的对接过程中&#xff0c;有了这么一个基于标准定义…

sqliabs靶场练习

关卡 熟悉Mqsql相关命令Less-1Less-2Less-3Less-4Less-5Less-6Less-7Less-8Less-9Less-10 熟悉Mqsql相关命令 首先先熟悉一下Mysql的相关命令&#xff0c;可能在SQL注入的时候会用到。 1.查询服务器主机信息 HOSTNAME 主机名称 datadir——数据库路径 version_compile_os—…

JavaDS —— 红黑树

前言 还是一样&#xff0c;这里的红黑树重点讲述插入代码的实现&#xff0c;如果对红黑树的删除感兴趣&#xff0c;可以去翻阅其他资料。 在数据结构专栏中已经对 AVL 树的旋转调整做了分析和讲解&#xff0c;这里红黑树也会使用到旋转调整的代码&#xff0c;就不讲述旋转代码…

成都云飞浩容文化传媒有限公司怎么样可靠吗?

在数字经济浪潮汹涌的今天&#xff0c;电商行业作为推动经济高质量发展的关键引擎&#xff0c;正以前所未有的速度重塑着商业版图。在这场变革中&#xff0c;成都云飞浩容文化传媒有限公司凭借其专业的电商服务能力和前瞻性的市场洞察&#xff0c;成为了众多品牌商家信赖的合作…

助力草莓智能自动化采摘,基于嵌入式端超轻量级模型LeYOLO全系列【n/s/m/l】参数模型开发构建果园种植采摘场景下草莓成熟度智能检测识别系统

随着科技的飞速发展&#xff0c;人工智能&#xff08;AI&#xff09;技术已经渗透到我们生活的方方面面&#xff0c;从智能家居到自动驾驶&#xff0c;再到医疗健康&#xff0c;其影响力无处不在。然而&#xff0c;当我们把目光转向中国的农业领域时&#xff0c;一个令人惊讶的…

Git分布式版本控制--2+day018 LeetCode235 701 450

基础操作: 1.分支: 几乎所有的版本控制系统都以某种形式支持分支。 使用分支意味着你可以把你的工作从开发主线上分离 开来进行重大的Bug修改、开发新的功能&#xff0c;以免影响开发主线。 2.指令操作: 2.1 查看本地分支: git branch 2.2 创建本地分支: git branch 分支名…

Spring Boot 3.3 新特性介绍

1. 引言 Spring Boot 3.1.x 停止维护了&#xff0c;而 3.3.x 作为最新发布的版本&#xff0c;带来了许多新特性和改进。本篇文章将详细介绍这些新特性&#xff0c;并通过样例代码加以解释&#xff0c;帮助开发者更好地掌握和应用这些新功能。 Spring Boot 3.3现已正式发布&…

RNN循环网络层

文章目录 1、简介2、RNN 网络原理3、PyTorch RNN 层的使用3.1、RNN送入单个数据3.2、RNN层送入批量数据 4、RNN三个维度4.1、解释4.2、输入数据的组织4.3、示例4.4、为什么需要这种格式&#xff1f;4.5、小结 &#x1f343;作者介绍&#xff1a;双非本科大三网络工程专业在读&a…