【问题标题】:Dynamic GUI Creation from ResultSet - Java从 ResultSet 创建动态 GUI - Java
【发布时间】:2014-02-27 10:56:24
【问题描述】:

我正在尝试通过从结果集中获取值并使用它来生成清单来动态创建 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 一切都很好。但是,我需要能够将 Combobox 和 Textfield 组件视为它们自己独立的实体。意思是,我希望能够从每个不同的组件中获取用户输入。现在,如果我要从文本字段请求信息,它只会给我来自最后一个文本字段的信息。这很完美,因为这就是 java 读取它的方式。我没有问题。

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

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

非常感谢任何帮助!

【问题讨论】:

  • 为什么不将它们保存在数组或列表中?我还建议修复警告而不是抑制它们。
  • 答案取决于。你在乎记录id吗?
  • @MadProgrammer,是的,ID 是相关的,因为我可能要求项目按特定顺序排列。
  • @MikeB 关于将它们保存在一个数组中......我可以让它很好地转储(创建 GUI),但我不知道如何获取每个单独 GUI 的输入元素。

标签: java mysql swing resultset


【解决方案1】:

您只有对您创建的最后一个文本字段或组合框的引用,因为您正在重用保存它们的变量。我会将它们放在一个 ArrayList 中,在您创建它们时存储每个新的文本字段和组合框,然后您可以在完成后返回并从它们中获取所有输入。

---------(在OP对上述段落的回应之后)

不,没有“推荐你的地方”——这是你的一套要求,找到已经存在的代码来做这件事是非常了不起的。 Java 和 Swing 为您提供了工具,您需要自己组装。

你没有展示你的“actionPerformed”例程,但让我们假设一下。它在动作完成时由框架调用,并传递一个“ActionEvent”对象。通过它的方法,我们发现它有“getSource()”,所以它会给你一个对生成事件的组件的引用。

让我们进一步思考一下我们所拥有的——UI 中的一组组件,以及我们感兴趣的可以生成事件的组件。在这种情况下,我们希望从生成事件的组件中检索一些内容。

如果我们有组件(来自 actionEvent.getSource())并且我们想用它做点什么,那么我们可以在最坏的情况下在 actionPerformed() 方法中做如下的事情:

Component sourceComponent = actionEvent.getSource();
if (sourceComponent instanceof JComboBox)
  { JComboBox sourceBox = (JComboBox) sourceComponent;
    // get the value from the combo box here
  }
else if (sourceComponent instanceof JTextField)
  { JTextField sourceTextField = (JTextField) sourceComponent;
    // get the value from the text field here
  }
// or else do nothing -- our action was not one of these.

这样做,您甚至不需要保留组件列表 - UI 会保留对所有组件的引用,并且您只需在 actionEvent 发生时使用该引用。

现在,这不是唯一的,甚至不是最好的或最简单的方法。如果你想用你自己的类扩展 JComboBox 和 JTextField,你可以让这些类都实现一个接口,该接口定义了类似 getValue()getText 的东西;那么你就不需要丑陋的instance of 运算符了,这通常可以通过更好的设计和规划来消除。

【讨论】:

  • 也许我不确定如何以正确的方式做到这一点。正如我所说:这是显而易见的解决方案,但每当我尝试它时,它只是......对我不起作用。你能指出我可以参考的地方吗?如果可以,我会接受这个答案。
【解决方案2】:

询问包含组件的容器(窗格)怎么样 getComponents() 方法并循环遍历子组件并检查 JComobox 和 JTextField 执行所需的转换并检索值

只是一个想法,以防您反对将子组件添加到某种列表中

【讨论】:

    【解决方案3】:

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

    如果您只执行批量更新,您可以使用Map 键控到行的id 并映射到Component

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

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

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

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

    现在...坦率地说,我会简单地使用 JTable 并创建一个自定义的 TableModel 来模拟所有这些。

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

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

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

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

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

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

    【讨论】:

    • 感谢您抽出宝贵时间回答这个详细的问题!它有助于看到不同的选择。但是,也许我未能解释我的程序到底需要什么。这很简单。假设您有一个清单,该清单中有两个项目。他们每个人都需要一个通过/失败的下拉框。举个例子: 1. 外面在下雨。 (它作为 Combobox 放入数据库中) - java 代码获取它的标题(正在下雨......)并创建 JLabel。然后创建组合框。 2.外面很热。和上面一样。但是,我需要检查这两个组合。
    • 然后,使用Mapid 映射到component。遍历Map,得到idcomponent 检查所有组件
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-05-07
    • 2021-05-02
    • 2018-12-02
    • 1970-01-01
    • 2016-12-07
    相关资源
    最近更新 更多