使用JTable会导致应用程序的其余部分无法正确加载

时间:2019-03-18 10:19:11

标签: java swing jtable jtabbedpane

首先,这是一个学校项目,因此我无法上传完整的代码。我们得到了一个空白项目,并要求为其添加功能。

我希望能够在多个表中显示来自服务器的数据,每个表都在单独的选项卡中,并且每个“页面”都可以选择向表中添加,编辑或删除记录。

由于我有很多不同的表,所以我写了一个父类Page,它将包含一个表和一些JPanel,可以在需要时向其中添加按钮。

但是,每当我尝试向页面添加JTable时,都会在选项卡之间导致无法预测的行为-请注意,无论我是否使用TableModel,都会发生这种情况。

我尝试使用标签显示组件的位置,效果很好,但是使用实际表会导致占位符标签和顶部的标签被隐藏。

Tab with labels to illustrate where the components SHOULD be

Tab after I try to add a JTable - the rest of the code (including other placeholder labels) is the same

我没有对线程做任何事情,所以当我尝试添加JTable时,应用程序发生了什么事。

这是Page.java的代码:

import java.awt.*;

import javax.swing.*;
import javax.swing.event.TableModelListener;
import javax.swing.table.TableModel;

public class Page extends JPanel {
    private static final long serialVersionUID = -4661566573959270000L;
    protected JTable table;
    protected JPanel addPanel;
    protected JPanel fieldPanel;
    protected JPanel buttonPanel;
    protected TableModel model;
    public Page(String[] names) {
        this.setLayout(new GridLayout(2,1));
        final class MyTableModel implements TableModel{
            private static final long serialVersionUID = -4661566573959270000L;
            private String[] columnNames;
            private Object[][] data;
            public MyTableModel(String[] names) {
                columnNames=names;
                data=new Object[names.length][0];
            }
            @Override
            public void setValueAt(Object aValue, int rowIndex, int columnIndex) {
                if(isCellEditable(rowIndex, columnIndex)) {
                    data[rowIndex][columnIndex] = aValue;
                }
             }     

             @Override
             public void removeTableModelListener(TableModelListener l) {

             }

             @Override
             public boolean isCellEditable(int rowIndex, int columnIndex) {
                      return false;
             }

             @Override
             public Object getValueAt(int rowIndex, int columnIndex) {
                 return data[rowIndex][columnIndex];
             }

             @Override
             public int getRowCount() {
                 return data.length;
             }

             @Override
             public String getColumnName(int columnIndex) {
                 return columnNames[columnIndex];
             }

             @Override
             public int getColumnCount() {
                 return columnNames.length;
             }

             @Override
             public Class<?> getColumnClass(int columnIndex) {
                 try {
                     return getValueAt(0, columnIndex).getClass();
                 }
                 catch (Exception e) {
                     return null;
                 }
             }

             @Override
             public void addTableModelListener(TableModelListener l) {

             }
         };


         /**
         * creating and adding a table
         * this is the problematic bit
         */
         table = new JTable(new MyTableModel(names));
         this.add(table);
         //this.add(new Label("Table"));



         //panels for layout
         addPanel = new JPanel();
         fieldPanel = new JPanel();
         buttonPanel = new JPanel();
         //assigning positions
         this.add(addPanel);
         addPanel.setLayout(new BorderLayout());
         addPanel.add(fieldPanel, BorderLayout.CENTER);
         addPanel.add(buttonPanel, BorderLayout.SOUTH);
         //placeholder labels
         fieldPanel.add(new JLabel("Insert fields here"));
         buttonPanel.add(new Button("Buttons"));
     }
 }

我使用以下代码将新页面添加到标签系统:

JTabbedPane tabs = new JTabbedPane();
tabs.add("A", new Page(new String[]{"A"}));
tabs.add("B", new Page(new String[]{"B"}));
tabs.add("C", new Page(new String[]{"C"}));
tabs.add("D", new Page(new String[]{"D"}));

使用String []作为我要显示的列的名称-请注意,一旦表正常工作,这些名称将被替换为实际有用的名称。

有什么主意我做错了吗?我尝试使用ChangeListener事件重画框架,但这似乎无济于事。

谢谢!

EDIT 其中包括完整的代码版本。将上面的代码用于Page,并在Main类中使用它:

import java.awt.*;
import javax.swing.*;
public class Main {
    public static void main(String[] args) {
        JFrame window = new JFrame("Components and Containers");
        window.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE) ;
        window.setSize(800,600);

        Container pane = window.getContentPane();
        JTabbedPane tabs = new JTabbedPane();
        tabs.add("A", new Page(new String[]{"A"}));
        tabs.add("B", new Page(new String[]{"B"}));
        tabs.add("C", new Page(new String[]{"C"}));
        tabs.add("D", new Page(new String[]{"D"}));
        pane.add(tabs);
        window.setVisible(true);
    }
}

我尝试将表添加到ScrollPanel中,并且只有将鼠标悬停在该表上时该表才可见。选项卡仍无法正确显示。

The result of adding the table to a separate ScrollPanel instead of directly to the component.

1 个答案:

答案 0 :(得分:2)

您的表模型类与行/列顺序不一致。

在构造函数中,您编写

data=new Object[names.length][0];

这意味着第一个索引是列号,第二个索引是行号。

但是,在访问数据(在getValueAtsetValueAt中)时,您将以另一种方式使用indizes:

data[rowIndex][columnIndex]

而且,这个:

public int getRowCount() {
    return data.length;
}

将列数返回为行数,这意味着JTable尝试调用ArrayIndexOutOfBoundsException时,您将获得getValueAt(0, 0)


要解决此问题,您应该使用以下方法初始化构造函数中的数据数组:

data=new Object[0][names.length];