Java高手速成│实战:应用数据库和GUI开发产品销售管理软件(1)

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
云数据库 RDS PostgreSQL,高可用系列 2核4GB
简介: 实战项目:应用数据库和GUI开发产品销售管理软件

image.png

01、项目分析

应用各种数据库编程技术,并利用GUI组件,例如按钮、标签、文本字段提供增添、更新、删除产品销售记录等功能。利用JTable显示产品销售数据表中的记录。图1显示了这个实战项目的典型GUI窗口、数据表和操作功能按钮。

image.png


▍图1 实战项目典型GUI窗口、数据表和操作功能按钮

02、类的设计

ButtonPanle——创建包括添加记录(Add)、更新记录(Update)、删除记录(Delete)以及停止运行(Stop)这四个GUI组件,利用布局管理将它们显示到窗口底部。并执行事件处理以及异常处理功能。

JDBCProductFrame——创建JTable对象用来显示记录的数据表、提供与数据库连接以及发送各种SQL指令的方法,执行将回传结果显示到数据表中的各种功能。

JDBCProductFrameApp——测试程序运行这个实战程序。

如下是JDBCProductFrame的主要代码:

//创建与数据库连接的有GUI组件用来执行显示、添加、更新、删除、退出操作的窗口
class JDBCProductFrame extends JFrame {
 String columnNames[] = { "Product Code", "Product Name", "Product Price" };//定义产品字段名
   String records[][]; //储存数据表
   String record[] = new String[3]; //储存选择的记录
   int rows = 0; //总记录行初始化
   int row = 0; //当前记录行初始化
   JTable table; //声明JTable
   DefaultTableModel model; //声明表模式
   JScrollPane scrollPane; //声明滑标
   JPanel panel; //声明控制板
   Connection connection; //声明连接
   Statement stmt = null ; //初始化
   ResultSet rs = null;
   JDBCProductFrame() { //构造方法
     makeJDBCConnection(); //调用自定义方法与数据库连接
     getResult(); //调用自定义方法得到数据表
     buildRecordTable(); //调用自定义方法建立数据表
    model = new DefaultTableModel(records, columnNames);//创建数据表显示
     table = new JTable(model); //创建表
     scrollPane = new JScrollPane(table); //创建滑标
     add(scrollPane, BorderLayout.CENTER); //注册显示表
     panel = new ButtonPanel(); //创建按钮控制板
     add(panel, BorderLayout.SOUTH); //注册显示控制板
    setDefaultCloseO

在JDBCProductFrame构造方法中,分别调用了3个自定义方法makeJDBCConnection()、getResult()、buildRecordTable()来完成对数据库连接、得到数据表以及设立用来储存数据表的二维数组records各元素值的任务。makeJDBCConnection()的代码如下:

private void makeJDBCConnection() { //自定义方法连接数据库
  try{
    Class.forName("com.mysql.cj.jdbc.Driver");
    String dbURL = "jdbc:mysql://localhost:3306/ProductDB";
    String username = "root";
    String password = "********";
    connection = DriverManager.getConnection(dbURL, username, password);
    stmt = connection.createStatement();
  }
  catch(ClassNotFoundException e){
     JOptionPane.showMessageDialog(null, "JDBC driver is not found.");
  }
  catch(SQLException e){
  JOptionPane.showMessageDialog(null,"Error: " + e.getMessage());
  }

在JDBCProductFrame构造方法中,分别调用了3个自定义方法makeJDBCConnection()、getResult()、buildRecordTable()来完成对数据库连接、得到数据表以及设立用来储存数据表的二维数组records各元素值的任务。makeJDBCConnection()的代码如下:

private void makeJDBCConnection() { //自定义方法连接数据库
  try{
    Class.forName("com.mysql.cj.jdbc.Driver");
    String dbURL = "jdbc:mysql://localhost:3306/ProductDB";
    String username = "root";
    String password = "********";
    connection = DriverManager.getConnection(dbURL, username, password);
    stmt = connection.createStatement();
  }
  catch(ClassNotFoundException e){
     JOptionPane.showMessageDialog(null, "JDBC driver is not found.");
  }
  catch(SQLException e){
  JOptionPane.showMessageDialog(null,"Error: " + e.getMessage());
  }

getResult()以及buildRecords()的代码如下:

private void getResult() { //自定义方法得到数据表
  try {
    rs = stmt.executeQuery("SELECT * FROM Products"); //选择产品数据表中的所有记录
  }
  catch (SQLException e) {
    JOptionPane.showMessageDialog(null, "Error in SQL statement...");
  }
}
private void buildRecordTable() { //自定义方法建立记录数组
  try {
    rs.last(); //记录指示器到最后记录
    rows = rs.getRow(); //得到记录数
    records = new String[rows][3]; //创建二维数组储存记录表
    int row = 0;
    rs.beforeFirst(); //设置记录指示器
    while (rs.next()) { //如果有下一个记录
      records[row][0] = rs.getString(1); //设置记录到数组
      records[row][1] = rs.getString(2);
      records[row][2] = "" + rs.getDouble(3);
      row++; //下一行记录
    }
    rs.close();
  }
  catch (SQLException e) {
    JOptionPane.showMessageDialog(null, " Error in SQL statement...");
  }
}

为了得到记录表中的总记录数,首先调用ResultSet的last()方法,将记录指示器指向最后一个记录,然后调用其getRow()得到总记录数。在提取rs中的记录之前,还必须重设记录指示器,以便在while循环中利用next()方法,控制记录的读出操作。

在内部类ButtonPanel中,创建和设置了所有GUI组件、事件处理以及布局管理功能。主要代码如下:

//这个控制板用来实现按钮和文本字段组件的创建、布局、事件处理等与记录操作有关的功能
class ButtonPanel extends JPanel implements ActionListener {
//内部类创建控制GUI组件
  JButton addButton, updateButton, deleteButton, submitButton, sendButton, returnButton, exitButton;
  JLabel codeLabel, titleLabel, priceLabel;
  JTextField codeField, titleField, priceField;
  FlowLayout flowLayout;
  String message = "You must select a record in the table first...";
  ButtonPanel() { //构造方法
    setupGUI(); //调用自定义方法设置组件
  }
  public void actionPerformed(ActionEvent e) { //完善事件处理功能
    Object source = e.getSource();
    if (source == addButton) { //如果是添加记录
      setUpdateComponents(); //调用自定义方法重设其他组件的显示
      submitButton.setVisible(true); //显示发送按钮
      sendButton.setVisible(false); //不显示提交更新按钮
    }
    else if(source == updateButton) { //如果是更新记录
      setUpdateComponents(); //调用自定义方法重设其他组件的显示
      sendButton.setVisible(true); //显示提交更新按钮
      submitButton.setVisible(false); //不显示发送按钮
      setUpdateRecord(); //调用自定义方法设置更新的记录
      model.removeRow(row); //将旧的记录从显示表中删除
    }
    else if(source == sendButton) { //如果是提交更新按钮
      updateRecord(); //调用自定义方法更新记录
      getLastRecord(); //调用自定义方法得到更新后的记录
      model.insertRow(row, record); //将这个记录加入显示表中原来位置
      clearFields(); //取出各字段的内容
      resetComponents(); //调用自定义方法重设组件显示
    }
    else if(source == deleteButton) { //如果是删除按钮
       deleteRecord(); //调用自定义方法删除记录
    }
    else if( source == submitButton) { //如果是添加记录的发送按钮
      String code = codeField.getText(); //得到记录的各字段值
      String title = titleField.getText();
      double price = Double.parseDouble(priceField.getText());
      insertRecord(code, title, price); //调用自定义方法加入记录
      setLastRecord(); //设置添加后的记录到数组record中
      clearFields(); //清除各字段的显示
        model.addRow(record); //将这个记录加在显示表的尾部
    }
     else if(source == returnButton) { //如果是返回按钮
       clearFields(); //清除各字段显示
       resetComponents(); //重设组件显示
    }
     else if(source == exitButton) { //如果是退出按钮
       System.exit(0); //结束程序运行
     }
  }

代码中,自定义方法setupGUI()将所有控制组件,例如按钮以及文本字段,布局以及显示到窗口的适当位置,并注册各按钮的事件处理。这里不再详细讨论其具体代码。

在事件处理代码中,由于添加记录和更新记录执行不同的操作,所以创建发送按钮submitButton来处理增加新记录的事件;而利用提交按钮sendButton处理更新记录的操作。

如果用户按下了添加记录按钮addButton,将调用自定义方法setUpdateComponents(),设置如图2的窗口显示,用来处理增添记录的操作。

image.png


▍图2 用来执行添加记录操作的窗口

setUpdateComponents()的显示协调部分代码如下:

//自定义方法设置更新的组件显示
private void setUpdateComponents() {
  codeLabel.setVisible(true);
  titleLabel.setVisible(true);
  priceLabel.setVisible(true);
  codeField.setVisible(true);
  titleField.setVisible(true);
  priceField.setVisible(true);
  addButton.setVisible(false);
  updateButton.setVisible(false);
  deleteButton.setVisible(false);
  exitButton.setVisible(false);
  returnButton.setVisible(true);
}

当用户输入记录信息并按下发送按钮后,将触发submitButton事件。得到3个文本字段的内容后,调用自定义方法insertRecord(),把这些字段值添加到数据表中。这个方法的代码如下:

//利用预备指令将记录添加到记录表中
private void insertRecord(String code, String titl, double price) { //自定义方法
  try {
    String insertSql = "INSERT INTO Products (Code, Title, Price) VALUES( ?, ?, ?)";//预备指令
    PreparedStatement ps = connection.prepareStatement(insertSql); //执行预备指令
    ps.setString(1, code);
    ps.setString(2, title);
    ps.setDouble(3, price);
    ps.executeUpdate(); //执行SQL指令
  ps.close();
    }
    catch (SQLException e) {
     JOptionPane.showMessageDialog(null, "Error in SQL statement...");
    }
}

可以看到,三个问号分别被赋予产品代码、产品名称以及产品价格的值。

除此之外,还必须更新显示表中的内容,使之显示新增添的记录。这个操作通过首先调用自定义方法setLastRecord(),并且调用DefaultTableModel的方法addRow()来实现。setLastRecord()的代码如下:

//自定义方法将新记录内容设置到数组record
private void setLastRecord() {
  record[0] = codeField.getText(); //record[0]储存产品代码
  record[1] = titleField.getText(); //record[1]储存产品名称
  record[2] = priceField.getText(); //record[2]储存产品价格
 }

最后清除文本字段,调用另外一个自定义方法resetComponents(),重设按钮的显示。具体代码与以上讨论的updateComponents()方法基本相同。你可参考这个例子,或者查阅ButtonPanel完整程序。其他对记录的操作,如更新记录、删除记录以及退出程序运行,遵循与添加记录操作相同的原则!

03、源代码下载

链接文字

相关实践学习
每个IT人都想学的“Web应用上云经典架构”实战
本实验从Web应用上云这个最基本的、最普遍的需求出发,帮助IT从业者们通过“阿里云Web应用上云解决方案”,了解一个企业级Web应用上云的常见架构,了解如何构建一个高可用、可扩展的企业级应用架构。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。   相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情: https://www.aliyun.com/product/rds/mysql 
目录
相关文章
|
1月前
|
SQL 关系型数据库 MySQL
阿里云RDS云数据库全解析:产品功能、收费标准与活动参考
与云服务器ECS一样,关系型数据库RDS也是很多用户上云必买的热门云产品之一,阿里云的云数据库RDS主要包含RDS MySQL、RDS SQL Server、RDS PostgreSQL、RDS MariaDB等几个关系型数据库,并且提供了容灾、备份、恢复、监控、迁移等方面的全套解决方案,帮助您解决数据库运维的烦恼。本文为大家介绍阿里云的云数据库 RDS主要产品及计费方式、收费标准以及活动等相关情况,以供参考。
|
1月前
|
存储 人工智能 NoSQL
AI大模型应用实践 八:如何通过RAG数据库实现大模型的私有化定制与优化
RAG技术通过融合外部知识库与大模型,实现知识动态更新与私有化定制,解决大模型知识固化、幻觉及数据安全难题。本文详解RAG原理、数据库选型(向量库、图库、知识图谱、混合架构)及应用场景,助力企业高效构建安全、可解释的智能系统。
|
2月前
|
JavaScript Java 大数据
基于JavaWeb的销售管理系统设计系统
本系统基于Java、MySQL、Spring Boot与Vue.js技术,构建高效、可扩展的销售管理平台,实现客户、订单、数据可视化等全流程自动化管理,提升企业运营效率与决策能力。
|
2月前
|
人工智能 数据挖掘 数据库
通义灵码产品演示: 数据库设计与数据分析
本演示展示如何使用通义灵码进行数据库设计与数据分析。通过SQLite构建电商订单表,利用AI生成表结构、插入样本数据,并完成多维度数据分析及可视化图表展示,体现AI在数据库操作中的高效能力。
296 8
|
2月前
|
存储 JSON 数据建模
鸿蒙 HarmonyOS NEXT端云一体化开发-云数据库篇
云数据库采用存储区、对象类型、对象三级结构,支持灵活的数据建模与权限管理,可通过AGC平台或本地项目初始化,实现数据的增删改查及端侧高效调用。
160 1
|
2月前
|
存储 弹性计算 安全
现有数据库系统中应用加密技术的不同之处
本文介绍了数据库加密技术的种类及其在不同应用场景下的安全防护能力,包括云盘加密、透明数据加密(TDE)和选择列加密。分析了数据库面临的安全威胁,如管理员攻击、网络监听、绕过数据库访问等,并通过能力矩阵对比了各类加密技术的安全防护范围、加密粒度、业务影响及性能损耗。帮助用户根据安全需求、业务改造成本和性能要求,选择合适的加密方案,保障数据存储与传输安全。
|
2月前
|
存储 弹性计算 Cloud Native
云原生数据库的演进与应用实践
随着企业业务扩展,传统数据库难以应对高并发与弹性需求。云原生数据库应运而生,具备计算存储分离、弹性伸缩、高可用等核心特性,广泛应用于电商、金融、物联网等场景。阿里云PolarDB、Lindorm等产品已形成完善生态,助力企业高效处理数据。未来,AI驱动、Serverless与多云兼容将推动其进一步发展。
182 8
|
3月前
|
存储 人工智能 数据库
视图是什么?为什么要用视图呢?数据库视图:定义、特点与应用
本文三桥君深入探讨数据库视图的概念与应用,从定义特点到实际价值全面解析。视图作为虚拟表具备动态更新、简化查询、数据安全等优势,能实现多角度数据展示并保持数据库重构的灵活性。产品专家三桥君还分析了视图与基表关系、创建维护要点及性能影响,强调视图是提升数据库管理效率的重要工具。三桥君通过系统讲解,帮助读者掌握这一常被忽视却功能强大的数据库特性。
914 0
|
4月前
|
存储 关系型数据库 数据库
附部署代码|云数据库RDS 全托管 Supabase服务:小白轻松搞定开发AI应用
本文通过一个 Agentic RAG 应用的完整构建流程,展示了如何借助 RDS Supabase 快速搭建具备知识处理与智能决策能力的 AI 应用,展示从数据准备到应用部署的全流程,相较于传统开发模式效率大幅提升。
附部署代码|云数据库RDS 全托管 Supabase服务:小白轻松搞定开发AI应用

热门文章

最新文章

下一篇
oss云网关配置