java JTable排序和过滤(JDK 6.0新功能)

JTable排序

  在Java SE 6中除了java.awt被更新外,javax.swing同时也有了很大的改进。在C/S程序中我们会经常使 用到"表"。如我们可以在查询数据库后将查询结果显示在表格中。在Java中显示表格使用的是JTable类。在以前的版本中,JTable只能简单地显 示数据,并没有什么附加的处理功能,而在Java SE 6中的JTable增加了排序和过滤功能。用户可以单击列头进行排序,也可以根据某一列来过滤表 中的数据。

为了使JTable可以对数据进行,必须将RowSorter类和JTable进行关联。RowSorter是一个抽象类,它负责将JTable中的 数据映射成可排序的数据。在真正使用时,我们将直接使用RowSorter的子类TableRowSorter。下面的代码显示了如何将TableRowSorter类和JTable相关联。

TableModel model = new DefaultTableModel(rows, columns);
JTable table = new JTable(model);
RowSorter sorter = new TableRowSorter(model);
table.setRowSorter(sorter);

上面代码首先建立一个TableModel,然后将这个TableModel的实例同时传递给了JTable和RowSorter。下面是一个使用JTable排序的简单的例子。

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;

public class TestSortedTable {
	public static void main(String args[]) {
		JFrame frame = new JFrame("JTable的排序测试");
		frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
		// 表格中显示的数据
		Object rows[][] = { { "王明", "中国", 44 }, { "姚明", "中国", 25 },
				{ "赵子龙", "西蜀", 1234 }, { "曹操", "北魏", 2112 },
				{ "Bill Gates", "美国", 45 }, { "Mike", "英国", 33 } };
		String columns[] = { "姓名", "国籍", "年龄" };
		TableModel model = new DefaultTableModel(rows, columns);
		JTable table = new JTable(model);
		RowSorter sorter = new TableRowSorter(model);
		table.setRowSorter(sorter);
		JScrollPane pane = new JScrollPane(table);
		frame.add(pane, BorderLayout.CENTER);
		frame.setSize(300, 150);
		frame.setVisible(true);
	}
}

图3显示的是按"年龄"进行降序排列。但我们发现一个奇怪的问题,就是"年龄"字段并不是按数值类型进行排序的,而是按字符类型进行排序的。



出现这种情况是因为在默认情况下DefaultTableModal的列是Object类型。而要想使JTable按数值进行排序,必须要覆盖DefaultTableModal的getColumnClass方法。

		TableModel model = new DefaultTableModel(rows, columns) {
			public Class getColumnClass(int column) {
				Class returnValue;
				if ((column >= 0) && (column < getColumnCount())) {
					returnValue = getValueAt(0, column).getClass();
				} else {
					returnValue = Object.class;
				}
				return returnValue;
			}
		};

图4显示了按"年龄"进行排序的界面,看看,是不是按数值进行排序了。

JTable过滤

在JTable中通过抽象类RowFilter类对行进行过滤。和排序不同,你可以不建立它们的子类,而使用这个抽象类的6个静态方法。

·andFilter
·dateFilter(RowFilter.ComparisonType type, Date date, int... indices)
·notFilter(RowFilter filter)
·numberFilter(RowFilter.ComparisonType type, Number number, int... indices)
·orFilter
·regexFilter(String regex, int... indices)

其中andFilter()、orFilter()以及notFilter()方法的功能是将当前的过滤条件和其它的过滤条件进行组合。如在同时比较日期和数值时需要将日期过滤和数值过滤进行组合。这些组合是非常简单的。

RowFilter的类型比较允许你进行4种关系的比较,等于、不等于、大于或小于。我们可以通过指定某一列进行过滤,也可以对所有的列进行过滤。这 其中最为有趣的也许是正则表达式过滤(regular expression filter,或简称为regex filter)。使用这个过滤器可以对 表中数据进行更高级的过滤。下面是实现一个简单过滤器的代码。

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;
import java.awt.event.*;

public class TestFilter {
	public static void main(String args[]) {
		JFrame frame = new JFrame("JTable的过滤测试");
		frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
		Object rows[][] = { { "王明", "中国", 44 }, { "姚明", "中国", 25 },
				{ "赵子龙", "西蜀", 1234 }, { "曹操", "北魏", 2112 },
				{ "Bill Gates", "美国", 45 }, { "Mike", "英国", 33 } };
		String columns[] = { "姓名", "国籍", "年龄" };
		TableModel model = new DefaultTableModel(rows, columns) {
			public Class getColumnClass(int column) {
				Class returnValue;
				if ((column >= 0) && (column < getColumnCount())) {
					returnValue = getValueAt(0, column).getClass();
				} else {
					returnValue = Object.class;
				}
				return returnValue;
			}
		};
		final JTable table = new JTable(model);
		final TableRowSorter sorter = new TableRowSorter(
				model);
		table.setRowSorter(sorter);
		JScrollPane pane = new JScrollPane(table);
		frame.add(pane, BorderLayout.CENTER);
		JPanel panel = new JPanel(new BorderLayout());
		JLabel label = new JLabel("过滤");
		panel.add(label, BorderLayout.WEST);
		final JTextField filterText = new JTextField("");
		panel.add(filterText, BorderLayout.CENTER);
		frame.add(panel, BorderLayout.NORTH);
		JButton button = new JButton("过滤");
		button.addActionListener(new ActionListener() {
			public void actionPerformed(ActionEvent e) {
				String text = filterText.getText();
				if (text.length() == 0) {
					sorter.setRowFilter(null);
				} else {
					sorter.setRowFilter(RowFilter.regexFilter(text));
				}
			}
		});
		frame.add(button, BorderLayout.SOUTH);
		frame.setSize(300, 250);
		frame.setVisible(true);
	}
}

