java设计模式--创建型设计模式

news2025/1/12 20:55:34

创建型模式可分为:单例模式、抽象工厂模式、原型模式、建造者模式、工厂模式

单例模式

单例模式

  • 就是采取一定的方法保证在整个软件系统中,对某个类只能存在一个对象实例,并且该类只提供一个获取其对象的方法(静态方法)
  • 单例模式共有8中写法
  • 对于一些重量级的对象,可以使用单例模式

单例模式的8种写法

  • 单例模式可以避免很重量级的对象频繁创建
饿汉式写法
  • 私有构造器
  • 类在内部创建静态对象
  • 提供一个共有的静态方法,获取对象的方法
  • 优点:
    • 写法简单,在装载类的时候就完成了实例化,避免了多线程问题
  • 缺点:
    • 载类加载的时候就完成实例化,没有懒加载的效果,如果从始至终都没有使用过这个实例,会造成内存的浪费
  • 这种方式线程安全可用,但是有可能会造成内存浪费

饿汉式有两种写法:

  • 静态变量写法
/**
 * 饿汉单例模式,静态变量写法
 */
public class HungryMan {
    
    private static HungryMan hungryMan =new HungryMan();
    
    private HungryMan(){
        
    }
    
    public static HungryMan getInstance(){
        return hungryMan;
    }
    
}
  • 静态代码块写法
/**
 * 饿汉单例模式,静态代码块写法
 */
public class HungryMan1 {

    private static HungryMan1 hungryMan1;

    static {
        HungryMan1 hungryMan1 =new HungryMan1();
    }
    
    private HungryMan1(){

    }

    public static HungryMan1 getInstance(){
        return hungryMan1;
    }
    
}
懒汉式

懒汉式,线程不安全写法

  • 优点:
    • 起到了懒加载的作用,只有用到了实例,才会创建对象
  • 缺点:
    • 只能在单线程下使用,多线程下会有线程安全问题
    • 多线程下如果A线程获取对象实例,判断对象没有实例,但是还未来得及创建对象,此时B线程也来获取对象实例,同样会判断没有实例,这样就会创建多个对象,
  • 所以在实际开发中不要使用这种方式
/**
 * 懒汉式,线程不安全
 */
public class LazyMan {

    private LazyMan() {
    }
    private static LazyMan lazMan;

    public static LazyMan getInstance() {
        if (lazMan == null) {
            // 这里增加是睡眠是为了验证线程不安全,实际上不加
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            lazMan = new LazyMan();
        }
        return lazMan;
    }

}

如下得到了两个对象

请添加图片描述

懒汉式,线程安全写法

  • 给获取对象的方法枷锁

  • 优点:保证了线程安全

  • 缺点:每次获取实例的时候都需要获取锁,然而实际上只有第一次才需要加锁,后续获取对象直接返回就好了,所以会存在性能问题

  • 所以并不推荐这种方法

/**
 * 懒汉式,线程安全写法
 */
public class LazyMan1 {

    private LazyMan1() {
    }

    private static LazyMan1 lazMan1;

    public static synchronized LazyMan1 getInstance() {
        if (lazMan1 == null) {
            // 这里增加是睡眠是为了验证线程安全,实际上不加
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            lazMan1 = new LazyMan1();
        }
        return lazMan1;
    }

}

多个线程,得到的是一个对象,线程安全

请添加图片描述

懒汉式,线程不安全,同步代码块写法

  • 在同步代码块种加锁,并没有解决线程安全问题,所以在实际开发中不要使用这种方式
/**
 * 懒汉式,线程不安全,同步代码块写法
 */
public class LazyMan2 {

    private LazyMan2() {
    }

    private static LazyMan2 lazMan2;

    public static  LazyMan2 getInstance() {
        if (lazMan2 == null) {
            // 这里增加是睡眠是为了验证线程安全,实际上不加
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            // 给代码块加锁
            synchronized (LazyMan2.class){
                lazMan2 = new LazyMan2();
            }
        }
        return lazMan2;
    }

}

