Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/349.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java 对数组中的每个标签使用addMouseListener()?_Java_Swing_Mouselistener - Fatal编程技术网

Java 对数组中的每个标签使用addMouseListener()?

Java 对数组中的每个标签使用addMouseListener()?,java,swing,mouselistener,Java,Swing,Mouselistener,我目前正在编写一个程序,该程序在for循环中创建并排列了28个标签,我想为每个标签添加一个鼠标侦听器。我可以向每个标签添加一个mouseListener,但是当我尝试指定标签编号并更改标签的边框颜色时,我得到一个“从内部类访问局部变量需要声明为最终”错误。我理解这个错误背后的原因,但我不能使“labelNum”变量成为最终变量,因为它需要更改,因为它在循环中。这里的变通解决方案是什么 for (int c = 0; c < 7; c++) { for (int d = 0; d &

我目前正在编写一个程序,该程序在for循环中创建并排列了28个标签,我想为每个标签添加一个鼠标侦听器。我可以向每个标签添加一个mouseListener,但是当我尝试指定标签编号并更改标签的边框颜色时,我得到一个“从内部类访问局部变量需要声明为最终”错误。我理解这个错误背后的原因,但我不能使“labelNum”变量成为最终变量,因为它需要更改,因为它在循环中。这里的变通解决方案是什么

for (int c = 0; c < 7; c++) {
    for (int d = 0; d < 4; d++) {
                labelNum = c*4+d;
                inventorySlotLabels[labelNum] = new JLabel();
                inventorySlotLabels[labelNum].setHorizontalAlignment(javax.swing.SwingConstants.CENTER);
                inventorySlotLabels[labelNum].setBorder(javax.swing.BorderFactory.createLineBorder(new java.awt.Color(0, 0, 0)));
                inventorySlotLabels[labelNum].setBounds(15+d*40+d*10, 15+c*30+c*10, 40, 30);
                inventorySlotLabels[labelNum].setIcon(new javax.swing.ImageIcon(getClass().getResource("Item.png")));

                inventorySlotLabels[labelNum].addMouseListener(new java.awt.event.MouseAdapter() {
                    public void mouseClicked(java.awt.event.MouseEvent evt) {
                        headerLabel.setText("Inventory Slot Clicked");
                        inventorySlotLabels[labelNum].setBorder(BorderFactory.createLineBorder(Color.WHITE));

                    }
                }
    }
}
for(int c=0;c<7;c++){
对于(int d=0;d<4;d++){
labelNum=c*4+d;
inventorySlotLabels[labelNum]=新的JLabel();
inventorySlotLabels[labelNum].setHorizontalAlignment(javax.swing.SwingConstants.CENTER);
inventorySlotLabels[labelNum].SetBordOrder(javax.swing.BorderFactory.createLineBorder(新java.awt.Color(0,0,0));
目录标签[labelNum].setBounds(15+d*40+d*10,15+c*30+c*10,40,30);
inventorySlotLabels[labelNum].setIcon(新的javax.swing.ImageIcon(getClass().getResource(“Item.png”));
inventorySlotLabels[labelNum].addMouseListener(新java.awt.event.MouseAdapter()){
public void mouseClicked(java.awt.event.MouseEvent evt){
headerLabel.setText(“单击的库存插槽”);
inventorySlotLabels[labelNum].setBorder(BorderFactory.createLineBorder(Color.WHITE));
}
}
}
}
不要依赖标签号变量。从MouseeEvent获取标签:

public void mouseClicked(java.awt.event.MouseEvent evt) 
{
    headerLabel.setText("Inventory Slot Clicked");
    //inventorySlotLabels[labelNum].setBorder(BorderFactory.createLineBorder(Color.WHITE));
    JLabel label = (JLabel)evt.getSource();
    label.setBorder( ... );
}
编辑:

实际上,没有理由创建28个鼠标侦听器。在循环之前,代码应该更像:

MouseListener ml = new MouseListener()
{
    public void mouseClicked(java.awt.event.MouseEvent evt) 
    {
        headerLabel.setText("Inventory Slot Clicked");
        //inventorySlotLabels[labelNum].setBorder(BorderFactory.createLineBorder(Color.WHITE));
        JLabel label = (JLabel)evt.getSource();
        label.setBorder( ... );
    }
};
然后在循环中,您只需使用:

inventorySlotLabels[labelNum].addMouseListener( ml );
无论您使用的是内部类还是内部类,更好的设计方法是使用MouseEvent中的源对象,不需要创建28个MouseListener来存储一个变量来索引数组中的标签。事实上,使用这种方法时甚至不需要保留数组,这将简化您的代码更重要的是,现在不再局限于硬编码值来确定数组的大小

此外,无需将图标读取28次。只需在循环外读取图标一次,然后将图标添加到每个标签。您也不需要创建28个边框。只需创建边框一次并共享即可

最后,不应使用setBounds(…)来定位和调整标签的大小。请使用适当的布局管理器(可能是GridLayout)在网格中显示标签

“我得到一个“从内部类访问局部变量需要声明为最终”错误。我理解此错误背后的原因,但我无法使“labelNum”变量为最终变量,因为它需要更改,因为它在循环中。这里的解决方案是什么?


简单回答:解决方法是将对象全局声明为类成员

简单修复:declare labelNum final。是的,这看起来很奇怪,但它可以工作

for (int c = 0; c < 7; c++) {
  for (int d = 0; d < 4; d++) {
    final int labelNum = c*4+d;
    ...
for(int c=0;c<7;c++){
对于(int d=0;d<4;d++){
最终int labelNum=c*4+d;
...
备用修复程序就像MJafars(使用内部类),所以我删除了它


而且,正如我在原始(现已大部分删除)帖子中所指出的,@Camickr有一个合适的解决方案,即查看事件源,只创建一个侦听器。

为什么要使用匿名内部类?使用一个简单的内部类

class InventoryMouseListener extends java.awt.event.MouseAdapter {
     private int labelNum;
     public InventoryMouseListener(int labelNum) {
         this.labelNum = labelNum;
     }
     @Override
     public void mouseClicked(java.awt.event.MouseEvent evt) {
         headerLabel.setText("Inventory Slot Clicked");
         inventorySlotLabels[labelNum].setBorder(BorderFactory.createLineBorder(Color.WHITE));
     }
}
并添加类似于

inventorySlotLabels[labelNum]
   .addMouseListener(new InventoryMouseListener(labelNum));

是否可行

确定可以使用内部类,但如果可以创建单个实例并使用MouseEvent的源对象编写泛型代码来获取JLabel,那么为什么要创建28个类实例?确定可以使用内部类,但如果可以创建,为什么要创建28个类实例吃了一个实例,并通过使用MouseeEvent的源对象编写通用代码来获取JLabel?在这个特定的示例中,您是100%正确的,但在其他情况下,他可能需要这个“技巧”。哇,非常感谢!正如您所知,我对这方面的知识还比较陌生,但这非常有用!
inventorySlotLabels[labelNum]
   .addMouseListener(new InventoryMouseListener(labelNum));