如何在JTable中对JComboBox的字符串进行排序?

How To Sort Strings of JComboBoxes in JTables?

我有以下问题:

我已经在我的 JTable 的第 5 列中为每一行插入了一个 JComboBox 对象。 一切都很好,直到我想使用 setAutoCreateRowSorter(true) 对列进行行排序。在这种情况下,我收到以下异常:

ClassCastException: java.lang.String cannot be cast to javax.swing.JComboBox

这是我用于我的 JTable 的 类:

表格模型:

private class MyTableModel implements TableModel {

        @Override
        public void addTableModelListener(TableModelListener l) {
        }

        @Override
        public Class<?> getColumnClass(int columnIndex) {

            switch (columnIndex) {
            case 0:
                return String.class;
            case 1:
                return String.class;
            case 2:
                return String.class;
            case 3:
                return Number.class;
            case 4:
                return Boolean.class;
            case 5:
                return JComboBox.class;  // modifyed!

            default:
                return null;
            }

        }

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

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

        @Override
        public int getRowCount() {
            return data[0].length - 1;
        }

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

        }

        @Override
        public boolean isCellEditable(int rowIndex, int columnIndex) {
            switch (columnIndex) {
            case 5:
                return true;
            default:
                return false;
            }

        }

        @Override
        public void removeTableModelListener(TableModelListener l) {
        }

        @Override
        public void setValueAt(Object aValue, int rowIndex, int columnIndex) {
            data[rowIndex][columnIndex] = aValue;

        }

    }

TableCellRenderer:

public class StringTableCellRenderer extends JLabel implements
        TableCellRenderer {

    private static final long serialVersionUID = 1L;

    public StringTableCellRenderer() {
        setOpaque(true);
        setLayout(new BorderLayout());
    }

    @Override
    public Component getTableCellRendererComponent(JTable table, Object value,
            boolean isSelected, boolean hasFocus, int row, int column) {
        Font font = getFont().deriveFont(Font.BOLD, 15);

        if ((row % 2) == 0) {
            setBackground(new Color(240, 255, 255));
        } else {
            setBackground(new Color(191, 239, 255));
        }

        if (isSelected) {
            setBackground(new Color(0, 191, 255));
        }

        setHorizontalAlignment(JLabel.CENTER);
        setForeground(Color.BLACK);
        setFont(font);

        if (value instanceof JComboBox) {
            System.out.println("Renderer: "+column+" "+row+" "+((JComboBox<?>)value).getSelectedIndex());

            setText((String) ((JComboBox<?>) value).getSelectedItem());
        } else {
            setText(value.toString());
        }

        return this;
    }

}

TableCellEditor:

  public class MyTableCellEditor extends AbstractCellEditor implements
        TableCellEditor, ActionListener {

    private static final long serialVersionUID = 1L;
    private JComboBox<?>[] comboList = new JComboBox<?>[5];
    private int column = 0;

    @SuppressWarnings("unchecked")
    public MyTableCellEditor() {

        for (int i = 0; i < comboList.length; i++) {
            comboList[i] = new JComboBox<String>();
            setComboBox((JComboBox<String>) comboList[i]);
            comboList[i].setName("ComboBox_" + i);
        }

    }

    public void setComboBox(JComboBox<String> comboBox) {
        comboBox.addItem("Me");
        comboBox.addItem("You");
        comboBox.addItem("They");
        comboBox.addItem("Us");
        comboBox.addItem("We");
    }

    @Override
    public void actionPerformed(ActionEvent event) {
    }

    @Override
    public Object getCellEditorValue() {
        System.out.println("Editor: getCellEditorValue() "+comboList[column].getSelectedIndex());
        if (comboList[column].getSelectedIndex()==-1) {
            return (String) comboList[column].getItemAt(0);
        } else {
            return (String) comboList[column].getSelectedItem(); // returns String not JCombobox!
        }
    }

    @Override
    public Component getTableCellEditorComponent(JTable table, Object value,
            boolean isSelected, int row, int column) {

        String fieldValue = null;
        System.out.println("Editor: getTableCellEditorComponent");
        if (value instanceof JComboBox<?>) {
            fieldValue = (String) ((JComboBox<?>) value).getSelectedItem();
        }
        this.column = column - 1;
        comboList[column - 1].setSelectedItem(fieldValue);

        return comboList[column - 1]; // for each row of column 5 an own JComboBox object

    }

}

我自己解决了这个问题:

在 TableModel 中,第 5 列的 getColumnClass()-方法 returns JCombobox.class-object.

在 TableCellEditor 中,第 5 列的每一行都有一个 自己的 JComboBox-ObjectgetCellEditorValue()-method return s 有关 JComboBox-Object.

getSelectedItem()

所以 Jtable 确实有一个 JComboBox-对象(getValueAt()-TableModel 第 5 列的方法和 TableCellEditor 的 getTableCellEditorComponent()-方法)但是 return 值尽管如此,它仍然是一个字符串(来自 TableCellEditor 的 getCellEditorValue() 和 getTableCellRendererComponent()-来自 TableCellRenderer 的方法)。

希望这对遇到类似问题的人有所帮助...

为什么要创建自定义比较器? TableModel 中的数据是 String 数据,而不是 JCombobox。 table 的 RowSorter 已经知道如何对 String 数据进行排序。

您永远不应该在 TableModel 中实际存储 JComboBox。如果你确实在这样做,那就摆脱它。

然后摆脱您的自定义比较器,它应该可以正常工作。

如果您尝试对不同行中的组合框使用不同的值,那么您应该尝试类似的方法:how to add different cell editors for one column in JTable?