图5是上面程序的运行界面。


以下是销售系统中表格中实现过滤功能,过滤第5列即销售列(字段包括“已售”、“预订"、”未售“),

if(displaySaleStatus==0&&displayBookStatus==0&&displayNotBookStatus==0)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("什么都不显示哦", 4));
if(displaySaleStatus==1&&displayBookStatus==0&&displayNotBookStatus==0)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售", 4));
else if(displaySaleStatus==0&&displayBookStatus==1&&displayNotBookStatus==0)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("预订", 4));
else if(displaySaleStatus==0&&displayBookStatus==0&&displayNotBookStatus==1)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("未订", 4));
else if(displaySaleStatus==1&&displayBookStatus==1&&displayNotBookStatus==0)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售|预订", 4));
else if(displaySaleStatus==0&&displayBookStatus==1&&displayNotBookStatus==1)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("预订|未订", 4));
else if(displaySaleStatus==1&&displayBookStatus==0&&displayNotBookStatus==1)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售|未订", 4));
else if(displaySaleStatus==1&&displayBookStatus==1&&displayNotBookStatus==1)
	((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售|预订|未订", 4));//全部显示,或setRowFilter(null)
一下是地震发布系统中过滤第2列即震级列(字段为Float型)

	/**
	 * (大、中、小地震)显示按钮事件处理过程
	 * */
	@SuppressWarnings("unchecked")
	private void showActionHandle(){
		boolean showBigEqStatus=showBigEqBut.isSelected();
		boolean showMiddleEqStatus=showMiddleEqBut.isSelected();
		boolean showSmallEqStatus=showSmallEqBut.isSelected();
		if(showBigEqStatus==false&&showMiddleEqStatus==false&&showSmallEqStatus==false)
			((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER, 100,2));
		if(showBigEqStatus==true&&showMiddleEqStatus==false&&showSmallEqStatus==false){
			((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
		}
		else if(showBigEqStatus==false&&showMiddleEqStatus==true&&showSmallEqStatus==false){
			List> filters = new ArrayList>(2);
			filters.add(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
			filters.add(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
			RowFilter fooBarFilter = RowFilter.andFilter(filters);
			((TableRowSorter)this.getRowSorter()).setRowFilter(fooBarFilter);
		}
		else if(showBigEqStatus==false&&showMiddleEqStatus==false&&showSmallEqStatus==true){
			((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
		}
		else if(showBigEqStatus==true&&showMiddleEqStatus==true&&showSmallEqStatus==false){
			((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
		}
		else if(showBigEqStatus==false&&showMiddleEqStatus==true&&showSmallEqStatus==true){
			((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
		}
		else if(showBigEqStatus==true&&showMiddleEqStatus==false&&showSmallEqStatus==true){
			List> filters = new ArrayList>(2);
			filters.add(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
			filters.add(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
			RowFilter fooBarFilter = RowFilter.orFilter(filters);
			((TableRowSorter)this.getRowSorter()).setRowFilter(fooBarFilter);
		}
		else if(showBigEqStatus==true&&showMiddleEqStatus==true&&showSmallEqStatus==true){
			((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER, 0,2));
		}
	}



补充:将表格视图中的行号映射到模型中对应的行号,才能正确地得到行的内容,尤其是自动排序后,视图中的行号与表格模型中的行号不一致,TableModel中的方法getValueAt(row,column)中的row指模型中的row,而通过table.getSelectedRow()或者事件处理中通过table.rowAtPoint(e.getPoint())得到row指视图中的row,因此必须将视图中的row转化为model中的row:

int  view_row    = table.rowAtPoint(e.getPoint()); //获得视图中的行索引
 int  view_col     = table.columnAtPoint(e.getPoint()); //获得视图中的列索引
 int  model_row= table.convertRowIndexToModel(row);//将视图中的行索引转化为数据模型中的行索引
 int  model_col = table.convertColumnIndexToModel(col);//将视图中的列索引转化为数据模型中的列索引

 

你可能感兴趣的:(java)