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

news2024/9/9 1:14:15

实战项目:应用数据库和GUI开发产品销售管理软件

01、项目分析

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

▍图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());
  }

 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的窗口显示,用来处理增添记录的操作。

▍图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、源代码下载

JDBCExamples.zip - 坚果云 - 云盘|网盘|企业网盘|同步|备份|无限空间|免费网络硬盘|企业云盘

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

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

相关文章

【目标检测】FPN网络全解

目录:FPN网络详解一、引言二、论文概述2.1 图像金字塔2.2 为什么需要构造特征金字塔三、论文详解四、FPN框架解析五、为什么FPN能够很好的处理小目标?六、FPN总结一、引言 这篇论文是CVPR2017年的文章,采用特征金字塔做目标检测,…

让程序“说话”pyttsx3模块

【小白从小学Python、C、Java】 【计算机等级考试500强双证书】 【Python-数据分析】 让程序“说话” pyttsx3模块 选择题 关于下列代码说法错误的是? import pyttsx3 enginepyttsx3.init() engine.setProperty(rate,180) engine.say(我喜欢学习python数据分析&#x…

统信软件根社区斩获CSDN两项大奖

在CSDN 2022中国开发者影响力盛典中,统信软件根社区凭借多年的深耕与投入以及社区影响力,荣获CSDN 2022年度开发者社区;旗下社区发行版deepin荣获CSDN 2022年度开源影响力项目。 CSDN颁奖现场 “CSDN 2022 中国开发者影响力年度评选”自 20…

温故知新:从计算机体系结构看操作系统

知识诅咒(Curse of Knowledge),是指我们掌握了某知识,就很难体会没有它的感觉,不理解获得知识以前的状态及还未获得该知识的人,从而产生沟通障碍。计算机体系结构正在消亡 (Architecture is dying)&#xf…

一篇用 AI 生成的跨年感怀

一言难尽的2022终于过去了虽然跨年,只是我们的星球围绕我们的恒星环行一周的时间标记,并不真的代表什么。但我们就是希望多事而沉重的一年能快点滚蛋,新的生活能开门重启。我用 Midjourney 生成了很多跨年场景的图。prompt 很简单&#xff0c…

2003-2021年飞机航线数据

1、时间为:2003-2021年 2、指标包括: 起点城市、起点城市所属地级市、起点城市所属省份、起点机场、终点城市、终点城市所属地级市、终点城市所属省份、终点机场、 航空公司、航班、机型、出发时间、到达时间、准点率、班次_周一、班次_周二、班次_周…

RabbitMQ路由模式

🍁博客主页:👉不会压弯的小飞侠 ✨欢迎关注:👉点赞👍收藏⭐留言✒ ✨系列专栏:👉Linux专栏 🔥欢迎大佬指正,一起学习!一起加油! 目录&…

每个程序员都需要掌握的 7 项基本技能

每个 Coder 需要掌握的 7 项基本技能是:1. 编程语言:程序员可以学习多种编程语言,最重要的具体语言取决于他们想从事的工作类型。一些最常见的编程语言包括 Java、C、Python 和 JavaScript。对于编码人员来说,精通至少一种编程语言…

51. N 皇后

51. N 皇后 按照国际象棋的规则,皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。 n 皇后问题 研究的是如何将 n 个皇后放置在 nn 的棋盘上,并且使皇后彼此之间不能相互攻击。 给你一个整数 n ,返回所有不同的 n 皇后问题 的解决方…

剑指offer----C语言版----第七天

目录 1. 旋转数组中的最小数字 1.1 题目描述 1.2 思路一 1.3 思路二 1.4 小试牛刀 1. 旋转数组中的最小数字 原题链接: 剑指 Offer 11. 旋转数组的最小数字 - 力扣(LeetCode)https://leetcode.cn/problems/xuan-zhuan-shu-zu-de-zui-xiao-…

GCC学习笔记

(1)什么是GCC (2)用gcc编译文件 gcc test.c -o app 【编译test.c文件得到app可执行文件】 ./app 【运行可执行文件】 (3)编程语言的发展 (4)gcc工作流程 预处理 作用:…

剑指 Offer 26. 树的子结构

题目 输入两棵二叉树A和B,判断B是不是A的子结构。(约定空树不是任意一个树的子结构) B是A的子结构, 即 A中有出现和B相同的结构和节点值。 例如: 给定的树 A: 给定的树 B: 返回 true,因为 B 与 A 的一个子树拥有相同的结构和节…

JavaScript中如何将十进制转换为十六进制?

​ 在本文中,我们将学习如何在 JavaScript 中轻松地将十进制数转换为其等效的十六进制数。 我们将研究一些需要执行此操作的真实场景。 数字toString() 方法 要在 JavaScript 中将十进制转换为十六进制,请对十进制调用 toString() 方法,将 1…

JUC并发编程学习笔记(四)callable接口与辅助类

6 Callable&Future 接口 6.1 Callable 接口 创建线程的多种方式: 继承Thread类实现Runnable接口Callable接口线程池 ​ 目前我们学习了有两种创建线程的方法一种是通过创建 Thread 类,另一种是通过使用 Runnable 创建线程。但是,Runn…

某电视 频道headers参数x-itouchtv-ca-signature逆向

本文仅供参考学习,如有侵权可联系本人 某电视(荔枝平台)平台逆向 目标链接 aHR0cHM6Ly93d3cuZ2R0di5jbi9jaGFubmVsRGV0YWlsLzI0Ng接口链接: aHR0cHM6Ly9nZHR2LWFwaS5nZHR2LmNuL2FwaS9jaGFubmVsL3YxL25ld3M/cGFnZVNpemU9NDAmY2hhbm5lbElkPTI0NiZjdXJy…

纯注解驱动SSM整合

纯注解驱动SSM整合 Jdbc.java package com.itheima.confing;import com.alibaba.druid.pool.DruidDataSource; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean;import javax.sql.DataSource;public class J…

思购趣拼秒杀模式的逻辑与模式特点

读:小镇上来了一个陌生的年轻人,年轻人交了一百块定金给旅馆的老板去楼上挑房间,旅馆老板拿着这一百块钱还给了卖猪肉的,卖猪肉的把钱还给了卖饲料的,卖饲料的把钱还了厂商,厂商拿着钱还给了按摩店的小姐&a…

一文说清MySQL索引数据结构

前言 接上篇说到,小A匆匆忙忙的赶回宿舍,因为晚上他要给女神整理讲解MySQL中索引数据结构资料。一边整理一边忍住不笑了起来,等小美看到这篇文章不得爱上自己。当上小美男朋友,从此踏上人生巅峰不是梦(该考虑一下孩子…

C#,图像二值化(12)——基于谷底最小值的全局阈值算法与源代码

1、基于谷底最小值的阈值 这个方法适用于通过有限的迭代次数,平滑后能得到双峰的图像,让双峰的谷底成为阈值。当执行完基于谷底最小值的阈值操作的时候,改变了直方图信息,使之成为处理过后的直方图信息,这时候显示Doc…

LeetCodeday05

面试题 02.07. 链表相交 给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表没有交点,返回 null 。 图示两个链表在节点 c1 开始相交: public class Leetcode0207 {public static void main(…