我正在尝试通过从结果集中获取值并使用它来生成清单来动态创建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
并映射到Component
的Map
。
这样,当您要将值保存回数据库时,只需迭代Map
的键值,提取与每个键关联的Component
,然后提取Component
的值...
我可能会考虑制作一个包装器interface
它具有一个简单的getText
方法并将组件包装在其中,使包装器的实现负责提取文本,但这只是我;)
如果要在更新单个组件时执行更新,则需要交换映射,以便Component
将键和id
映射到它。
这意味着当发生某种会触发和更新的事件(即ActionEvent
)时,您可以从事件中提取source
并根据导致事件的Component
查找Map
中的id
......
现在。。。坦率地说,我只需使用JTable
并创建一个可以对所有这些进行建模的自定义TableModel
。
这将需要您创建表的 POJO,在单个对象中维护id
、type
和value
。 这将在表中定义一个基本行。
唯一的问题是您需要创建一个(合理的)复杂TableCellEditor
,该可以采用type
并为表返回适当的编辑器。 并非不可能,这只是超出表正常使用范围的额外复杂性。
这将在表中单行的单个对象中提供您需要的所有信息。
查看如何使用表格了解更多详情
同样,您可以在上面的Map
想法中使用类似的想法......
您也可以简单地创建一个自包含的"编辑器"(从类似JPanel
的东西扩展),它维护有关id
和type
的信息,您可以从中提取值并简单地保留这些列表。例如。。。
询问包含组件的容器(窗格)怎么样 getComponents() 方法并遍历子组件并检查 JComobox 和 JTextField 是否执行所需的强制转换并检索值
只是一个想法,以防您反对将子组件添加到一种列表中
您只能引用您创建的最后一个文本字段或组合框,因为您正在重用保存它们的变量。 我会将它们放在 ArrayList 中,在创建它们时存储每个新文本字段和梳状框,然后您可以在完成后返回并从所有这些字段中获取输入。
----------(在OP对上述段落的答复之后)
不,没有"推荐你的地方" - 这是你的一组需求,找到已经存在的代码来做这件事是非常了不起的。 Java和Swing给你的工具,你需要自己把东西放在一起。
您没有显示您的"动作执行"例程,但让我们假设一下。 当操作完成时,框架会调用它,并传递一个"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
运算符,这通常可以通过更好的设计和规划来消除。