- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个 JTable
,其中为表头设置了组件菜单。它包含用于删除列的条目。当删除触发组件菜单的同一列时,出现问题,并且似乎将此操作视为被拖动的列,这会导致 ArrayIndexOutOfBoundsException
与 -1
.
如何安全地从组件菜单中删除当前列而不引发此错误?
这是一个如何触发这种行为的最小示例。只需运行它并删除您所在的同一列(请注意,当您删除最后一列时,它似乎可以工作,但这与我无关):
import java.awt.BorderLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;
import javax.swing.JTable;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableColumn;
public class BasicExample extends JFrame {
private JTable t;
private DefaultTableModel dtm;
public BasicExample() {
init();
}
private void init() {
dtm = new DefaultTableModel(new String[][]{{"a", "b"}, {"c", "d"}}, new String[]{"A", "B"});
t = new JTable(dtm);
t.getTableHeader().setComponentPopupMenu(new PopupMenu(t));
this.setLayout(new BorderLayout());
add(t.getTableHeader(), BorderLayout.NORTH);
add(t, BorderLayout.CENTER);
pack();
setVisible(true);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
public class PopupMenu extends JPopupMenu {
private JTable table;
public PopupMenu(JTable table) {
this.table = table;
init();
}
private void init() {
for (int i = 0; i < table.getModel().getColumnCount(); i++) {
String columnName = table.getModel().getColumnName(i);
JMenuItem item = new JMenuItem(columnName);
item.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
TableColumn tc = table.getColumn(columnName);
table.getColumnModel().removeColumn(tc);
}
});
this.add(item);
}
}
}
public static void main(String[] args) {
BasicExample be = new BasicExample();
}
}
在堆栈跟踪中,我看到它似乎将该列视为已拖动,因为它在 BasicTableHeaderUI.java
中输入条件 if (draggedColumn != null) {
,而我正在执行的操作并不是真正的阻力。完整的堆栈跟踪如下:
Exception in thread "AWT-EventQueue-0" java.lang.ArrayIndexOutOfBoundsException: -1
at java.util.Vector.elementData(Vector.java:734)
at java.util.Vector.elementAt(Vector.java:477)
at javax.swing.table.DefaultTableColumnModel.getColumn(DefaultTableColumnModel.java:294)
at javax.swing.plaf.basic.BasicTableHeaderUI.getHeaderRenderer(BasicTableHeaderUI.java:693)
at javax.swing.plaf.basic.BasicTableHeaderUI.paintCell(BasicTableHeaderUI.java:709)
at javax.swing.plaf.basic.BasicTableHeaderUI.paint(BasicTableHeaderUI.java:685)
at javax.swing.plaf.ComponentUI.update(ComponentUI.java:161)
at javax.swing.JComponent.paintComponent(JComponent.java:780)
at javax.swing.JComponent.paint(JComponent.java:1056)
at javax.swing.JComponent.paintToOffscreen(JComponent.java:5210)
at javax.swing.RepaintManager$PaintManager.paintDoubleBuffered(RepaintManager.java:1579)
at javax.swing.RepaintManager$PaintManager.paint(RepaintManager.java:1502)
at javax.swing.RepaintManager.paint(RepaintManager.java:1272)
at javax.swing.JComponent._paintImmediately(JComponent.java:5158)
at javax.swing.JComponent.paintImmediately(JComponent.java:4969)
at javax.swing.RepaintManager$4.run(RepaintManager.java:831)
at javax.swing.RepaintManager$4.run(RepaintManager.java:814)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(ProtectionDomain.java:76)
at javax.swing.RepaintManager.paintDirtyRegions(RepaintManager.java:814)
at javax.swing.RepaintManager.paintDirtyRegions(RepaintManager.java:789)
at javax.swing.RepaintManager.prePaintDirtyRegions(RepaintManager.java:738)
at javax.swing.RepaintManager.access$1200(RepaintManager.java:64)
at javax.swing.RepaintManager$ProcessingRunnable.run(RepaintManager.java:1732)
at java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:311)
at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:756)
at java.awt.EventQueue.access$500(EventQueue.java:97)
at java.awt.EventQueue$3.run(EventQueue.java:709)
at java.awt.EventQueue$3.run(EventQueue.java:703)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(ProtectionDomain.java:76)
at java.awt.EventQueue.dispatchEvent(EventQueue.java:726)
at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:201)
at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:116)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:105)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:101)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:93)
at java.awt.EventDispatchThread.run(EventDispatchThread.java:82)
如果相关的话,这会在 Java 8 中发生。更改 JRE/JDK 不是一个选项。
最佳答案
我们的环境中意外地遇到了相同的错误,但确实是零星的,所以我无法重现它。但你的例子是完美的,我能够重现我的错误。请参阅下面的代码。
我在调查过程中修改了我的答案。旧的注释不会反射(reflect)最新的代码。
您的问题是已报告的错误: https://bugs.java.com/bugdatabase/view_bug.do?bug_id=8068824
这是一个基于错误解决方法的解决方案:
import java.awt.BorderLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;
import javax.swing.JTable;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableColumn;
public class BasicExample extends JFrame {
private JTable t;
private DefaultTableModel dtm;
public BasicExample() {
init();
}
private void init() {
dtm = new DefaultTableModel(new String[][] { { "a", "b" }, { "c", "d" } },
new String[] { "A", "B" });
t = new JTable(dtm);
t.getTableHeader().setComponentPopupMenu(new PopupMenu(t));
this.setLayout(new BorderLayout());
add(t.getTableHeader(), BorderLayout.NORTH);
add(t, BorderLayout.CENTER);
pack();
setVisible(true);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
public class PopupMenu extends JPopupMenu {
private JTable table;
public PopupMenu(JTable table) {
this.table = table;
init();
}
private void init() {
for (int i = 0; i < table.getModel().getColumnCount(); i++) {
String columnName = table.getModel().getColumnName(i);
JMenuItem item = new JMenuItem(columnName);
item.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
TableColumn tc = table.getColumn(columnName);
t.getTableHeader().setDraggedColumn(null);
table.getColumnModel().removeColumn(tc);
}
});
this.add(item);
}
}
}
public static void main(String[] args) {
BasicExample be = new BasicExample();
}
}
只需在删除之前将draggedColumn设置为null即可。
我一直在挖掘,因为它没有解释为什么在我们的环境中该错误很少发生并且仅在 QA 中发生。 :)
这是代码在我们的环境中的外观(再次感谢非常好的小示例)。
import java.awt.BorderLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.JFrame;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;
import javax.swing.JTable;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableColumn;
public class BasicExample extends JFrame {
private JTable t;
private DefaultTableModel dtm;
public BasicExample() {
init();
}
private void init() {
dtm = new DefaultTableModel(new String[][] { { "a", "b" }, { "c", "d" } },
new String[] { "A", "B" });
t = new JTable(dtm);
PopupMenu lPopupMenu = new PopupMenu(t);
// t.getTableHeader().setComponentPopupMenu(new PopupMenu(t));
t.getTableHeader().addMouseListener(new MouseAdapter() {
@Override
public void mousePressed(MouseEvent pE) {
lPopupMenu.show(pE.getComponent(), pE.getX(), pE.getY());
}
});
this.setLayout(new BorderLayout());
add(t.getTableHeader(), BorderLayout.NORTH);
add(t, BorderLayout.CENTER);
pack();
setVisible(true);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
public class PopupMenu extends JPopupMenu {
private JTable table;
public PopupMenu(JTable table) {
this.table = table;
init();
}
private void init() {
for (int i = 0; i < table.getModel().getColumnCount(); i++) {
String columnName = table.getModel().getColumnName(i);
JMenuItem item = new JMenuItem(columnName);
item.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
TableColumn tc = table.getColumn(columnName);
// t.getTableHeader().setDraggedColumn(null);
table.getColumnModel().removeColumn(tc);
}
});
this.add(item);
}
}
}
public static void main(String[] args) {
BasicExample be = new BasicExample();
}
}
与您的代码的区别在于我们不使用setComponentPopupMenu(),而是手动显示PopupMenu。并且您的用例不会在我们的代码中产生异常。
为什么?
好问题,但我没有答案。
不知何故,您的代码 JPopupMenu 消耗了鼠标事件。并使用我们的代码 JPopupMenu 将鼠标事件传播到底层组件。如果你用 UI 代替表头就可以看到它。
这就是我用我们的代码重现异常的方式:在标题上按鼠标右键,但不要释放它,并继续将鼠标移到远离标题的弹出窗口上。当指针远离标题时释放右键(这样标题就不会处理 MOUSE_RELEASED 事件)。获取异常。
关于java - 从同一列的组件菜单中删除 JTable 列时出错,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58432804/
我正在 csv 上使用 hadoop 来分析一些数据。我使用sql/mysql(不确定)来分析数据,现在陷入了僵局。 我花了好几个小时在谷歌上搜索,却没有找到任何相关的东西。我需要一个查询,在该查询中
我正在为 Bootstrap 网格布局的“简单”任务而苦苦挣扎。我希望在大视口(viewport)上有 4 列,然后在中型设备上有 2 列,最后在较小的设备上只有 1 列。 当我测试我的代码片段时,似
对于这个令人困惑的标题,我深表歉意,我想不出这个问题的正确措辞。相反,我只会给你背景信息和目标: 这是在一个表中,一个人可能有也可能没有多行数据,这些行可能包含相同的 activity_id 值,也可
具有 3 列的数据库表 - A int , B int , C int 我的问题是: 如何使用 Sequelize 结果找到 A > B + C const countTasks = await Ta
我在通过以下功能编写此查询时遇到问题: 首先按第 2 列 DESC 排序,然后从“不同的第 1 列”中选择 只有 Column1 是 DISTINCT 此查询没有帮助,因为它首先从第 1 列中进行选择
使用 Bootstrap 非常有趣和有帮助,目前我在创建以下需求时遇到问题。 “使用 bootstrap 在桌面上有 4 列,在平板电脑上有 2 列,在移动设备上有 1 列”谁能告诉我正确的结构 最佳
我是 R 新手,正在问一个非常基本的问题。当然,我在尝试从所提供的示例中获取指导的同时做了功课here和 here ,但无法在我的案例中实现这个想法,即可能是由于我的问题中的比较维度更大。 我的实
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个 df , delta1 delta2 0 -1 2 0 -1 0 0 0 我想知道如何分配 delt
您好,我想知道是否可以执行以下操作。显然,我已经尝试在 phpMyAdmin 中运行它,但出现错误。也许还有另一种方式来编写此查询。 SELECT * FROM eat_eat_restaurants
我有 2 个列表(标题和数据值)。我想要将数据值列 1 匹配并替换为头文件列 1,以获得与 dataValue 列 1 和标题值列 2 匹配的值 头文件 TotalLoad,M0001001 Hois
我有两个不同长度的文件,file2 是一个很大的引用文件,我从中提取文件 1 的数据。 我有一行 awk,我通常会对其进行调整以在我的文件中进行查找和替换,但它总是在同一列中进行查找和替换。 所以对于
假设我有两个表,如下所示。 create table contract( c_ID number(1) primary key, c_name varchar2(50) not
我有一个带有 varchar 列的 H2 表,其检查约束定义如下: CONSTRAINT my_constraint CHECK (varchar_field <> '') 以下插入语句失败,但当我删
这是最少量的代码,可以清楚地说明我的问题: One Two Three 前 2 个 div 应该是 2 个左列。第三个应该占据页面的其余部分。最后,我将添加选项来隐藏和
在 Azure 中的 Log Analytics 中,我为 VM Heartbeat 选择一个预定义查询,我在编辑器中运行查询正常,但当我去创建警报时,我不断收到警报“查询未返回 TimeGenera
在 Azure 中的 Log Analytics 中,我为 VM Heartbeat 选择一个预定义查询,我在编辑器中运行查询正常,但当我去创建警报时,我不断收到警报“查询未返回 TimeGenera
今天我开始使用 JexcelApi 并遇到了这个:当您尝试从特定位置获取元素时,不是像您通常期望的那样使用sheet.getCell(row,col),而是使用sheet.getCell(col,ro
我有一个包含 28 列的数据库。第一列是代码,第二列是名称,其余是值。 public void displayData() { con.Open(); MySqlDataAdapter
我很沮丧:每当我缩小这个网页时,一切都变得一团糟。我如何将网页居中,以便我可以缩小并且元素不会被错误定位。 (它应该是 2 列,但所有内容都合并为 1)我试过 但由于某种原因,这不起作用。 www.o
我是一名优秀的程序员,十分优秀!