用JavaFX写了一个简易的管理系统

文章目录

  • 前言
  • 正文
    • 一、最终效果
      • 1.1 主页面
      • 1.2 动物管理页面-初始化
      • 1.3 动物管理页面-修改&新增
      • 1.4 动物管理页面-删除&批量删除
    • 二、核心代码展示
      • 2.1 启动类
      • 2.2 数据库配置-db.setting
      • 2.3 日志文本域组件
      • 2.4 自定义表格视图组件
      • 2.5 自定义分页组件
      • 2.6 动物管理页面
      • 2.7 页面工厂
  • 附录
    • 附1:sql脚本

前言

自学一下Java FX ,用Java代码写一个简易的客户端。

本文项目是Maven项目,使用了Java17,以及mysql。
代码仓库:https://gitee.com/fengsoshuai/java-fx-management-system-demo/tree/dev-with-db/

正文

一、最终效果

1.1 主页面

启动项目后,自动展示主页面。
在这里插入图片描述

1.2 动物管理页面-初始化

支持分页查询,搜索,新增,修改,删除,批量删除等功能。
整体分为以下几层:

  • 操作按钮,包含新增,修改,删除,刷新列表;
  • 搜索框,支持搜索动物的几个常用信息;
  • 列表,展示动物数据(分页);
  • 分页功能相关按钮,支持首页,下一页,最后一页,跳转到指定页,切换每页大小;
  • 操作记录,记录操作日志,内存存储(有兴趣的可以改为数据库存储),支持清空记录;

在这里插入图片描述

1.3 动物管理页面-修改&新增

在这里插入图片描述

1.4 动物管理页面-删除&批量删除

使用ctrl+鼠标左键选择要删除的数据。然后点击删除按钮。
出现弹窗后,点击确定,即可删除。
在这里插入图片描述

二、核心代码展示

2.1 启动类

渲染主页,左侧菜单栏,以及基本的布局。

package org.feng.demofx;

import javafx.application.Application;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.*;
import javafx.scene.paint.Color;
import javafx.stage.Screen;
import javafx.stage.Stage;
import org.feng.demofx.pages.AnimalManagePage;
import org.feng.demofx.pages.MainPage;
import org.feng.demofx.sys.*;
import org.feng.demofx.util.ArrayUtil;
import org.feng.demofx.util.StyleUtil;

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

/**
 * 启动类
 *
 * @author feng
 */
public class StartApplication extends Application {

    /**
     * 屏幕宽度
     */
    public static final Double SCREEN_WIDTH;

    /**
     * 屏幕高度
     */
    public static final Double SCREEN_HEIGHT;

    /**
     * 左侧菜单宽度
     */
    public static final Double LEFT_MENU_WIDTH;

    /**
     * root容器,使用HBox类型
     */
    private HBox root;

    /**
     * 左侧菜单栏
     */
    private VBox leftMenu;

    /**
     * 当前页的节点
     */
    private Node currentPageNode;

    /**
     * 左侧菜单的当前索引
     */
    private Integer currentMenuIndex;
    /**
     * 左侧菜单的临时索引,选中时的菜单项索引
     */
    private Integer tempIndex;

    /**
     * 左侧菜单面板背景颜色
     */
    private static final Color LEFT_MENU_BACKGROUND_COLOR = Color.web("#FFF8DC");
    /**
     * 左侧菜单按钮背景颜色
     */
    private static final Color LEFT_MENU_BUTTON_BACKGROUND_COLOR = Color.web("#DEB887");
    /**
     * 左侧菜单按钮hover时(鼠标移动到按钮上)的颜色
     */
    private static final Color LEFT_MENU_BUTTON_HOVER_BACKGROUND_COLOR = Color.web("D2691E");
    /**
     * 左侧菜单按钮点击时的字体颜色
     */
    private static final Color LEFT_MENU_BUTTON_CLICK_BACKGROUND_COLOR = Color.web("#6495ed");

    static {
        SCREEN_WIDTH = Screen.getPrimary().getBounds().getWidth();
        SCREEN_HEIGHT = Screen.getPrimary().getBounds().getHeight();
        LEFT_MENU_WIDTH = 240.0;
    }


    @Override
    public void start(Stage stage) throws Exception {
        // 注册页面渲染器
        registerPageNodeRender();

        root = new HBox();
        // 渲染左侧菜单栏
        leftMenu = renderLeftMenu(root);
        root.getChildren().add(leftMenu);

        // 主面板
        currentPageNode = routePage(root, MainPage.PAGE_KEY);
        HBox.setHgrow(currentPageNode, Priority.ALWAYS);
        root.getChildren().add(currentPageNode);
        StyleUtil.setPaneBackground(root, Color.WHITE);

        Scene scene = new Scene(root, SCREEN_WIDTH * 0.8, SCREEN_HEIGHT * 0.8);

        // 窗口的标题
        stage.setTitle("管理系统");
        stage.setScene(scene);
        // 禁止拖拽窗口大小
        stage.setResizable(false);
        stage.show();
    }


    private VBox renderLeftMenu(HBox root) {
        VBox vbox = new VBox();
        vbox.setMinHeight(root.getPrefHeight());
        vbox.setMinWidth(LEFT_MENU_WIDTH);
        StyleUtil.setPaneBackground(vbox, LEFT_MENU_BACKGROUND_COLOR);
        // 增加菜单中的项目
        vbox.getChildren().addAll(getLeftMenuItemList());
        return vbox;
    }