双检测写法

  • 给变量加上volatile
  • 先判断对象是否存在,对象存在就直接返回,不会每次都需要加锁从而影响效率
  • 对象不存在就去获取锁,获取到锁了再次判断对象是否存在,存在就说明有其他线程初始化了对象,直接返回对象
  • 不存在就实例化对象,这样就保证了效率和线程安全,并且做到了和懒加载,推荐使用这个写法
/**
 * 懒汉式,双检测写法,线程安全
 */
public class LazyMan3 {

    private LazyMan3() {
    }

    private static volatile  LazyMan3 lazMan3;

    public static LazyMan3 getInstance() {
        // 先判断对象是否存在,存在就直接返回,不存在再去获取锁,做到了懒加载和线程安全
        if (lazMan3 == null) {
            // 给代码块加锁
            synchronized (LazyMan3.class){
                // 在同步代码块中再次判断对象收存在
                if (lazMan3 == null) {
                    lazMan3 = new LazyMan3();
                }
            }
        }
        return lazMan3;
    }

}

静态内部类的单例模式

静态内部类写法:

  • 在外部类被装载的时候,静态内部类并不会被装载
  • 只有在用到静态内部类的时候,才会去装载它,而且类装载的时候是线程安全的
  • 所以这种写法,能同时满足懒加载并且保证线程安全,推荐使用
/**
 * 单例模式,静态内部类写法
 */
public class StaticInternalClass {

    private StaticInternalClass(){
        
    }

    private static class InternalClass {
        private static final StaticInternalClass staticInternalClass = new StaticInternalClass();
    }

    public static StaticInternalClass getInstance() {
        return InternalClass.staticInternalClass;
    }

}
枚举的单例模式
  • 借助jdk1.5提供的枚举可以实现单例模式
  • 枚举可以避免多线程同步问题,并且可以防止反序列化重新创建对象
  • 这种方式是提倡的方式
public enum SingleEnum {


    RED("红色", 1),
    GREEN("绿色", 2),
    BLANK("白色", 3),
    YELLO("黄色", 4);

    // 成员变量
    private String name;
    private int index;
    // 构造方法
    private SingleEnum(String name, int index) {
        this.name = name;
        this.index = index;
    }
}

单例模式的注意事项

可以使用的写法:

  • 饿汉式写法
  • 懒汉式的双重检测
  • 静态内部类的写法
  • 枚举

单例模式的注意事项

  • 单例模式保证了系统内部中该类只存在一个对象,节省了资源,对于一些需要频繁创建和销毁的对象,使用单例模式可以提高性能
  • 单例模式使用场景:需要频繁创建和销毁对象、创建对象耗时很多或者耗费资源很多,但又经常用到的对象,例如工具类对象、数据源等

工厂设计模式

工厂模式的意义:

  • 将对象实例化的代码提取出来,进行统一管理,达到解耦的目的,提高项目的扩展和维护性

三种工厂模式

  • 简单工厂模式
  • 工厂方法模式
  • 抽象工厂模式

根据依赖倒转原则:要依赖抽象不要依赖具体,创建对象时不要直接New,而是放在一个工厂里

简单工厂模式
  • 也叫静态工厂模式,属于创建型模式,是工厂模式的一种
  • 简单工厂模式是工厂模式中最简单也是最实用的(开放银行,协议签署时就用到了工厂模式)
  • 是由一个工厂对象决定创建出哪一种产品的实例:定义一个创建对象的类,由这个类来实例化对象
  • 当我们会用到大量的创建某种对象时,就可以使用工厂模式

实例:

  • 如下,后续新增新的是实现,也只需要修改工厂类即可,使用方不需要修改,增强了可扩展性
public class Factory {

    // 根据类型创建对应的对象
    public static TestFactory getTestFactory(String type) {
        TestFactory testFactory = null;
        switch (type) {
            case "1":
                testFactory = new TestFactoryImplOne();
                break;
            case "2":
                testFactory = new TestFactoryImplTwo();
                break;
        }
        return testFactory;
    }
    
    public static void main(String[] args) {
        getTestFactory("1").test();
        getTestFactory("2").test();
    }
}



