利用ListModel构造JList

利用ListModel构造JList:

ListModel是一个interface,主要的功能是定义一些方法,让JList或JComboBox这些组件取得每个项目的值,并可限定项目的显示 时机与方式,下面为ListModel这个interface所定义的方法:

ListModel interface定义的方法:

    void                    addListDataListener(ListDataListener l):当data model的长度或内容值有任何改变时,利用此
                            方法就可以处理ListDataListener的事件。data model是vector或array的数据类型,里面存放List
                            中的值。
    Object                  getElementAt(int index):返回在index位置的Item值。
    int                     getSize():返回List的长度。
    void                    removeListDataListener(ListDataListener l):删除ListDataListener.

还记得我们一开始在介绍JList时所提到的构造函数吗?其中有一个JList的构造函数是这样的:

JList(ListModel dataModel)

因此我们必须实作ListModel所有的方法,才能利用上面这个构造函数建立JList.不过要实现ListModel所有的方法有点麻烦,因 为一般我们不会用到addListDataListener()与removeListDataListener()这两个方法。因此java提供了AbstractListModel这个抽 象类,此抽象类实作了addListDataListener()与removeListDataListener()这两个方法。若我们继承AbstractListModel,就不需 实现这两个方法,只需要实作getElementAt()与getSize()方法即可,我们来看下面的范例:

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

public class JList3 {
	public JList3() {
		JFrame f = new JFrame("JList");
		Container contentPane = f.getContentPane();
		/*
		 * 由于我们在DataModelodel类中继承AbstractListModel,并实作了getElementAt()与getSize()方法
		 * ,因此我们可以由DataModel类产生一个ListModel的实体来。
		 */
		ListModel mode = new DataModel();
		JList list = new JList(mode);// 利用ListModel建立一个JList.
		list.setVisibleRowCount(5);// 设置程序一打开时所能看到的数据项个数。
		list.setBorder(BorderFactory.createTitledBorder("你最喜欢到哪个国家玩呢?"));

		contentPane.add(new JScrollPane(list));
		f.pack();
		/*
		 * 当程序要show出list时,系统会先自动调用getSize()方法,看看这个list长度有多少;然后再调
		 * 用setVisibleRowCount()方
		 * 法,看要一次输出几笔数据;最后调用getElementAt()方法,将list中的项目值(item
		 * )填入list中。读者若还不太清楚,可直接
		 * 在getSize()与getElementAt()方法中个别加入System.out.println
		 * ("size");与System.out.println("element")叙述,就可以 在dos
		 * console清楚看出整个显示list调用的过程。
		 */
		f.show();
		f.addWindowListener(new WindowAdapter() {
			public void windowClosing(WindowEvent e) {
				System.exit(0);
			}
		});
	}

	public static void main(String[] args) {
		new JList3();
	}
}

/*
 * 由于我们在下面程序中继承AbstractListModel抽象类,因此我们分别在下面程序中实现了getElementAt()与getSize()方法。
 */
class DataModel extends AbstractListModel {
	String[] s = { "美国", "越南", "大陆", "英国", "法国", "大陆", "意大利", "澳洲" };

	public Object getElementAt(int index) {// getxElementAt()方法中的参数index,系统会自动由0开始计算,不过要自己作累加
		// 的操作.
		return (index + 1) + "." + s[index++];
	}

	public int getSize() {
		return s.length;
	}
}

事实上,java本身还提供另一个类,DefaultListModel实体类。此类继承了AbstractListModel抽象类,并实现里面所有的抽象方 法,因此你不需要再自行实作任何的方法,可以说是相当的方便。不过既然所有的抽象都已经被实现,因此在设计的弹性上就会有 有所降低。若你是喜欢自行管理JList项目的设计者,你可以不要使用DefaultListModel这个类,只需要在AbstractListModel上多 下功夫即可。

下面的例子我们改写上面的例子,直接使用DefaultListModel类:

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import java.util.Vector;

