java - 从结果集动态创建 GUI - Java

标签 java mysql swing resultset

我正在尝试通过从结果集中获取值并使用它生成 list 来动态创建 Java GUI。我创建了一个小型演示程序来演示我所做的事情:

SQL命令

CREATE USER 'test'@'localhost' IDENTIFIED BY 'testpw';
CREATE DATABASE combotest;
USE combotest;

CREATE TABLE combotable (
id INT(5) NOT NULL PRIMARY KEY auto_increment,
type VARCHAR(50) NOT NULL);

INSERT INTO combotable (id, type) VALUES
(default, 'Label'),
(default, 'Textfield'),
(default, 'Combo'),
(default, 'Label'),
(default, 'Textfield'),
(default, 'Combo'),
(default, 'Combo');

GRANT SELECT ON combotest.* TO 'test'@'localhost';

为方便起见,如果您想自己测试一下,我已将所有 SQL 命令放在上面。

现在,对于我的 Java 代码:

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.sql.*;  
import javax.swing.*;

public class resToComboDemo implements ActionListener {

//JDBC Variables
static Connection connect = null;
static Statement statement = null;
static ResultSet res = null;

@SuppressWarnings("rawtypes")
//Other Variables
JComboBox comboBox;
JButton submit;
JFrame frame;
JLabel label;
JTextField textField;
Container pane;

public static void main(String[] args) throws SQLException {
    new resToComboDemo();
}

public resToComboDemo() throws SQLException {
    try {
        Class.forName("com.mysql.jdbc.Driver");
        // Setup the connection with the DB

        connect = DriverManager
                .getConnection("jdbc:mysql://localhost/combotest?"
                        + "user=test&password=testpw");

        statement = connect.createStatement();
        //Note: in this specific case I do realize that "order by id" is not necessary. I want it there, though.
        res = statement.executeQuery("SELECT * FROM combotable ORDER BY id");

        createStuff(res);

    } catch (Exception e) {
        JOptionPane.showMessageDialog(null, "Error 1: "+e, "Error!", JOptionPane.ERROR_MESSAGE);
    } finally {
        connect.close();
    }

}

@SuppressWarnings({"rawtypes", "unchecked" })
public void createStuff (ResultSet res) throws SQLException {

    frame = new JFrame("Testing dynamic gui");
    Dimension sD = Toolkit.getDefaultToolkit().getScreenSize();
    int width = sD.width;
    int height = sD.height - 45;
    frame.setSize(width,height);

    pane = frame.getContentPane();

    pane.setLayout(new GridLayout(0, 2));

    while (res.next()) {
        Object[] options = { "Pass", "Fail"};
        String type = res.getString("type");

        JLabel label = new JLabel("<html><small>"+type+"</small></html>");
        JLabel blank = new JLabel(" ");
        blank.setBackground(Color.black);
        blank.setOpaque(true);

        if (type.equals("Label")) {
            label.setBackground(Color.black);
            label.setForeground(Color.white);
            label.setOpaque(true);
            pane.add(label);
            pane.add(blank);

        } else if (type.equals("Combo")) {
            pane.add(label);
            comboBox = new JComboBox(options);
            pane.add(comboBox);

        } else if (type.equals("Textfield")) {
            pane.add(label);
            textField = new JTextField(20);
            pane.add(textField);

        }   
    }

     JLabel blank2 = new JLabel(" ");
     pane.add(blank2);

     submit = new JButton("Submit");
     submit.addActionListener(this);
     pane.add(submit);

     frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
     frame.setVisible(true);

}

@Override
public void actionPerformed(ActionEvent e) {

}
}

现在,在这里创建 GUI 一切都很顺利。但是,我需要能够将组合框和文本字段组件视为它们自己的单独实体。意思是,我希望能够从每个不同的组件获取用户输入。现在,如果我要从文本字段请求信息,它只会向我提供最后一个文本字段的信息。这样就很完美了,因为 java 就是这样读取它的。我对此没有任何问题。

我一生都无法弄清楚如何分别获取每个组件的输入。也许通过获取结果集并将结果添加到某种类型的数组中?我已经尝试过多次不同的口味,但我无法让它按照我需要的方式出现。你们中的一些人会要求我向你们展示我所尝试过的内容...但说实话,这不值得。

并且,在有人问之前:不,我不会使用 FlowLayout。 :)

非常感谢任何帮助!

最佳答案

根据您想要做的事情,可能有几种方法可以实现这一目标...

如果您仅执行批量更新,则可以使用键控到行的 id 并映射到 ComponentMap .

这样,当您想要将值保存回数据库时,您只需迭代 Map 的键值,提取与每个键关联的 Component 并然后提取Component的值...

我可能会考虑制作一个包装器接口(interface),它有一个简单的getText方法并将组件包装在其中,使包装器的实现负责提取文本,但是这就是我;)

如果您想在更新单个组件时执行更新,则需要交换映射,以便 Component 为键,id 为映射到它。

这意味着当发生某种会触发和更新的事件(即 ActionEvent)时,您可以从事件中提取并查找Map 中的 id 基于引发事件的 Component...

现在...坦率地说,我只需使用 JTable 并创建一个可以对所有这些进行建模的自定义 TableModel

这需要您创建表的 POJO,并在单个对象中维护 idtypevalue。这将定义表中的基本行。

唯一的问题是您需要创建一个(相当)复杂的TableCellEditor,它可以采用类型并返回适合表格的编辑器。并非不可能,这只是表的正常使用之外的额外复杂性。

这将使您需要的所有信息都可以在表中单行的单个对象中获得。

看看How to use tables了解更多详情

同样,您可以使用与上面的 Map 想法类似的想法...

您还可以简单地创建一个自包含的“编辑器”(从 JPanel 之类的东西扩展),它维护有关 idtype 的信息> 并且您可以从中提取值并简单地保留这些值的列表......例如......

关于java - 从结果集动态创建 GUI - Java,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21536113/

相关文章:

Java 用鼠标调整形状大小

java - 使用 java 的 KeyPairGenerator 的 openssl 等效项是什么?

java - 在 GWT 中比较两个 JavaScriptObject 的最佳方法是什么?

mysqldump : Couldnt execute SET OPTION SQL_QUOTE_SHOW_CREATE=1 mysql 5. 6

php - 避免电子商务网站中的产品重复

php - SQL 棘手的顺序

java - 更改 JTextPane 中显示的 HTMLDocument 的内容

Java - 如何为文本字段创建循环?

Java bean 机器

java - 如何捕获窗口状态?