版权声明:本文为博主原创文章,未经博主允许不得转载。
参考资料:
http://263229365.iteye.com/blog/1040329
https://www.java.net/node/650758
https://community.oracle.com/thread/2332288?start=0&tstart=0
问题的产生:
在看JComboBox组件的监听事件时,运行讲解到的实例(增加了几个输出语句来跟踪它的反应)碰到下面的问题:
1, JComboBox组件在添加了ActionListener和ItemListener监听器时, 两个监听器都对JComboBox值的改变做出了相应的操作。
2, 尤其是ActionListener的事件处理,不管怎么样都始终执行了两次。。。。
(1)运行下面的程序代码:
- package org.forfan06.jcomboboxdemo;
- import java.awt.Container;
- import java.awt.Font;
- import java.awt.GridLayout;
- import java.awt.event.WindowAdapter;
- import java.awt.event.WindowEvent;
- import java.awt.event.ItemListener;
- import java.awt.event.ItemEvent;
- import java.awt.event.ActionEvent;
- import java.awt.event.ActionListener;
- import javax.swing.BorderFactory;
- import javax.swing.JComboBox;
- import javax.swing.JFrame;
- import javax.swing.JLabel;
- class MyComboBox implements ActionListener, ItemListener{
- private JFrame jframe = new JFrame("Welcome to my blog");
- private Container container = jframe.getContentPane();
- private JComboBox jcb1 = null;
- private JLabel label = new JLabel("www.csnd.net");
- private String fontSize[] = {"10", "12", "14", "16", "18", "20", "22", "24", "26", "48", "72"};
- public MyComboBox(){
- this.jframe.setLayout(new GridLayout(2, 1));
- this.jcb1 = new JComboBox(this.fontSize);
- jcb1.setBorder(BorderFactory.createTitledBorder("请选择显示文字大小"));
- jcb1.addItemListener(this);
- jcb1.addActionListener(this);
- jcb1.setMaximumRowCount(3);;
- jcb1.setEditable(true);
- jcb1.configureEditor(jcb1.getEditor(), "12");
- this.changeFontSize(12);
- container.add(this.jcb1);
- container.add(label);
- this.jframe.setSize(300, 150);
- this.jframe.setVisible(true);
- this.jframe.addWindowListener(new WindowAdapter(){
- public void windowClosing(WindowEvent e){
- System.exit(1);
- }
- });
- }
- public void itemStateChanged(ItemEvent e){
- if(e.getStateChange() == ItemEvent.SELECTED){
- String itemSize = (String) e.getItem();
- try{
- System.out.println("---ItemEvent performed:" + e.paramString() + "---");
- this.changeFontSize(Integer.parseInt(itemSize)); //改变字体
- }catch(Exception ex){
- }
- }
- }
- public void actionPerformed(ActionEvent e){
- String itemSize = (String) this.jcb1.getSelectedItem();
- int size = 12;
- try{
- size = Integer.parseInt(itemSize);
- }catch(Exception ex){
- this.jcb1.getEditor().setItem("输入的不是数字。");
- }
- System.out.println("---action performed " +e.getActionCommand() + "---");
- this.changeFontSize(size);
- if(!this.isExists(itemSize)){
- this.jcb1.addItem(jcb1.getSelectedItem());
- }
- }
- public boolean isExists(String item){
- boolean flag = false;
- for(int i = 0; i < this.jcb1.getItemCount(); i++){
- if(item.equals(this.jcb1.getItemAt(i))){
- flag = true;
- }
- }
- return flag;
- }
- public void changeFontSize(int size){
- Font fnt = new Font("Serief", Font.BOLD, size);
- this.label.setFont(fnt);
- }
- }
- public class JComboBoxDemo03 {
- public static void main(String args[]){
- MyComboBox myComboBox = new MyComboBox();
- }
- }
(2)查看运行结果:
Option 1: 从下拉列表里面选择一个值
- ---ItemEvent performed: ITEM_STATE_CHANGED, item=48, stateChange=SELECTED---
- ---action performed comboBoxChanged---
Option 2: 直接键盘输入一个值
- ---ItemEvent performed: ITEM_STATE_CHANGED, item=25, stateChange=SELECTED---
- ---action performed comboBoxChanged---
- ---action performed comboBoxEdited---
可以发现,如果此时在JComboBox同时加入ItemListener和 ActionListener的话,两个监听器都会做出相应的动作(并且ActionEvent做出了两次反应 twice)!!!
分析:
1, 如果在覆些ItemListener的itemStateChanged方法中的判断去掉的话,ItemListener同样也会做出两次动作!!第一次是前一个值的DESELECTED触发;第二次是后一个值的SELECTED触发。 这里通过判断触发源的StateChange来进行过滤,只对后一次值的SELECTED做出动作!!!!!
- ItemListener接口中定义的itemStateChanged(ItemEvent e)将执行需要在已选定(或已取消选定)项时发生的操作;
- 上次被选中的项的State由SELECTED变为DESELECTED,即取消选择;
- 本次选中的项的State由DESELECTED变为SELECTED,即新选中;
- itemStateChanged()事件与itemState有关!!!itemState在这里有两个状态,Selected和Ddselected
所以itemStateChanged事件中的代码要被执行两次!!!!!!
2, https://www.java.net/node/650758
查看JComboBox的源代码例子,对于JComboBox有个actionCommand的String存在!!!其可能的两个值“comboBoxChanged”和“comboBoxEdited”被hardcoded到源码里面了。同样也造成了有两个ActionPerformed产生了。
对于有两个actionPerformed产生,有以下解决方案:
Step1, 保存上次选定项的值
Step2, 仅对选项值是新(新选项并且该选项的值不等于上次选定项的值)的事件进行捕获
3,针对ActionListener,因为与两次的输出是当actionCommand的值分别是comboBoxChanged和comboBoxEdited做出reaction。
由于actionCommand的值是String类型,所以只需要do a simple equals test 就可以了。。修改actionPerformed(ActionEvent e)方法如下就可以过滤掉comboBoxEdited的触发事件了
- public void actionPerformed(ActionEvent e){
- if(e.getActionCommand().equals("comboBoxChanged")){
- String itemSize = (String) this.jcb1.getSelectedItem();
- int size = 12;
- try{
- size = Integer.parseInt(itemSize);
- }catch(Exception ex){
- this.jcb1.getEditor().setItem("输入的不是数字。");
- }
- System.out.println("---action performed " +e.getActionCommand() + "---");
- this.changeFontSize(size);
- if(!this.isExists(itemSize)){
- this.jcb1.addItem(jcb1.getSelectedItem());
- }
- }
- }
还有另外一种解决方案来自 https://community.oracle.com/thread/2332288?start=0&tstart=0 同学
- //The Listener has to be attached to the editor in stead of the comboBox itself
- comboBox.getEditor().addActionListener(new ActionListener()){
- ...
- }
但是好像,上面的这个editor监听器的方案不好处理啊。还是我没有找到正确的处理方法~~~它失去了监听JComboBox组件 这一最初始的目的啊~~~~~~~~~
4,JComboBox is a little bit different than other components, because there are two actions, you shouldn't set the command, instead you should check them, as follows.
if ("comboBoxEdited".equals(evt.getActionCommand()))
else if ("comboBoxChanged".equals(evt.getActionCommand()))
总结
1,ActionListener是所有监听器的父类,其他监听器可以监听的事件都可以被它捕获
2, ItemListener用于捕获带Item的组件产生的事件,抽象方法itemStateChanged(ItemEvent e)执行需要在已选定(或已取消选定)项时发生的操作
3, (仔细查看event事件)ActionListener报告的2个不同的事件: 一个是编辑文本框 comboBoxEdited,另外一个是选定值 comboBoxChanged。
问题的解决:
1, 网上有人说针对JComboBox一般添加ItemListener监听器即可 (ActionListener是所有监听器的父类,其他监听器可以监听的事件都可以被ActionListener捕获)
2, 通过对ActionEvent的actionCommand值的判断,来过滤掉不需要的监听事件。 例如,我们只需要捕获值的改变(actionCommand = comboBoxChanged)。
3, 增加一点ItemListener监听器的事件的过滤,是通过事件的 getStateChanged()的值(它只有两个值: ItemEvent.SELECTED 和 ItemEvent.DESELECTED)来判断的。