    /**
     * 生成左侧菜单按钮
     */
    private List<Button> getLeftMenuItemList() {
        double buttonHeight = 30;
        List<Button> buttonList = new ArrayList<>(3);
        Map<String, String> pageNameMap = PageNodeRenderFactory.getPageNameMap();
        String[] itemNames = pageNameMap.values().toArray(new String[0]);

        for (String name : itemNames) {
            Button button = new Button(name);
            button.setMinWidth(LEFT_MENU_WIDTH);
            button.setMinHeight(buttonHeight);
            StyleUtil.setButtonBackground(button, LEFT_MENU_BUTTON_BACKGROUND_COLOR, Color.WHITE);
            // 增加鼠标移动到菜单上到hover效果
            button.setOnMouseMoved(event -> {
                StyleUtil.setButtonBackground(button, LEFT_MENU_BUTTON_HOVER_BACKGROUND_COLOR, Color.WHITE);
                StyleUtil.setFont(button, Color.web("#E6A23C"), -1);
            });
            button.setOnMouseExited(event -> {
                if (currentMenuIndex == null || !button.getText().equals(itemNames[currentMenuIndex])) {
                    StyleUtil.setButtonBackground(button, LEFT_MENU_BUTTON_BACKGROUND_COLOR, Color.WHITE);
                } else {
                    StyleUtil.setButtonBackground(button, LEFT_MENU_BUTTON_BACKGROUND_COLOR, Color.WHITE);
                }
            });
            button.setOnMouseClicked(event -> {
                currentMenuIndex = ArrayUtil.getIndexForArray(itemNames, button.getText());
                currentPageNode = routePage(root, PageNodeRenderFactory.getPageKeyByName(name));
                root.getChildren().remove(1);    //清除右侧页面路由组件节点
                HBox.setHgrow(currentPageNode, Priority.ALWAYS);
                root.getChildren().add(currentPageNode);
                StyleUtil.setFont(button, Color.WHITE, -1);
                // 选中状态逻辑
                if (tempIndex != null) {
                    Button node = (Button) leftMenu.getChildren().get(tempIndex);
                    // 清空选中状态样式
                    StyleUtil.setFont(node, Color.WHITE, -1);
                    StyleUtil.setButtonBackground(node, LEFT_MENU_BUTTON_BACKGROUND_COLOR, Color.WHITE);
                }
                // 设置选中样式
                StyleUtil.setFont(button, LEFT_MENU_BUTTON_CLICK_BACKGROUND_COLOR, -1);
                tempIndex = currentMenuIndex;
            });
            buttonList.add(button);
        }
        return buttonList;
    }

    /**
     * 右侧页面路由
     */
    private Node routePage(Pane root, String pageKey) {
        return PageNodeRenderFactory.newInstance(pageKey).render(root);
    }

    private void registerPageNodeRender() {
        PageNodeRenderFactory.registerNodeRender(MainPage.class);
        PageNodeRenderFactory.registerNodeRender(AnimalManagePage.class);

        PageNodeRenderFactory.fillPageNameMap();
    }

    public static void main(String[] args) {
        launch();
    }
}

2.2 数据库配置-db.setting

在这里插入图片描述

#------------------------------------------------------------------------------------------
## 基本配置信息
# JDBC URL,根据不同的数据库,使用相应的JDBC连接字符串
url = jdbc:mysql://localhost:3306/fx_demo?useUnicode=true&serverTimezone=UTC
# 用户名,此处也可以使用 user 代替
username = root
# 密码,此处也可以使用 pass 代替
password = root
# JDBC驱动名,可选(Hutool会自动识别)
# driver = com.mysql.jdbc.Driver

## 可选配置
# 是否在日志中显示执行的SQL
showSql = true
# 是否格式化显示的SQL
formatSql = false
# 是否显示SQL参数
showParams = true
# 打印SQL的日志等级,默认debug
sqlLevel = debug
#------------------------------------------------------------------------------------------

2.3 日志文本域组件

对应于效果中的“操作记录”文本域。

package org.feng.demofx.pages;

import javafx.geometry.Insets;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.TextArea;
import javafx.scene.layout.VBox;
import javafx.scene.text.Font;
import javafx.scene.text.FontWeight;
import lombok.Getter;
import org.feng.demofx.util.ControlUtil;
import org.feng.demofx.util.TimeUtil;

import java.time.LocalDateTime;

/**
 * 日志文本域组件
 *
 * @author feng
 */
@Getter
public class LogTextAreaComponent {

    private final TextArea textArea;
    private final VBox logBox;

    public LogTextAreaComponent(double height, double width) {
        // 控制台输入操作记录
        textArea = new TextArea();
        textArea.setScrollTop(Double.MAX_VALUE);
        textArea.setMaxHeight(height);
        textArea.setMaxWidth(width);
        textArea.setEditable(false);
        textArea.setStyle("-fx-text-fill: red");

        // 日志盒子
        logBox = new VBox();
        logBox.setMaxWidth(width);

        logBox.getChildren().add(new Label(""));

        Label operLogLabel = new Label("操作记录:");
        operLogLabel.setFont(Font.font(null, FontWeight.BOLD, 15));
        operLogLabel.setPadding(new Insets(5, 0, 5, 0));

        logBox.getChildren().add(operLogLabel);
        logBox.getChildren().add(textArea);

        // 清空日志记录按钮
        logBox.getChildren().add(new Label(""));
        Button clearLog = ControlUtil.genClearLogButton();
        logBox.getChildren().add(clearLog);
        clearLog.setOnMouseClicked(event -> {
            textArea.setText("");
        });
    }

    /**
     * 记录日志
     *
     * @param text 日志内容
     */
    public void log(String text) {
        textArea.appendText("【" + TimeUtil.parse(LocalDateTime.now()) + "】" + text + "\r\n");
    }
}

2.4 自定义表格视图组件

package org.feng.demofx.pages;

import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TableView;
import lombok.Data;
import org.feng.demofx.StartApplication;
import org.feng.demofx.util.TableColumnUtil;
import org.feng.demofx.vo.AnimalVo;

import java.util.List;

/**
 * 自定义表格视图组件
 *
 * @author 01434188
 */
@Data
public class CustomTableViewComponent<T> {

    /**
     * 表格视图
     */
    private TableView<T> tableView;

    private ObservableList<T> data;

    public CustomTableViewComponent(Class<T> beanClass, List<T> dataList) {
        tableView = new TableView<>();

        // 设置可编辑
        tableView.setEditable(true);
        // 表格宽度
        tableView.setMaxWidth(((StartApplication.SCREEN_WIDTH * 0.8) - StartApplication.LEFT_MENU_WIDTH) * 0.8);
        // 设置可选择多行数据
        tableView.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
        // 解析表头和列信息
        TableColumnUtil.parseColumnsByClass(tableView.getColumns(), beanClass);
        // 添加表格数据
        data = FXCollections.observableArrayList();
        data.addAll(dataList);
        tableView.setItems(data);
    }


    public void resetData(List<T> newDataList) {
        ObservableList<T> list = FXCollections.observableArrayList();
        list.addAll(newDataList);
        data = list;
        tableView.setItems(list);
    }
}

2.5 自定义分页组件

package org.feng.demofx.pages;

import javafx.geometry.Insets;
import javafx.scene.Node;
import javafx.scene.control.Button;
import javafx.scene.control.ComboBox;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import javafx.scene.layout.GridPane;
import javafx.scene.layout.HBox;
import javafx.scene.layout.Priority;
import javafx.scene.layout.Region;
import lombok.Data;
import org.feng.demofx.util.ControlUtil;

/**
 * 自定义分页组件
 *
 * @author feng
 */
@Data
public class CustomPagingComponent {

    /**
     * 每页条数
     */
    private ComboBox<String> pageSize;