public interface TestFactory {

    void test();

}

public class TestFactoryImplOne implements TestFactory {
    @Override
    public void test() {
        System.out.println("TestFactoryImplOne 执行");
    }
}

public class TestFactoryImplTwo implements TestFactory{
    @Override
    public void test() {
        System.out.println("TestFactoryImplTwo 执行");
    }
}

工厂方法模式

工厂方法模式

  • 定义一个创建对象的抽象方法,由子类决定要实例化的类
  • 工厂方法模式将对象实例化推迟到子类
  • 在工厂方法模式中,工厂父类负责定义创建产品对象的公共接口,而工厂子类则负责生成具体的产品对象,这样做的目的是将产品类的实例化操作延迟到工厂子类中完成,即通过工厂子类来确定究竟应该实例化哪一个具体产品类。

工厂方法的模式结构

  • 抽象工厂(Factory)角色:是工厂 方法模式的核心,与应用程序无关。任何在模式中创建的对象的工厂类必须实现这个接口。
  • 具体工厂(ConcreteCreator)角色:这是实现抽象工厂接口的具体工厂类,包含与应用程序密切相关的逻辑,并且受到应用程序调用以创建产品对象。
  • 抽象产品(Product)角色:工厂方法模式所创建的对象的超类型,也就是产品对象的共同父类或共同拥有的接口。
  • 具体产品(ConcreteProduct)角色:这个角色实现了抽象产品角色所定义的接口。某具体产品有专门的具体工厂创建,它们之间往往一一对应。

工厂类

/**
 * 抽象工厂
 */
public interface Factory {

    Product createProduct(String type);

}


/**
 * 具体工厂1
 */
public class ConcreteFactory1 implements Factory {

    @Override
    public Product createProduct(String type) {
        Product product = null;
        switch (type) {
            case "A":
                product = new ProductA();
                break;
            case "B":
                product =  new ProductB();
                break;
        }
        return product;
    }
}

/**
 * 具体工厂2
 */
public class ConcreteFactory2 implements Factory {


    @Override
    public Product createProduct(String type) {
        Product product = null;
        switch (type) {
            case "C":
                product = new ProductC();
                break;
            case "D":
                product =  new ProductD();
                break;
        }
        return product;
    }
}

产品类

public interface Product {
    void test();
}

public class ProductA implements Product{
    @Override
    public void test() {
        System.out.println("执行ProductA逻辑");
    }
}

public class ProductB implements Product{
    @Override
    public void test() {
        System.out.println("执行ProductB逻辑");
    }
}

public class ProductC implements Product{
    @Override
    public void test() {
        System.out.println("执行ProductC逻辑");
    }
}

public class ProductD implements Product{
    @Override
    public void test() {
        System.out.println("执行ProductD逻辑");
    }
}
抽象工厂模式

抽象工厂模式:

  • 定义一个接口用于创建相关或有依赖关系的对象
  • 抽象工厂模式可以将简单工厂模式和工厂方法模式进行整和
  • 从设计层面看,抽象工厂模式就是对简单工厂模式的改进
  • 将工厂分为两层:抽象工厂和具体实现的工厂子类,这样可以把的单个的简单工厂变成工厂簇,更利于代码的维护和扩展

抽象工厂模式的优点包括:

  • 提供了一种封装对象创建过程的方式:抽象工厂模式将对象的创建过程封装在了具体的工厂类中,客户端只需要关注工厂接口和产品接口,无需关心具体的实现细节。
  • 实现了产品族的切换:通过使用不同的具体工厂类,可以方便地切换整个产品族的产品,从而实现了接口与实现的分离。
    保持了代码的一致性:抽象工厂模式保持了一致性,即所有由同一工厂创建的产品都相互关联,使得产品之间可以很方便地进行配合使用。

缺点:

  • 抽象工厂模式的最大缺点就是产品族扩展非常困难,我们以通用代码为例,如果要增加一个产品C,也就是说产品等级结构由原来的2个增加到3个。此时,抽象类AbstractFactory要增加一个方法createProductC(),然后具体工厂实现类都要修改,这严重违反了开闭原则,会带来较大的不便。
  • 增加了系统的复杂度:抽象工厂模式引入了多个抽象和具体类,增加了系统的复杂度和理解难度。

