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相关联。

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

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

  1. import javax.swing.*;
  2. import javax.swing.table.*;
  3. import java.awt.*;
  4. public class TestSortedTable {
  5. public static void main(String args[]) {
  6. JFrame frame = new JFrame("JTable的排序测试");
  7. frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  8. // 表格中显示的数据
  9. Object rows[][] = { { "王明", "中国", 44 }, { "姚明", "中国", 25 },
  10. { "赵子龙", "西蜀", 1234 }, { "曹操", "北魏", 2112 },
  11. { "Bill Gates", "美国", 45 }, { "Mike", "英国", 33 } };
  12. String columns[] = { "姓名", "国籍", "年龄" };
  13. TableModel model = new DefaultTableModel(rows, columns);
  14. JTable table = new JTable(model);
  15. RowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
  16. table.setRowSorter(sorter);
  17. JScrollPane pane = new JScrollPane(table);
  18. frame.add(pane, BorderLayout.CENTER);
  19. frame.setSize(300, 150);
  20. frame.setVisible(true);
  21. }
  22. }

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

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

  1. TableModel model = new DefaultTableModel(rows, columns) {
  2. public Class getColumnClass(int column) {
  3. Class returnValue;
  4. if ((column >= 0) && (column < getColumnCount())) {
  5. returnValue = getValueAt(0, column).getClass();
  6. } else {
  7. returnValue = Object.class;
  8. }
  9. return returnValue;
  10. }
  11. };
图4显示了按"年龄"进行排序的界面,看看,是不是按数值进行排序了。

JTable过滤

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

·andFilter
·dateFilter(RowFilter.ComparisonType type, Date date, int... indices)
·notFilter(RowFilter<M,I> 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)。使用这个过滤器可以对 表中数据进行更高级的过滤。下面是实现一个简单过滤器的代码。

  1. import javax.swing.*;
  2. import javax.swing.table.*;
  3. import java.awt.*;
  4. import java.awt.event.*;
  5. public class TestFilter {
  6. public static void main(String args[]) {
  7. JFrame frame = new JFrame("JTable的过滤测试");
  8. frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  9. Object rows[][] = { { "王明", "中国", 44 }, { "姚明", "中国", 25 },
  10. { "赵子龙", "西蜀", 1234 }, { "曹操", "北魏", 2112 },
  11. { "Bill Gates", "美国", 45 }, { "Mike", "英国", 33 } };
  12. String columns[] = { "姓名", "国籍", "年龄" };
  13. TableModel model = new DefaultTableModel(rows, columns) {
  14. public Class getColumnClass(int column) {
  15. Class returnValue;
  16. if ((column >= 0) && (column < getColumnCount())) {
  17. returnValue = getValueAt(0, column).getClass();
  18. } else {
  19. returnValue = Object.class;
  20. }
  21. return returnValue;
  22. }
  23. };
  24. final JTable table = new JTable(model);
  25. final TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(
  26. model);
  27. table.setRowSorter(sorter);
  28. JScrollPane pane = new JScrollPane(table);
  29. frame.add(pane, BorderLayout.CENTER);
  30. JPanel panel = new JPanel(new BorderLayout());
  31. JLabel label = new JLabel("过滤");
  32. panel.add(label, BorderLayout.WEST);
  33. final JTextField filterText = new JTextField("");
  34. panel.add(filterText, BorderLayout.CENTER);
  35. frame.add(panel, BorderLayout.NORTH);
  36. JButton button = new JButton("过滤");
  37. button.addActionListener(new ActionListener() {
  38. public void actionPerformed(ActionEvent e) {
  39. String text = filterText.getText();
  40. if (text.length() == 0) {
  41. sorter.setRowFilter(null);
  42. } else {
  43. sorter.setRowFilter(RowFilter.regexFilter(text));
  44. }
  45. }
  46. });
  47. frame.add(button, BorderLayout.SOUTH);
  48. frame.setSize(300, 250);
  49. frame.setVisible(true);
  50. }
  51. }

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

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

  1. if(displaySaleStatus==0&&displayBookStatus==0&&displayNotBookStatus==0)
  2. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("什么都不显示哦", 4));
  3. if(displaySaleStatus==1&&displayBookStatus==0&&displayNotBookStatus==0)
  4. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售", 4));
  5. else if(displaySaleStatus==0&&displayBookStatus==1&&displayNotBookStatus==0)
  6. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("预订", 4));
  7. else if(displaySaleStatus==0&&displayBookStatus==0&&displayNotBookStatus==1)
  8. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("未订", 4));
  9. else if(displaySaleStatus==1&&displayBookStatus==1&&displayNotBookStatus==0)
  10. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售|预订", 4));
  11. else if(displaySaleStatus==0&&displayBookStatus==1&&displayNotBookStatus==1)
  12. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("预订|未订", 4));
  13. else if(displaySaleStatus==1&&displayBookStatus==0&&displayNotBookStatus==1)
  14. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售|未订", 4));
  15. else if(displaySaleStatus==1&&displayBookStatus==1&&displayNotBookStatus==1)
  16. ((TableRowSorter)this.table.getRowSorter()).setRowFilter(RowFilter.regexFilter("已售|预订|未订", 4));//全部显示,或setRowFilter(null)