public class JList4 {
	public JList4() {
		JFrame f = new JFrame("JList");
		Container contentPane = f.getContentPane();

		ListModel mode = new DataModel();
		JList list = new JList(mode);
		list.setVisibleRowCount(5);
		list.setBorder(BorderFactory.createTitledBorder("您最喜欢到哪个国家玩呢?"));

		contentPane.add(new JScrollPane(list));
		f.pack();
		f.show();
		f.addWindowListener(new WindowAdapter() {
			public void windowClosing(WindowEvent e) {
				System.exit(0);
			}
		});
	}

	public static void main(String args[]) {
		new JList4();
	}
}

/*
 * 使DataModel继承DefaultListModel实体类,因此就不需要再实作getSize()与getElementAt()两个方法,
 * 直接将所要的项目用addElementAt()方法加入即可。系统会自动将所加入的项目放入一个Vector对象中,
 * 并在输出JList时自动调用getSize()与getElementAt()方法。你可以在DefaultListModel类中找到getSize()与
 * getElementAt()两个方法,你可以发现这两个方法已经被实作,不再是抽象方法了。
 */
class DataModel extends DefaultListModel {
	String[] s = { "美国", " 日本", "大陆", "英国", "法国", "意大利", "澳洲", "韩国" };

	DataModel() {
		for (int i = 0; i < s.length; i++)
			addElement((i + 1) + "." + s[i]);
	}
}

程序运行结果与上个范例相同。

好奇怪,这不是跟我们使用Vector方式,利用JList(Vector v)构造函数来建立新的JList一样吗?如同JList1.java中的例子,为 什么还要多此一举呢?其实若读者去查看DefaultListModel类,可发现此类提供不少好用的方法,例如你可以随意的增加一个项目( addElement())、或是删除一个项目(removeElement)、甚至你可以很方便地做到查询(getElementAt())与汇出(copyInto())项目的 操作。你可以发现,利用DefaultListModel可以直接动态地更改JList的项目值,而不需要自行产生一个Vecotr对象;相对于JList( Vector v)这个构造函数,可说更方便且实用许多.

至于利用ListModel或AbstractListModel来构造JList有什么好处?读者只要这么想,ListModel中文就是“列出模式”,那么每 个老师都会有自己开课的学生成绩,老师应该可以看到每个同学的成绩,而深长应该只能看到自己的成绩,所以我们就会有两种不 同的“列出模式”。我们只需要去改写getElementAt()方法,就会有不同的列出模式产生,如下面的范例:

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import java.util.Vector;

public class JList5 {
	public JList5() {
		JFrame f = new JFrame("JList");
		Container contentPane = f.getContentPane();
		contentPane.setLayout(new GridLayout(1, 2));
		ListModel mode = new DataModel(1);
		JList list1 = new JList(mode);
		list1.setVisibleRowCount(5);
		list1.setBorder(BorderFactory.createTitledBorder("您玩过哪些软件?"));

		mode = new DataModel(2);
		JList list2 = new JList(mode);
		list2.setBorder(BorderFactory.createTitledBorder("您玩过哪些数据库软件?"));

		contentPane.add(new JScrollPane(list1));
		contentPane.add(new JScrollPane(list2));
		f.pack();
		f.show();
		f.addWindowListener(new WindowAdapter() {
			public void windowClosing(WindowEvent e) {
				System.exit(0);
			}
		});
	}

	public static void main(String args[]) {
		new JList5();
	}
}

class DataModel extends AbstractListModel {
	String[] s = { "MS SQL", "MySQL", "IBM DB2", "ORACLE", "Windows 2000",
			"Linux", "UNix", "OS2" };
	int flag;

	DataModel(int flag) {
		this.flag = flag;
	}

	public Object getElementAt(int index) {
		String tmp = null;

		if (flag == 1)
			tmp = (index + 1) + "." + s[index++];
		if (flag == 2) {
			if (index < 4)
				tmp = (index + 1) + "." + s[index++];
		}

		return tmp;
	}

	public int getSize() {
		return s.length;
	}
}

你可能感兴趣的:(利用ListModel构造JList)