工厂模式的退化

  • 当抽象工厂模式中每一个具体工厂类只创建一个产品对象,也就是只存在一个产品等级结构时,抽象工厂模式退化成工厂方法模式。
  • 当工厂方法模式中抽象工厂与具体工厂合并,提供一个统一的工厂来创建产品对象,并将创建对象的工厂方法设计为静态方法时,工厂方法模式退化成简单工厂模式。

抽象工厂模式与工厂方法模式的区别

  • 抽象工厂模式与工厂方法模式最大的区别在于,工厂方法模式针对的是一个产品等级结构,而抽象工厂模式则需要面对多个产品等级结构,一个工厂等级结构可以负责多个不同产品等级结构中的产品对象的创建 。
  • 当一个工厂等级结构可以创建出分属于不同产品等级结构的一个产品族中的所有对象时,抽象工厂模式比工厂方法模式更为简单、有效率。

原型模式

java中的Object类是所有类的父类,并且提供了一个clone()方法,用于对象的拷贝,但是想要实现clone的类必须实现Cloneable接口,该接口表示该类能够复制且具有复制的能力

原型模式(prototype)

  • 原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能,属于创建型模式

  • 这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。

  • 当直接创建对象的代价比较大时,则采用这种模式简化对象的创建过程

  • 实现克隆操作:在 Java 中,实现 Cloneable 接口,重写 clone() 方法,默认为浅拷贝。

  • 但是原型模式需要给每个类配置一个克隆方法,对于新类不是问题,但是对已有的类,需要修改源代码,违背了OCP原则

/**
 * 原型模式
 */
public class PrototypeTest implements Cloneable {

    private String name;

    private Integer age;

    public PrototypeTest(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "PrototypeTest{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

    @Override
    public PrototypeTest clone() {
        try {
            return (PrototypeTest) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new AssertionError();
        }
    }
}


    /**
     * 对象的属性一样,但是对象并不是同一个
     * @param args
     */
    public static void main(String[] args) {
        PrototypeTest prototypeTest =new PrototypeTest("lisi",20);
        System.out.println(prototypeTest.hashCode()+"---"+prototypeTest);
        PrototypeTest clone = prototypeTest.clone();
        System.out.println(clone.hashCode()+"---"+clone);
    }


浅拷贝

  • 指在拷贝对象时,仅复制对象本身和其内部的基本数据类型字段,而不会复制引用类型字段所引用的对象
  • 这意味着新对象和原始对象会共享同一个引用对象,修改其中一个对象的引用字段会影响到另一个对象。
  • 简单来说,浅拷贝只是创建了一个指向原始对象的新对象的引用
  • 在‌ Java 中,实现浅拷贝的一种常见方式是通过重写对象的clone()方法,并实现Cloneable接口。这样,当调用clone()方法时,会创建一个新的对象,但新对象和原始对象共享相同的引用
  • 浅拷贝的特点是效率较高且节省内存,因为它只是复制了对象的引用而不是对象本身。然而,这也意味着对原始对象的修改会影响到拷贝对象,反之亦然。因此,浅拷贝适用于那些不需要独立修改的对象,或者对象的结构相对简单的情况。

深拷贝(Deep Copy):

  • 深拷贝是指在拷贝对象时,不仅复制对象本身和其内部的基本数据类型字段,还会递归复制引用类型字段所引用的对象。这样,新对象和原始对象将完全独立,对任何一个对象的修改都不会影响到另一个对象。
  • 简而言之,深拷贝会创建一个全新的对象及其所有关联的对象。
  • 实现深拷贝的方式可以是通过实现Cloneable接口并重写clone()方法,或者使用序列化和反序列化等方法来复制对象及其引用的对象。需要根据具体的需求选择适合的方式进行深拷贝操作。
  • 需要注意的是,浅拷贝和深拷贝的概念适用于对象的拷贝操作,而不同于对象的赋值操作。在赋值操作中,无论是基本数据类型还是引用类型,都只是将一个对象的引用赋值给了另一个对象,它们仍然指向同一个对象,修改其中一个对象会影响到另一个对象。

深拷贝的实现方式:

  • clone
  • 序列化和反序列化:先把对象序列化为流,再把流反序列化变成对象,这样就可以完成深拷贝

clone方式:

@AllArgsConstructor
public class User implements Cloneable {

    private String name;

    private Role role;

    public void setRole(Role role) {
        this.role = role;
    }

    public Role getRole() {
        return role;
    }

    @Override
    public User clone() {
        try {
            User clone = (User) super.clone();
            // 把role对象单独复制一份
            clone.setRole(clone.getRole().clone());
            return clone;
        } catch (CloneNotSupportedException e) {
            throw new AssertionError();
        }
    }
}

@AllArgsConstructor
public class Role implements Cloneable{

    // 角色名
    private String roleName;

    // 权限
    private String power;

    @Override
    public Role clone() {
        try {
            Role clone = (Role) super.clone();
            return clone;
        } catch (CloneNotSupportedException e) {
            throw new AssertionError();
        }
    }
}

测试方法:

    public static void main(String[] args) {
        User user = new User("lisi", new Role("admin", "all"));
        System.out.println(user.hashCode() + "---" + user + ",role:" + user.getRole().hashCode());
        User clone = user.clone();
        System.out.println(clone.hashCode() + "---" + clone + ",role:" + user.getRole().hashCode());
    }


可以看到两个对象是相互独立的

请添加图片描述

建造者模式

建造者模式

  • 建造者模式是一种创建型设计模式,它的主要目的是将一个复杂对象的构建过程与其表示相分离,从而可以创建具有不同表示形式的对象。
  • 使用场景:需要生成的对象具有复杂的内部结构,需要生成的对象内部属性相互依赖。

优点

  • 分离构建过程和表示,使得构建过程更加灵活,可以构建不同的表示:客户端(使用者)不需要知道产品内部组成的细节,将产品本身和产品的创建过程解耦,使得相同的创建过程可以创建不同的产品对象

  • 可以更好地控制构建过程,隐藏具体构建细节。

  • 每一个建造者都是相对独立的,可以很方便的替换具体建造者或增加新的具体建造者

  • 代码复用性高,可以在不同的构建过程中重复使用相同的建造者。

缺点

  • 如果产品的属性较少,建造者模式可能会导致代码冗余。
  • 增加了系统的类和对象数量。

建造者模式包含以下几个主要角色:

  • 产品(Product):要构建的复杂对象。产品类通常包含多个部分或属性。
  • 抽象建造者(Builder):定义了构建产品的抽象接口,包括构建产品的各个部分的方法。
  • 具体建造者(Concrete Builder):实现抽象建造者接口,具体确定如何构建产品的各个部分,并负责返回最终构建的产品。
  • 指导者(Director):负责调用建造者的方法来构建产品,指导者并不了解具体的构建过程,只关心产品的构建顺序和方式。

建造者模式与工厂模式的区别

  • 工厂方法模式注重是整体对象的创建方式;
  • 而建造者模式注重的是部件一步一步的创建出一个复杂对象的过程

示例:

/**
 * 产品
 */
@Data
public class House {

    // 地基
    private String foundation;

    // 墙壁
    private String wall;

    // 屋顶
    private String roof;

    // 装修
    private String renovation;

}

/**
 * 抽象建造者
 */
public interface Builder {

    // 构建地基
    String foundation();

    // 构建墙壁
    String wall();

    // 构建屋顶
    String roof();

    // 构建装修
    String renovation();
    
}

/**
 * 具体建造者1
 */
public class BigBuilderImpl implements Builder {

    @Override
    public String foundation() {
        return "大房子,打五米地基";
    }

    @Override
    public String wall() {
        return "用混凝土浇筑墙体";
    }

    @Override
    public String roof() {
        return "房顶是预制板";
    }