    /**
     * 当前页
     */
    private TextField currentPage;

    /**
     * 上一页
     */
    private Button lastPageButton;
    /**
     * 下一页
     */
    private Button nextPageButton;

    /**
     * 跳转
     */
    private Button skipPageButton;

    /**
     * 首页
     */
    private Button firstPageButton;
    /**
     * 最后一页
     */
    private Button finallyPageButton;

    /**
     * 分页面板
     */
    private GridPane pagingGridPane;

    /**
     * 总页数
     */
    private Label pageCount;

    public CustomPagingComponent(double width, double height) {
        this(width, height, null, null);
    }

    public CustomPagingComponent(double width, double height, Integer pageCount, Integer pageSize) {
        pagingGridPane = new GridPane();
        pagingGridPane.setMaxWidth(width);
        pagingGridPane.setMinWidth(width);
        pagingGridPane.setMaxHeight(height);
        pagingGridPane.setMinHeight(height);
        pagingGridPane.setPadding(new Insets(8, 0, 8, 0));

        // 设置下拉框,选择每页条数;默认15条
        this.pageSize = new ComboBox<>();
        this.pageSize.getItems().addAll("10", "15", "30", "50", "100");
        this.pageSize.setValue(pageSize == null ? "15" : String.valueOf(pageSize));

        // 当前页默认设置为1
        currentPage = new TextField();
        currentPage.setText("1");
        currentPage.setMaxWidth(35);
        // 上一页,下一页
        lastPageButton = ControlUtil.pagingButton("上一页");
        nextPageButton = ControlUtil.pagingButton("下一页");
        skipPageButton = ControlUtil.pagingButton("跳转");
        firstPageButton = ControlUtil.pagingButton("首页");
        finallyPageButton = ControlUtil.pagingButton("最后一页");

        // 总页数
        this.pageCount = new Label(String.format(" %s ", pageCount == null ? "0" : String.valueOf(pageCount)));

        // 设置垂直间距,水平间距
        pagingGridPane.setVgap(5);
        pagingGridPane.setHgap(3);

        pagingGridPane.add(new Label("总页数:"), 0, 0);
        pagingGridPane.add(this.pageCount, 1, 0);
        Node spacer1 = new Region();
        HBox.setHgrow(spacer1, Priority.ALWAYS);
        pagingGridPane.add(spacer1, 2, 0);

        pagingGridPane.add(firstPageButton, 3, 0);
        pagingGridPane.add(lastPageButton, 4, 0);
        pagingGridPane.add(currentPage, 5, 0);
        pagingGridPane.add(nextPageButton, 6, 0);
        pagingGridPane.add(finallyPageButton, 7, 0);
        pagingGridPane.add(skipPageButton, 8, 0);

        Node spacer2 = new Region();
        HBox.setHgrow(spacer2, Priority.ALWAYS);
        pagingGridPane.add(spacer2, 9, 0);
        pagingGridPane.add(this.pageSize, 10, 0);
    }

    public int getCurrentPageValue() {
        return Integer.parseInt(this.getCurrentPage().getText());
    }

    public void setCurrentPageValue(int currentPageValue) {
        this.getCurrentPage().setText(String.valueOf(currentPageValue));
    }

    public int getPageSizeValue() {
        return Integer.parseInt(this.getPageSize().getValue());
    }

    public void setPageSizeValue(int pageSizeValue) {
        this.getPageSize().setValue(String.valueOf(pageSizeValue));
    }

    public void setPageCountValue(int pageCountValue) {
        this.getPageCount().setText(String.valueOf(pageCountValue));
    }
}

2.6 动物管理页面

package org.feng.demofx.pages;

import cn.hutool.core.bean.BeanUtil;
import javafx.collections.ObservableList;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.Node;
import javafx.scene.control.*;
import javafx.scene.layout.*;
import javafx.scene.paint.Color;
import lombok.Data;
import org.feng.demofx.StartApplication;
import org.feng.demofx.dao.impl.AnimalDao;
import org.feng.demofx.entity.bo.AnimalBO;
import org.feng.demofx.enums.AnimalSexEnum;
import org.feng.demofx.enums.AnimalTypeEnum;
import org.feng.demofx.service.AnimalService;
import org.feng.demofx.service.impl.AnimalServiceImpl;
import org.feng.demofx.sys.PageNodeKey;
import org.feng.demofx.sys.PageNodeRender;
import org.feng.demofx.util.*;
import org.feng.demofx.vo.AnimalVo;

import java.time.LocalDate;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;

/**
 * 动物管理界面
 *
 * @author feng
 */
@PageNodeKey(value = AnimalManagePage.PAGE_KEY, pageName = "动物管理")
public class AnimalManagePage implements PageNodeRender {
    public static final String PAGE_KEY = "animalManage";

    private final AnimalService animalService;
    public AnimalManagePage() {
        animalService = new AnimalServiceImpl(new AnimalDao());
    }