一下是地震发布系统中过滤第2列即震级列(字段为Float型)

  1. /**
  2. * (大、中、小地震)显示按钮事件处理过程
  3. * */
  4. @SuppressWarnings("unchecked")
  5. private void showActionHandle(){
  6. boolean showBigEqStatus=showBigEqBut.isSelected();
  7. boolean showMiddleEqStatus=showMiddleEqBut.isSelected();
  8. boolean showSmallEqStatus=showSmallEqBut.isSelected();
  9. if(showBigEqStatus==false&&showMiddleEqStatus==false&&showSmallEqStatus==false)
  10. ((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER, 100,2));
  11. if(showBigEqStatus==true&&showMiddleEqStatus==false&&showSmallEqStatus==false){
  12. ((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
  13. }
  14. else if(showBigEqStatus==false&&showMiddleEqStatus==true&&showSmallEqStatus==false){
  15. List<RowFilter<Object,Object>> filters = new ArrayList<RowFilter<Object,Object>>(2);
  16. filters.add(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
  17. filters.add(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
  18. RowFilter<Object,Object> fooBarFilter = RowFilter.andFilter(filters);
  19. ((TableRowSorter)this.getRowSorter()).setRowFilter(fooBarFilter);
  20. }
  21. else if(showBigEqStatus==false&&showMiddleEqStatus==false&&showSmallEqStatus==true){
  22. ((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
  23. }
  24. else if(showBigEqStatus==true&&showMiddleEqStatus==true&&showSmallEqStatus==false){
  25. ((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
  26. }
  27. else if(showBigEqStatus==false&&showMiddleEqStatus==true&&showSmallEqStatus==true){
  28. ((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
  29. }
  30. else if(showBigEqStatus==true&&showMiddleEqStatus==false&&showSmallEqStatus==true){
  31. List<RowFilter<Object,Object>> filters = new ArrayList<RowFilter<Object,Object>>(2);
  32. filters.add(RowFilter.numberFilter(ComparisonType.AFTER,TableAttribute.BIG_EARTHQUAKE_THRESHOLD,2));
  33. filters.add(RowFilter.numberFilter(ComparisonType.BEFORE,TableAttribute.MIDDLE_EARTHQUAKE_THRESHOLD,2));
  34. RowFilter<Object,Object> fooBarFilter = RowFilter.orFilter(filters);
  35. ((TableRowSorter)this.getRowSorter()).setRowFilter(fooBarFilter);
  36. }
  37. else if(showBigEqStatus==true&&showMiddleEqStatus==true&&showSmallEqStatus==true){
  38. ((TableRowSorter)this.getRowSorter()).setRowFilter(RowFilter.numberFilter(ComparisonType.AFTER, 0,2));
  39. }
  40. }

补充:将表格视图中的行号映射到模型中对应的行号,才能正确地得到行的内容,尤其是自动排序后,视图中的行号与表格模型中的行号不一致,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);//将视图中的列索引转化为数据模型中的列索引

本文转自:http://blog.csdn.net/b_h_l/article/details/7771944

【转】java JTable排序和过滤的更多相关文章

  1. [Android分享] 【转帖】Android ListView的A-Z字母排序和过滤搜索功能

      感谢eoe社区的分享   最近看关于Android实现ListView的功能问题,一直都是小伙伴们关心探讨的Android开发问题之一,今天看到有关ListView实现A-Z字母排序和过滤搜索功能 ...

  2. Android 实现ListView的A-Z字母排序和过滤搜索功能,实现汉字转成拼音

    转载:http://blog.csdn.net/xiaanming/article/details/12684155 转载请注明出处:http://blog.csdn.net/xiaanming/ar ...

  3. 【JUnit4.10源码分析】6.1 排序和过滤

    abstract class ParentRunner<T> extends Runner implements Filterable,Sortable 本节介绍排序和过滤. (尽管JUn ...

  4. 手把手教你如何用java8新特性将List中按指定属性排序,过滤重复数据

    在java中常常会遇到这样一个问题,在实际应用中,总会碰到对List排序并过滤重复的问题,如果List中放的只是简单的String类型过滤so easy,但是实际应用中并不会这么easy,往往List ...

  5. Java实现敏感词过滤

    敏感词.文字过滤是一个网站必不可少的功能,如何设计一个好的.高效的过滤算法是非常有必要的.前段时间我一个朋友(马上毕业,接触编程不久)要我帮他看一个文字过滤的东西,它说检索效率非常慢.我把它程序拿过来 ...

  6. Java常用排序算法+程序员必须掌握的8大排序算法+二分法查找法

    Java 常用排序算法/程序员必须掌握的 8大排序算法 本文由网络资料整理转载而来,如有问题,欢迎指正! 分类: 1)插入排序(直接插入排序.希尔排序) 2)交换排序(冒泡排序.快速排序) 3)选择排 ...

  7. 让Asp.Net WebAPI支持OData查询,排序,过滤。

    让Asp.Net WebAPI支持OData后,就能支持在url中直接输入排序,过滤条件了. 一.创建Asp.Net WebAPI项目: 二.使用NuGet安装Asp.Net WebAPI 2.2和O ...

  8. Java 常用排序算法/程序员必须掌握的 8大排序算法

    Java 常用排序算法/程序员必须掌握的 8大排序算法 分类: 1)插入排序(直接插入排序.希尔排序) 2)交换排序(冒泡排序.快速排序) 3)选择排序(直接选择排序.堆排序) 4)归并排序 5)分配 ...

  9. Contoso 大学 - 3 - 排序、过滤及分页

    原文 Contoso 大学 - 3 - 排序.过滤及分页 目录 Contoso 大学 - 使用 EF Code First 创建 MVC 应用 原文地址:http://www.asp.net/mvc/ ...

随机推荐

  1. vim删除^M

    1.进入命令模式.vim的命令模式,就是在编辑模式下输入":",光标就会跳到屏幕最后一行,并在那里显示冒号,此时就已经进入命令模式. 命令模式的内容均显示在屏幕的最后一行,按下回车 ...

  2. Linux/UNIX之信号(2)

    信号(2) sigaction函数 sigaction函数的功能是检查或改动与制定信号相关联的处理动作.此函数代替了signal函数. #include <signal.h> int si ...

  3. 一张图总结Google C++编程规范(Google C++ Style Guide)

    Google C++ Style Guide是一份不错的C++编码指南,我制作了一张比較全面的说明图,能够在短时间内高速掌握规范的重点内容.只是规范毕竟是人定的,记得活学活用.看图前别忘了阅读以下三条 ...

  4. 浅谈Storm流式处理框架(转)

    Hadoop的高吞吐,海量数据处理的能力使得人们可以方便地处理海量数据.但是,Hadoop的缺点也和它的优点同样鲜明——延迟大,响应缓慢,运维复杂. 有需求也就有创造,在Hadoop基本奠定了大数据霸 ...

  5. hdu4489(递推dp)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4489 题意:给一个n,求n个高矮不同的人排成一排使得高.矮依次排列的种数. 详细思路参考:http:/ ...

  6. hdu2861(递推)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2861 题意:n个板凳有m个人坐,求刚好将序列分成k段的方式. 分析: a[n][m][k]=a[n-1 ...

  7. Eclipse扩展点实践之添加快捷菜单项(Command方式实现)

    有两种方式,一种是Action的方式,另一种是Command的方式(这两种方式的区别详见:http://wiki.eclipse.org/FAQ_What_is_the_difference_betw ...

  8. three.js是JavaScript编写的WebGL第 三方库

    three.js是JavaScript编写的WebGL第 三方库.提供了非常多的3D显示功能.Three.js 是一款运行在浏览器中的 3D 引擎,你可以用它创建各种三维场景,包括了摄影机.光影.材质 ...

  9. thinkphp框架相关研究(一)

    小编最近开始正式研究thinkphp框架,在此写下研究的整个历程,从最最基本的搭建网站开始,一步步记录.希望对大家有所帮助. 1.菜鸟从下载框架到建站 参考网址:http://blog.csdn.ne ...

  10. HttpMime 处理 多部件 POST 请求

    HttpMime 处理 多部件 POST 请求 在有的场合例如我们要用到上传文件的时候,就不能使用基本的GET请求和POST 请求了,我们要使用多部件的POST请求.由于Android 附带的 Htt ...