    @Override
    public String renovation() {
        return "装修很豪华";
    }
}


/**
 * 具体建造者2
 */
public class SmallBuilderImpl implements Builder {


    @Override
    public String foundation() {
        return "小房子,打两米地基";
    }

    @Override
    public String wall() {
        return "砌砖墙";
    }

    @Override
    public String roof() {
        return "房顶是水泥板";
    }

    @Override
    public String renovation() {
        return "装修比较简陋";
    }
}

/**
 * 指导者,复则构建产品
 */
public class Director {

    private Builder builder;

    public Director(Builder builder) {
        this.builder = builder;
    }

    public House createSession() {
        House house = new House();
        house.setFoundation(builder.foundation());
        house.setWall(builder.wall());
        house.setRoof(builder.roof());
        house.setRenovation(builder.renovation());
        return house;
    }
    
}

// 测试类
public class Test {

    public static void main(String[] args) {
        
        House h0 = new Director(new BigBuilderImpl()).createHouse();
        System.out.println(h0);

        House h1 = new Director(new SmallBuilderImpl()).createHouse();
        System.out.println(h1);
    }
}

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

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

相关文章

Unity编辑器扩展之Project视图扩展

内容将会持续更新,有错误的地方欢迎指正,谢谢! Unity编辑器扩展之Project视图扩展 TechX 坚持将创新的科技带给世界! 拥有更好的学习体验 —— 不断努力,不断进步,不断探索 TechX —— 心探索、心进取&#xff01…

河南萌新联赛2024第(六)场:郑州大学(补题ABCDFGIL)

文章目录 河南萌新联赛2024第(六)场:郑州大学A 装备二选一(一)简单介绍:思路:代码: B 百变吗喽简单介绍:思路:代码: C 16进制世界简单介绍&#x…

第二十七节、人物可互动标识

一、多个场景同时存在 方法:将另一个场景拖拽进 当前场景中 这样在一个场景中保存物体,另一个场景切换即可 创建一个场景名为上图(这是一个持久化的场景) 被激活的场景是粗体字的 二、代码 使用第二个代码获得player的子物体 …

uniapp在线下载安装包更新app以及热更新

首先用getSystemInfo判断平台、 再通过json文件模拟接口 判断版本号是否一致 不一致则下载服务器apk进行更新 外加网络波动导致失败重新下载更新包 uni.getSystemInfo({success: function (e) {// #ifndef H5// 获取手机系统版本const system e.system.toLowerCase();const pl…

《黑神话:悟空》解锁+35项修改器,开启上帝模式!

哈喽,各位小伙伴们好,我是给大家带来各类黑科技与前沿资讯的小武。 8 月 20 日,国产 3A 大作《黑神话:悟空》闪亮登场!这是一款由中国游戏开发商游戏科学开发的动作角色扮演游戏,此前在研发阶段就备受关注…

Python 使用 matplotlib 显示图像

如果没有安装 matplotlib 需要先安装: pip install matplotlib一、读取图片并显示 import matplotlib.pyplot as pltimage_path "/Users/AlanWang4523/Desktop/Debug/files/image.png" image_array plt.imread(image_path)plt.figure("ImageShow…

会声会影作为视屏制作软件如何?会声会影最新免费版

二、核心功能与特点 拖放式编辑:会声会影2024提供了拖放式标题、转场、覆叠和滤镜功能,使得视频编辑变得更加直观和高效。 色彩分级与动态分屏:软件支持色彩分级功能,可以对视频进行精细的色彩调整。同时,动态分屏功能…

STM32通用定时器,端口复用和重映射

STM32定时器是一种内置在STM32微控制器中的硬件模块,用于测量和控制时间。它具有高精度、可配置性和灵活性的特点,能够支持多种不同的工作模式和应用场景。以下是对STM32定时器的详细讲解: 一、定时器的基本构成 STM32定时器主要由以下几个…

内存管理————基础理解

目录 我们先看以下代码和内存分布图 对一些名词的解释说明: 题目 答案 问题解释: C语言中动态内存管理方式 C语言中动态内存管理方式:malloc/calloc/realloc/free 这里提一下要注意的点 C内存管理方式 基本内容 new/delete操作内置…

常用API:object

文章目录 Object类toString()方法equals()方法总结其他方法 黑马学习笔记 Object类 是所有类的父类,所有的类都默认继承Object类。Java中所有的类的对象都可以直接使用Object类提供的一些方法。 toString()方法 equals()方法 默认是判断两个对象的地址 也是判断…

MATLAB-疲劳检测

首先,我们需要解决一个问题,什么样可以视为疲劳? 频繁的眨眼/眨一次眼所用时间很长 不停的打哈欠 不停的点头 通过分析,我们得到了检测疲劳的三个标准:眨眼,打哈欠,头部姿态。 这三个标准可以…

OpenCV findTours函数及其用法

OpenCV findTours函数主要用以寻找图像中物体的轮廓,其原型如下: indTours 函数参数: Image 输入图像,需8位单通道图像。非零像素被视为1。零像素保持为0,因 此图 像被视为 二进制。您可以使用co…

【数据分享】《全国能源-分行业煤油消费总量》(2000-2020年)

而今天要限时免费分享的数据就是2000-2020年间出版的《全国能源-分行业煤油消费总量》并以多格式提供免费下载。(无需分享朋友圈即可获取) 数据介绍 在过去的二十年里,中国作为全球能源消费的重要力量,其能源结构的变化不仅影…

基于x86 平台opencv的图像采集和seetaface6的图像质量评估功能

目录 一、概述二、环境要求2.1 硬件环境2.2 软件环境三、开发流程3.1 编写测试3.2 配置资源文件3.2 验证功能一、概述 本文档是针对x86 平台opencv的图像采集和seetaface6的图像质量评估功能,opencv通过摄像头采集视频图像,将采集的视频图像送给seetaface6的图像质量评估模块…

21.1 基于Netty实现聊天

21.1 基于Netty实现聊天 一. 章节概述二. `Netty`介绍三. 阻塞与非阻塞1. 阻塞与非阻塞简介2. BIO同步阻塞3. NIO同步非阻塞4. AIO异步非阻塞IO5. 异步阻塞IO(用的极少)6. 总结四. Netty三种线程模型1. 单线程模型2. 多线程模型3. 主从线程模型五. 构建Netty服务器************…

DDL——三范式与表约束

目录 一、三大范式 1.范式的定义 2.第一范式 3.第二范式 4.第三范式 二、表约束 1.约束的定义 2.非空约束(not null) 3.唯一性约束(unique) 4.主键约束(primary key) 5.外键约束(fo…

可集成多模型的机器人开发框架 dora:让机器人编程走向大众

作者 | Annie Xu 责编 | 何苗 出品 | GOSIM 开源创新汇 C 和 C是机器人编程中常用的编程语言,但学起来存在一定难度。那些对机器人感兴趣,但没有精力花费数周时间学习 C 和 C编程的用户只能望而却步。 为了让机器人技术变得更简单,陶海轩&a…

AWS SAM CLI 备忘单!

安装 AWS SAM CLI brew tap aws/tap brew 安装 aws-sam-cli 验证安装 $ sam --version 升级 SAM $ brew upgrade aws-sam-cli 您需要 AWS 凭证才能在 AWS 上工作。 构建并部署简单应用程序 $ sam init→ 下载示例应用程序 $ sam build→ 构建您的应用程序 $ sam deploy --guid…

机器学习:决策树回归树实现

1、决策树: 是一种树形结构,用于通过一系列的是非问题来预测目标值。在决策树回归中,树的叶子节点代表预测的连续值。 2、数据准备: 收集数据集,并对其进行清洗和预处理。 将数据集分为特征(X)…

git-20240822

目录 初始化仓库 Git init Git init project --bare 查看提交的记录 git log --prettyoneline 查看当前git远程库地址 git remote -v 查看详细提交记录 git log 撤出暂存区的文件 git reset HEAD file(.代表全部文件) 提交数据到远程仓库 git config --global push.…