    @Override
    public Node render(Pane pane) {
        VBox vbox = new VBox();
        // 当前容器内,布局居中,局上
        vbox.setAlignment(Pos.TOP_CENTER);
        // 当前面板背景颜色
        StyleUtil.setPaneBackground(vbox, Color.web("#fffff0"));
        // 页面标题
        vbox.getChildren().add(genPageTitle("动物管理页面"));
        final double width = ((StartApplication.SCREEN_WIDTH * 0.8) - StartApplication.LEFT_MENU_WIDTH) * 0.8;

        // 操作按钮,
        VBox oper = new VBox();
        HBox hBox = new HBox();
        hBox.setMaxWidth(250);
        oper.getChildren().add(hBox);
        Button addButton = ControlUtil.genAddButton();
        Button updateButton = ControlUtil.genUpdateButton();
        Button deleteButton = ControlUtil.genDeleteButton();
        Button refreshButton = ControlUtil.genRefreshButton();
        // 设置外边距,上、右,下,左
        oper.setPadding(new Insets(10, 0, 12, 0));
        oper.setAlignment(Pos.TOP_LEFT);
        oper.setMaxWidth(width);
        oper.setMaxHeight(200);

        Node spacer1 = new Region();
        HBox.setHgrow(spacer1, Priority.ALWAYS);
        Node spacer2 = new Region();
        HBox.setHgrow(spacer2, Priority.ALWAYS);
        Node spacer3 = new Region();
        HBox.setHgrow(spacer3, Priority.ALWAYS);

        hBox.getChildren().add(addButton);
        hBox.getChildren().add(spacer1);
        hBox.getChildren().add(updateButton);
        hBox.getChildren().add(spacer2);
        hBox.getChildren().add(deleteButton);
        hBox.getChildren().add(spacer3);
        hBox.getChildren().add(refreshButton);

        vbox.getChildren().add(oper);

        //  搜索面板
        SearchAnimalPane searchAnimalPane = new SearchAnimalPane(oper.getMaxWidth());
        vbox.getChildren().add(searchAnimalPane.getSearchPane());
        // 分页组件
        CustomPagingComponent customPagingComponent = new CustomPagingComponent(width, 60);
        // 控制台输入操作记录
        LogTextAreaComponent logger = new LogTextAreaComponent(400, width);
        // 表格视图
        PageBean<AnimalVo> pageBean = new PageBean<>(customPagingComponent.getPageSizeValue(),
                customPagingComponent.getCurrentPageValue(), 0);
        animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
        customPagingComponent.setPageCountValue(pageBean.getSumPageCount());
        CustomTableViewComponent<AnimalVo> tableViewComponent = new CustomTableViewComponent<>(AnimalVo.class, pageBean.getData());

        TableView<AnimalVo> animalTableView = tableViewComponent.getTableView();
        // 表格展示数据+分页
        vbox.getChildren().add(animalTableView);
        vbox.getChildren().add(customPagingComponent.getPagingGridPane());
        // 分隔线
        vbox.getChildren().add(ControlUtil.separator(animalTableView.getMaxWidth()));
        // 操作记录
        vbox.getChildren().add(logger.getLogBox());

        // 注册刷新
        refreshButton.setOnAction(event -> {
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 更改每页大小-事件处理
        customPagingComponent.getPageSize().setOnAction(event -> {
            int pageSize = customPagingComponent.getPageSizeValue();
            logger.log(String.format("修改每页大小为:%s,重新刷新列表!", pageSize));
            // 设置每页大小
            pageBean.setPageSize(pageSize);
            // 重新分页查询第一页数据
            refreshToFirstPage(pageBean, searchAnimalPane, customPagingComponent);
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 上一页
        customPagingComponent.getLastPageButton().setOnAction(event -> {
            // 当前页
            int currentPage = customPagingComponent.getCurrentPageValue();
            String canQuery = pageBean.setCurrentPageNo(currentPage - 1);
            if (StringUtil.isNotEmpty(canQuery)) {
                logger.log(canQuery);
                return;
            }
            animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
            customPagingComponent.setPageCountValue(pageBean.getSumPageCount());
            customPagingComponent.setCurrentPageValue(currentPage - 1);
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 下一页
        customPagingComponent.getNextPageButton().setOnAction(event -> {
            // 当前页
            int currentPage = customPagingComponent.getCurrentPageValue();
            String canQuery = pageBean.setCurrentPageNo(currentPage + 1);
            if (StringUtil.isNotEmpty(canQuery)) {
                logger.log(canQuery);
                return;
            }
            animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
            customPagingComponent.setCurrentPageValue(currentPage + 1);
            customPagingComponent.setPageCountValue(pageBean.getSumPageCount());
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 首页事件
        customPagingComponent.getFirstPageButton().setOnAction(event -> {
            refreshToFirstPage(pageBean, searchAnimalPane, customPagingComponent);
            logger.log("跳转首页成功!");
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 最后一页
        customPagingComponent.getFinallyPageButton().setOnAction(event -> {
            pageBean.setCurrentPageNo(pageBean.getSumPageCount());
            animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
            customPagingComponent.setPageCountValue(pageBean.getSumPageCount());
            customPagingComponent.setCurrentPageValue(pageBean.getSumPageCount());
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 跳转事件
        customPagingComponent.getSkipPageButton().setOnAction(event -> {
            int currentPage = customPagingComponent.getCurrentPageValue();
            if(currentPage < 1) {
                currentPage = 1;
            }
            pageBean.setCurrentPageNo(currentPage);
            animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
            // 如果输入的当前页大于总页数,修复跳转至最后一页
            if(currentPage > pageBean.getSumPageCount()) {
                currentPage = pageBean.getSumPageCount();
                pageBean.setCurrentPageNo(currentPage);
                animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
            }

            customPagingComponent.setPageCountValue(pageBean.getSumPageCount());
            customPagingComponent.setCurrentPageValue(currentPage);
            logger.log(String.format("跳转第【%s】页成功!", currentPage));
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 当选择发生变化时调用该函数-选择某一行时
        animalTableView.getSelectionModel().selectedItemProperty().addListener((obs, oldVal, newVal) -> {
            if (newVal != null && !animalTableView.getSelectionModel().isSelected(newVal.getId())) {
                logger.log("已选择 " + newVal.getName());
            }
        });

        // 注册搜索
        searchAnimalPane.getSearchButton().setOnAction(event -> {
            refreshToFirstPage(pageBean, searchAnimalPane, customPagingComponent);
            logger.log("搜索成功!");
            // 刷新列表
            tableViewComponent.resetData(pageBean.getData());
            ControlUtil.refreshTableView(animalTableView, logger);
        });

        // 注册删除
        deleteButton.setOnAction(ControlUtil.alertConfirm("确认删除吗?", "", buttonType -> {
            ButtonBar.ButtonData buttonData = buttonType.getButtonData();
            if (buttonData.isDefaultButton()) {
                // 确认删除
                ObservableList<AnimalVo> items = animalTableView.getSelectionModel().getSelectedItems();
                List<Long> ids = items.stream().mapToLong(vo -> Long.valueOf(vo.getId())).boxed().toList();
                logger.log("正在删除 " + items.stream().map(AnimalVo::getName).collect(Collectors.joining(",")));
                animalService.delete(ids);

                // 刷新列表
                refreshToFirstPage(pageBean, searchAnimalPane, customPagingComponent);
                tableViewComponent.resetData(pageBean.getData());
                ControlUtil.refreshTableView(animalTableView, logger);
                logger.log("删除 " + items.stream().map(AnimalVo::getName).collect(Collectors.joining(",")) + "成功!");
            }
        }));

        // 注册新增
        addButton.setOnAction(event -> {
            AddAndUpdateAnimalDialog addDialog = new AddAndUpdateAnimalDialog(null);
            addDialog.getDialog().showAndWait().ifPresent(buttonType -> {
                if (buttonType.getButtonData().isDefaultButton()) {
                    logger.log("开始新增数据");
                    // 获取文本框中数据
                    AnimalBO animalBO = addDialog.genAnimalBo();
                    logger.log("正在新增 " + animalBO.getName());
                    // 添加动物
                    animalService.add(animalBO);

                    // 刷新列表
                    refreshToFirstPage(pageBean, searchAnimalPane, customPagingComponent);
                    tableViewComponent.resetData(pageBean.getData());
                    ControlUtil.refreshTableView(animalTableView, logger);
                    logger.log("新增 " + animalBO.getName() + "成功!");
                }
            });
            addDialog.getDialog().close();
        });

        // 注册更新
        updateButton.setOnAction(event -> {
            ObservableList<AnimalVo> selectedItems = animalTableView.getSelectionModel().getSelectedItems();
            if (selectedItems == null || selectedItems.size() != 1) {
                logger.log("请选择一条数据进行修改");
                return;
            }
            // 获取选中的数据原值
            AnimalVo selectedAnimal = selectedItems.get(0);
            AddAndUpdateAnimalDialog updateDialog = new AddAndUpdateAnimalDialog(selectedAnimal);
            // 注册点击事件
            updateDialog.getDialog().showAndWait().ifPresent(buttonType -> {
                if (buttonType.getButtonData().isDefaultButton()) {
                    logger.log("开始修改数据");
                    // 获取文本框中数据
                    AnimalBO animalBO = updateDialog.genAnimalBo();
                    animalBO.setId(selectedAnimal.getId().longValue());

                    logger.log(String.format("正在修改ID为【%s】,名字为【%s】 ", selectedAnimal.getId(), updateDialog.getOldAnimal().getName()));

                    // 更新动物
                    animalService.update(animalBO);
                    // 刷新列表
                    refreshToFirstPage(pageBean, searchAnimalPane, customPagingComponent);
                    tableViewComponent.resetData(pageBean.getData());
                    ControlUtil.refreshTableView(animalTableView, logger);
                    logger.log(String.format("修改完成! ID为【%s】,修改名字【%s】为【%s】,动物类型【%s】为【%s】,年龄【%s】为【%s】", selectedAnimal.getId(), updateDialog.getOldAnimal().getName(), animalBO.getName(),
                            updateDialog.getOldAnimal().getAnimalType(), animalBO.getAnimalType().getDesc(), updateDialog.getOldAnimal().getAge(), animalBO.getAge()));
                }
            });
            updateDialog.getDialog().close();
        });

        // 动物管理页面初始化完成,刷新一次
        ControlUtil.refreshTableView(animalTableView, logger);
        return vbox;
    }

    /**
     * 刷新列表数据至第一页
     */
    private void refreshToFirstPage(PageBean<AnimalVo> pageBean, SearchAnimalPane searchAnimalPane, CustomPagingComponent customPagingComponent) {
        pageBean.setCurrentPageNo(1);
        animalService.listPage(searchAnimalPane.genAnimalBo(), pageBean);
        customPagingComponent.setPageCountValue(pageBean.getSumPageCount());
        customPagingComponent.setCurrentPageValue(1);
    }


    @Data
    private static class SearchAnimalPane {
        // 折叠面板
        private TitledPane searchPane;

        private TextField searchNameTextField;
        private TextField searchAgeTextField;
        private ComboBox<String> searchTypeComboBox;
        private ComboBox<String> searchSexComboBox;

        private DatePicker createDateStart;
        private DatePicker createDateEnd;
        private DatePicker updateDateStart;
        private DatePicker updateDateEnd;

        private Button searchButton;

        public SearchAnimalPane(double width) {
            // 定义搜索网格面板
            GridPane searchGridPane = new GridPane();
            // 设置垂直间距,水平间距
            searchGridPane.setVgap(10);
            searchGridPane.setHgap(5);
            searchNameTextField = new TextField();
            searchAgeTextField = new TextField();
            searchTypeComboBox = new ComboBox<>();
            searchTypeComboBox.getItems().addAll(Arrays.stream(AnimalTypeEnum.values()).map(AnimalTypeEnum::getDesc).toList());
            searchSexComboBox = new ComboBox<>();
            searchSexComboBox.getItems().addAll(Arrays.stream(AnimalSexEnum.values()).map(AnimalSexEnum::getDesc).toList());

            searchButton = ControlUtil.genSearchButton();
            Button clearButton = ControlUtil.genClearButton();
            createDateStart = new DatePicker();
            createDateEnd = new DatePicker();
            updateDateStart = new DatePicker();
            updateDateEnd = new DatePicker();
            createDateStart.setEditable(true);
            createDateEnd.setEditable(true);
            updateDateStart.setEditable(true);
            updateDateEnd.setEditable(true);
            searchGridPane.add(new Label("名字 "), 0, 0);
            searchGridPane.add(searchNameTextField, 1, 0);
            searchGridPane.add(new Label("年龄 "), 0, 1);
            searchGridPane.add(searchAgeTextField, 1, 1);
            searchGridPane.add(new Label("类型 "), 0, 2);
            searchGridPane.add(searchTypeComboBox, 1, 2);
            searchGridPane.add(new Label("性别 "), 0, 3);
            searchGridPane.add(searchSexComboBox, 1, 3);
            searchGridPane.add(new Label("创建日期起止 "), 0, 4);
            searchGridPane.add(new Label("更新日期起止 "), 0, 5);
            searchGridPane.add(createDateStart, 1, 4);
            searchGridPane.add(createDateEnd, 2, 4);
            searchGridPane.add(updateDateStart, 1, 5);
            searchGridPane.add(updateDateEnd, 2, 5);
            searchGridPane.add(clearButton, 0, 7);
            searchGridPane.add(searchButton, 1, 7);

            // 折叠面板-搜索
            searchPane = new TitledPane("高级搜索", searchGridPane);
            // 设置外边距,上、右,下,左
            searchPane.setPadding(new Insets(0, 0, 8, 0));
            searchPane.setMaxWidth(width);
            // 设置可折叠
            searchPane.setCollapsible(true);
            // 设置折叠时的动画
            searchPane.setAnimated(true);

            // 注册清空参数事件
            clearButton.setOnAction(event -> {
                clearSearchParam();
            });
        }

        public void clearSearchParam() {
            searchNameTextField.setText("");
            searchAgeTextField.setText("");
            searchTypeComboBox.setValue("");
            searchSexComboBox.setValue("");
            createDateStart.setValue(null);
            createDateEnd.setValue(null);
            updateDateStart.setValue(null);
            updateDateEnd.setValue(null);
        }

        public AnimalBO genAnimalBo() {
            String searchNameText = this.getSearchNameTextField().getText();
            String searchAgeText = this.getSearchAgeTextField().getText();
            String searchTypeValue = this.getSearchTypeComboBox().getValue();
            String searchSexValue = this.getSearchSexComboBox().getValue();
            LocalDate createDateStartValue = this.getCreateDateStart().getValue();
            LocalDate createDateEndValue = this.getCreateDateEnd().getValue();
            LocalDate updateDateStartValue = this.getUpdateDateStart().getValue();
            LocalDate updateDateEndValue = this.getUpdateDateEnd().getValue();

            AnimalVo animalVo = new AnimalVo();
            if (StringUtil.isNotEmpty(searchNameText)) {
                animalVo.setName(searchNameText);
            }
            if (StringUtil.isNotEmpty(searchAgeText)) {
                animalVo.setAge(Integer.parseInt(searchAgeText));
            }
            if (StringUtil.isNotEmpty(searchTypeValue)) {
                animalVo.setAnimalType(searchTypeValue);
            }
            if (StringUtil.isNotEmpty(searchSexValue)) {
                animalVo.setAnimalSex(searchSexValue);
            }

            // 创建时间起、止选择后生效
            if (Objects.nonNull(createDateStartValue) || Objects.nonNull(createDateEndValue)) {
                animalVo.setCreateDateRange(new DateRange(createDateStartValue, createDateEndValue));
            }
            // 更新时间起、止选择后生效
            if (Objects.nonNull(updateDateStartValue) || Objects.nonNull(updateDateEndValue)) {
                animalVo.setUpdateDateRange(new DateRange(updateDateStartValue, updateDateEndValue));
            }

            return animalVo.toAnimalBo();
        }
    }

    @Data
    private static class AddAndUpdateAnimalDialog {

        private Dialog<ButtonType> dialog;

        private TextField nameTextField;
        private TextField ageTextField;
        private ComboBox<String> typeComboBox;
        private ComboBox<String> sexComboBox;
        private TextArea remarkTextArea;

        private AnimalVo oldAnimal;

        public AddAndUpdateAnimalDialog(AnimalVo animalVo) {
            Label nameLabel = new Label("名字 ");
            nameTextField = new TextField();
            Label ageLabel = new Label("年龄 ");
            ageTextField = new TextField();
            Label typeLabel = new Label("类型 ");
            typeComboBox = new ComboBox<>();
            typeComboBox.getItems().addAll(Arrays.stream(AnimalTypeEnum.values()).map(AnimalTypeEnum::getDesc).toList());
            Label sexLabel = new Label("性别 ");
            sexComboBox = new ComboBox<>();
            sexComboBox.getItems().addAll(Arrays.stream(AnimalSexEnum.values()).map(AnimalSexEnum::getDesc).toList());
            Label remarkLabel = new Label("备注 ");
            remarkTextArea = new TextArea();
            remarkTextArea.setScrollTop(Double.MAX_VALUE);
            remarkTextArea.setEditable(true);
            remarkTextArea.setMaxHeight(250);
            remarkTextArea.setMaxWidth(300);

            if (animalVo == null) {
                typeComboBox.setValue(AnimalTypeEnum.UN_KNOW.getDesc());
                sexComboBox.setValue(AnimalSexEnum.UN_KNOW.getDesc());
            } else {
                oldAnimal = new AnimalVo();
                BeanUtil.copyProperties(animalVo, oldAnimal);

                // 数据回显
                nameTextField.setText(oldAnimal.getName());
                ageTextField.setText(String.valueOf(oldAnimal.getAge()));
                typeComboBox.setValue(oldAnimal.getAnimalType());
                sexComboBox.setValue(oldAnimal.getAnimalSex());
                remarkTextArea.setText(oldAnimal.getRemark());
            }

            // 定义网格面板
            GridPane gridPane = new GridPane();
            // 设置垂直间距,水平间距
            gridPane.setVgap(10);
            gridPane.setHgap(5);
            gridPane.add(nameLabel, 0, 0);
            gridPane.add(nameTextField, 1, 0);
            gridPane.add(ageLabel, 0, 1);
            gridPane.add(ageTextField, 1, 1);
            gridPane.add(typeLabel, 0, 2);
            gridPane.add(typeComboBox, 1, 2);
            gridPane.add(sexLabel, 0, 3);
            gridPane.add(sexComboBox, 1, 3);
            gridPane.add(remarkLabel, 0, 4);
            gridPane.add(remarkTextArea, 1, 4);

            DialogPane dialogPane = new DialogPane();
            dialogPane.setPrefHeight(300);
            dialogPane.setPrefWidth(500);
            dialogPane.setMaxHeight(400);
            dialogPane.setMaxWidth(550);

            dialogPane.setContent(gridPane);
            dialogPane.getButtonTypes().addAll(ButtonType.CANCEL, ButtonType.OK);
            dialog = new Dialog<>();
            dialog.setDialogPane(dialogPane);
        }

        public AnimalBO genAnimalBo() {
            AnimalBO animalBO = new AnimalBO();
            // 年龄
            animalBO.setAge(Integer.parseInt(this.getAgeTextField().getText()));
            // 姓名
            animalBO.setName(this.getNameTextField().getText());
            // 获取下拉框的数据:动物类型
            String type = this.getTypeComboBox().getValue();
            animalBO.setAnimalType(AnimalTypeEnum.getByDesc(type));
            // 获取下拉框的数据:性别
            String sex = this.getSexComboBox().getValue();
            animalBO.setAnimalSex(AnimalSexEnum.getByDesc(sex));
            // 获取文本域内容:备注
            animalBO.setRemark(this.getRemarkTextArea().getText());
            return animalBO;
        }
    }
}

2.7 页面工厂

package org.feng.demofx.sys;

import org.feng.demofx.util.StringUtil;

import java.lang.reflect.InvocationTargetException;
import java.util.*;

/**
 * PageNodeRender工厂
 *
 * @author feng
 */
public class PageNodeRenderFactory {

    private static final Map<String, PageNodeRender> PAGE_NODE_RENDER_MAP = new HashMap<>(8);

    private static final Set<PageNodeDefine> PAGE_NODE_DEFINES = new TreeSet<>(Comparator.comparingInt(PageNodeDefine::getOrder)
            .thenComparing(PageNodeDefine::getPageKey));

    private static final Map<String, String> PAGE_NAME_MAP = new LinkedHashMap<>();
    private static final Map<String, String> RESERVE_PAGE_NAME_MAP = new LinkedHashMap<>();

    public static PageNodeRender newInstance(String pageKey) {
        return PAGE_NODE_RENDER_MAP.get(pageKey);
    }

    public static Map<String, String> getPageNameMap() {
        return PAGE_NAME_MAP;
    }

    public static String getPageKeyByName(String pageName) {
        return RESERVE_PAGE_NAME_MAP.get(pageName);
    }


    public static void registerNodeRender(Class<? extends PageNodeRender> renderClass) {
        try {
            // 处理PageNodeRender#getPageKey()
            PageNodeRender pageNodeRender = renderClass.getDeclaredConstructor().newInstance();
            String pageKey = pageNodeRender.getPageKey();
            if (StringUtil.isNotEmpty(pageKey)) {
                PAGE_NODE_RENDER_MAP.put(pageKey, pageNodeRender);

                PageNodeDefine pageNodeDefine = new PageNodeDefine();
                pageNodeDefine.setPageKey(pageKey);
                pageNodeDefine.setPageName(pageNodeRender.getPageName());
                pageNodeDefine.setOrder(pageNodeRender.order());
                PAGE_NODE_DEFINES.add(pageNodeDefine);
                return;
            }

            // 处理PageNodeKey注解
            if (renderClass.isAnnotationPresent(PageNodeKey.class)) {
                PageNodeKey pageNodeKey = renderClass.getAnnotation(PageNodeKey.class);
                pageKey = pageNodeKey.value();
                if (StringUtil.isNotEmpty(pageKey)) {
                    PAGE_NODE_RENDER_MAP.put(pageKey, pageNodeRender);

                    PageNodeDefine pageNodeDefine = new PageNodeDefine();
                    pageNodeDefine.setPageKey(pageKey);
                    pageNodeDefine.setPageName(pageNodeKey.pageName());
                    pageNodeDefine.setOrder(pageNodeKey.order());
                    PAGE_NODE_DEFINES.add(pageNodeDefine);
                    return;
                }
            }

            throw new IllegalArgumentException(renderClass.getName() + "定义错误,请检查");
        } catch (NoSuchMethodException | InvocationTargetException | InstantiationException |
                 IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }


    public static void fillPageNameMap() {
        PAGE_NODE_DEFINES.forEach(pageNodeDefine -> {
            PAGE_NAME_MAP.put(pageNodeDefine.getPageKey(), pageNodeDefine.getPageName());
            RESERVE_PAGE_NAME_MAP.put(pageNodeDefine.getPageName(), pageNodeDefine.getPageKey());
        });
    }
}

附录

附1:sql脚本

CREATE TABLE `fx_animal`
(
    `id`          BIGINT       NOT NULL AUTO_INCREMENT,
    `name`        VARCHAR(255) NOT NULL COMMENT '名字',
    `animal_type` INT          NOT NULL DEFAULT - 1 COMMENT '动物类型,-1 未知',
    `animal_sex`  INT          NOT NULL DEFAULT - 1 COMMENT '动物性别,-1 未知 0 雄性 1雌性',
    `age`         INT          NOT NULL DEFAULT 0 COMMENT '年龄',
    `update_time` datetime     NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
    `create_time` datetime     NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
    `remark`      VARCHAR(255) NOT NULL DEFAULT '' COMMENT '备注',
    PRIMARY KEY (`id`)
);

INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (1, '小名', 0, 1, 2, '2024-01-19 23:10:36', '2024-01-19 22:31:53', '呜呜呜汪汪汪');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (2, '小红', 1, 0, 0, '2024-01-19 23:50:17', '2024-01-19 22:31:59', '00000');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (3, '啊华', 1, 1, 2, '2024-01-19 23:13:46', '2024-01-19 23:13:46', '喵喵喵');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (4, '小吗', 1, 1, 2, '2024-01-22 23:34:09', '2024-01-22 23:34:09', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (5, '待机', -1, -1, 2, '2024-01-22 23:34:56', '2024-01-22 23:34:56', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (6, '松紧带哦哦', -1, -1, 21, '2024-01-22 23:35:05', '2024-01-22 23:35:05', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (7, '额温枪', -1, -1, 21, '2024-01-22 23:35:10', '2024-01-22 23:35:10', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (8, '大萨达', -1, -1, 2, '2024-01-22 23:35:16', '2024-01-22 23:35:16', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (9, '哇打算', -1, -1, 21, '2024-01-22 23:35:21', '2024-01-22 23:35:21', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (10, '大东方闪电', -1, -1, 22, '2024-01-22 23:35:37', '2024-01-22 23:35:37', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (11, '2多发点', -1, -1, 4322, '2024-01-22 23:35:50', '2024-01-22 23:35:50', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (12, '绕弯儿', -1, -1, 32, '2024-01-22 23:37:23', '2024-01-22 23:37:23', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (13, '法国掉头发', -1, -1, 32, '2024-01-22 23:37:30', '2024-01-22 23:37:30', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (14, '43给对方刚刚', -1, -1, 32, '2024-01-22 23:37:36', '2024-01-22 23:37:36', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (15, '手动挡', -1, -1, 2, '2024-01-22 23:37:45', '2024-01-22 23:37:45', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (16, '发得分', -1, -1, 32, '2024-01-22 23:37:53', '2024-01-22 23:37:53', '');
INSERT INTO `fx_demo`.`fx_animal`(`id`, `name`, `animal_type`, `animal_sex`, `age`, `update_time`, `create_time`, `remark`) VALUES (17, '明明', 1, 1, 13, '2024-01-24 21:03:58', '2024-01-24 21:03:21', '修改了一次年龄');


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

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

相关文章

Git教程学习:09 Git分支

文章目录 1 分支的简介2 分支的相关操作2.1 分支的创建2.2 分支的切换2.3 分支的合并2.4 分支推送到远程2.5 分支的删除2.6 分支的重命名 3 分支开发工作流程3.1 长期分支3.2 短期分支 1 分支的简介 几乎所有的版本控制系统都以某种形式支持分支。使用分支意味着我们可以把我们…

计算机硬件 6.1BIOS

第六章 计算机基本程序 第一节 BIOS与CMOS芯片 一、认识BIOS 1.中文含义&#xff1a;基本输入输出系统。 2.材质&#xff1a;ROM&#xff08;Flash Rom&#xff09; 3.地位&#xff1a;是操作系统与硬件之间的接口。 4.存放内容&#xff1a;①基本输入输出系统&#xff1b;…

自动化防DDoS脚本

简介 DDoS &#xff08;分布式拒绝服务攻击&#xff09;是一种恶意的网络攻击&#xff0c;旨在通过占用目标系统的资源&#xff0c;使其无法提供正常的服务。在DDoS攻击中&#xff0c;攻击者通常控制大量的被感染的计算机或其他网络设备&#xff0c;同时将它们协调起来向目标系…

行业分析|中国人工智能发展的优势与差距

​人工智能&#xff0c;被誉为第四次工业革命的催化剂&#xff0c;吸引着发达国家和众多科技公司大举投入研发。我国积极构筑人工智能发展的先发优势&#xff0c;党的二十大报告提出推动战略性新兴产业集群&#xff0c;构建一系列新的增长引擎&#xff0c;包括信息技术、人工智…

基于变异混合蛙跳算法的车间调度最优化matlab仿真,可以任意调整工件数和机器数,输出甘特图

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.本算法原理 4.1 车间调度问题&#xff08;JSSP&#xff09;描述 4.2 蛙跳算法&#xff08;SFLA&#xff09;基本原理 4.2.1 初始化 4.2.2 局部搜索 4.2.3 全局信息交换 4.2.4 变异策略 4.2.5 终止…

编写后端代码,使用yakit将任意字典进行编码发送,后端解码输出到页面上

挂上代理&#xff08;小狐狸&#xff09;&#xff1a; yakit中挂上劫持&#xff1a; 编写后端代码&#xff1a; 一个简单的登录代码 访问页面&#xff1a; 给予传参后看yakit劫持了没有 yakit劫持&#xff1a; 将劫持到的数据发送到webFuzzer&#xff1a; 右键选择标签/字典—…

c\c++队列的链式表示(对小白友好)

文章目录 1.链式队列的定义2.初始化3.判断空4.入队5. 出队6.打印全部元素7.源代码 本篇中的链式表示都是带头结点的链式表示。 1.链式队列的定义 typedef struct LinkNode { //链式队列的结点int data;struct LinkNode *next; }LinkNode; typedef struct { //链式…

如何创建以业务为中心的AI?

AI是企业的未来&#xff0c;这一趋势越来越明显。各种AI模型可以帮助企业节省时间、提高效率并增加收入。随着越来越多的企业采用AI&#xff0c;AI很快就不再是一种可有可无的能力&#xff0c;而是企业参与市场竞争的必备能力。 然而&#xff0c;作为一名业务决策者&#xff0c…

pcl之滤波器(二)

pcl滤波器 pcl一共是有十二个主要模块&#xff0c;详细了解可以查看官网。https://pcl.readthedocs.io/projects/tutorials/en/latest/#basic-usage 今天学习一下pcl的滤波器模块。 滤波器模块&#xff0c;官网一共是提供了6个例程&#xff0c;今天看第三个、第四个。 滤波…

(学习日记)2024.01.23:结构体、位操作和枚举类型

写在前面&#xff1a; 由于时间的不足与学习的碎片化&#xff0c;写博客变得有些奢侈。 但是对于记录学习&#xff08;忘了以后能快速复习&#xff09;的渴望一天天变得强烈。 既然如此 不如以天为单位&#xff0c;以时间为顺序&#xff0c;仅仅将博客当做一个知识学习的目录&a…

【学网攻】 第(3)节 -- 交换机配置聚合端口

文章目录 【学网攻】 第(1)节 -- 认识网络【学网攻】 第(2)节 -- 交换机认识及使用 前言 网络已经成为了我们生活中不可或缺的一部分&#xff0c;它连接了世界各地的人们&#xff0c;让信息和资源得以自由流动。随着互联网的发展&#xff0c;我们可以通过网络学习、工作、娱乐…

最新综述!3D Gaussian Splatting

作者&#xff1a;小柠檬 | 来源&#xff1a;3DCV 在公众号「3DCV」后台&#xff0c;回复「原论文」可获取论文 文章介绍了3D高斯喷洒在场景重建和渲染中的应用&#xff0c;并探讨了其在机器学习和计算机视觉领域的潜在应用。文章还提供了3D高斯喷洒的基本原理和优化方法&#x…

MoEs学习

和多任务学习的mmoe很像哦&#xff08;有空再学习一下&#xff09;moe layer的起源&#xff1a;Switch Transformers paper MoE moe由两个结构组成&#xff1a; Moe Layer &#xff1a;这些层代替了传统 Transformer 模型中的前馈网络 (FFN) 层。MoE 层包含若干“专家”(例如…

解读顺网算力与AI,破局AIGC落地“最后一公里”

全球知名AI科学家吴恩达和李飞飞在CES 2024上预测&#xff0c;2024年将是AI技术继续深化的一年&#xff0c;将成为下一次数字或工业革命真正的变革性驱动力。吴恩达还预测了2024年AI可能的突破性进展&#xff0c;其中包括边缘AI。吴恩达对边缘AI寄予厚望&#xff0c;他认为在笔…

luceda ipkiss教程 57:画微环调制器

案例分享&#xff1a;画微环调制器 全部代码如下&#xff1a; from si_fab import all as pdk from ipkiss3 import all as i3class DC(i3.PCell):straight_length i3.PositiveNumberProperty(default200)radius i3.PositiveNumberProperty(default50)spacing i3.Positive…

坚持刷题 |对称二叉树

文章目录 题目考察点代码实现实现总结扩展用迭代的方式判断是否为对称二叉树递归和迭代的对比可能的扩展提问 坚持刷题&#xff0c;老年痴呆追不上我&#xff0c;今天真的好累&#xff0c;就不难为自己了&#xff0c;刷个简单级别的吧&#xff1a;对称二叉树 题目 101.对称二叉…

Scapy编程指南(基础概念)

Scapy编程指南&#xff08;基础概念&#xff09; Scapy是什么 Scapy是Python中一个非常强大的库&#xff0c;它专门用于处理、发送和捕获网络协议中的数据包&#xff0c;它允许开发人员通过Python代码构建、解析和发送自定义网络协议的数据包。Scapy提供了一种直观、灵活的方…

luffy商城项目(二)

路飞后端配置 二次封装response drf提供的Response对象&#xff0c;不能很方便的加入code和msg字段&#xff0c;自己封装一个Response类&#xff0c;以后都用我们自己封装的&#xff0c;方便咱们写code和msg 封装步骤&#xff1a; 1 在utils/common_response.py from rest_…

GitLab升级版本(任意用户密码重置漏洞CVE-2023-7028)

目录 前言漏洞分析影响范围查看自己的GitLab版本升级路程 升级过程13.1.1113.8.8 - 14.0.1214.3.614.9.5 - 16.1.6 前言 最近GitLab发了个紧急漏洞需要修复&#xff0c;ok接到命令立刻着手开始修复&#xff0c;在修复之前先大概了解一下这个漏洞是什么东西 漏洞分析 1、组件…

【立创EDA-PCB设计基础】6.布线铺铜实战及细节详解

前言&#xff1a;本文进行布线铺铜实战及详解布线铺铜的细节 在本专栏中【立创EDA-PCB设计基础】前面完成了布线铺铜前的设计规则的设置&#xff0c;接下来进行布线 布局原则是模块化布局&#xff08;优先布局好确定位置的器件&#xff0c;例如排针、接口、主控芯片&#xff…