【发布时间】:2013-09-18 14:41:13
【问题描述】:
首先,这不是this question 的重复,尽管它们的标题相同。这个问题是指传递与 C 函数基本相同的方法:它们不需要属于特定对象。在这种情况下,您可以传递 Runnable 或 Callable 对象。
相反,我要问的是:是否可以传递对类中特定方法的引用,并为特定对象调用该方法?
例如,我在 Swing 中查看 FlowLayout 的代码,并注意到 preferredLayoutSize 和 minimumLayoutSize 的实现完全相同,除了一行:
public Dimension preferredLayoutSize(Container target) {
synchronized (target.getTreeLock()) {
Dimension dim = new Dimension(0, 0);
int nmembers = target.getComponentCount();
boolean firstVisibleComponent = true;
boolean useBaseline = getAlignOnBaseline();
int maxAscent = 0;
int maxDescent = 0;
for (int i = 0 ; i < nmembers ; i++) {
Component m = target.getComponent(i);
if (m.isVisible()) {
Dimension d = m.getPreferredSize();
dim.height = Math.max(dim.height, d.height);
if (firstVisibleComponent) {
firstVisibleComponent = false;
} else {
dim.width += hgap;
}
dim.width += d.width;
if (useBaseline) {
int baseline = m.getBaseline(d.width, d.height);
if (baseline >= 0) {
maxAscent = Math.max(maxAscent, baseline);
maxDescent = Math.max(maxDescent, d.height - baseline);
}
}
}
}
if (useBaseline) {
dim.height = Math.max(maxAscent + maxDescent, dim.height);
}
Insets insets = target.getInsets();
dim.width += insets.left + insets.right + hgap*2;
dim.height += insets.top + insets.bottom + vgap*2;
return dim;
}
}
public Dimension minimumLayoutSize(Container target) {
synchronized (target.getTreeLock()) {
boolean useBaseline = getAlignOnBaseline();
Dimension dim = new Dimension(0, 0);
int nmembers = target.getComponentCount();
int maxAscent = 0;
int maxDescent = 0;
boolean firstVisibleComponent = true;
for (int i = 0 ; i < nmembers ; i++) {
Component m = target.getComponent(i);
if (m.visible) {
Dimension d = m.getMinimumSize();
dim.height = Math.max(dim.height, d.height);
if (firstVisibleComponent) {
firstVisibleComponent = false;
} else {
dim.width += hgap;
}
dim.width += d.width;
if (useBaseline) {
int baseline = m.getBaseline(d.width, d.height);
if (baseline >= 0) {
maxAscent = Math.max(maxAscent, baseline);
maxDescent = Math.max(maxDescent,
dim.height - baseline);
}
}
}
}
if (useBaseline) {
dim.height = Math.max(maxAscent + maxDescent, dim.height);
}
Insets insets = target.getInsets();
dim.width += insets.left + insets.right + hgap*2;
dim.height += insets.top + insets.bottom + vgap*2;
return dim;
}
}
preferredLayoutSize 方法调用m(i-th 组件)的preferredLayoutSize 方法,而minimumLayoutSize 调用m 的minimumLayoutSize 方法。据我所知,这两种方法在其他方面是相同的。
任何programmer 都会告诉你,重复代码是一件坏事。然而,在这种情况下,如何摆脱重复代码并不明显。很明显,应该有一个私有方法,其中包含两个公共方法调用的代码,传入对Component 类的preferredLayoutSize 和minimumLayoutSize 方法的引用。在 C 中,我可以使用函数指针来做到这一点,因此在 Java 中应该有某种方式来做到这一点是有道理的。传入Runnable 或Callable 几乎可以工作,但都没有返回值。 编辑:这是错误的。 Callable 中的重写方法确实 返回一个值,并且它所作用的对象可以作为参数传入。
既然我已经输入了所有内容,我想到了一个解决方案:您可以使用名为 Dimension layoutSize(Component comp) 的方法编写一个 interface 并编写两个实现,其中一个返回 comp.preferredLayoutSize(),另一个返回返回comp.minimumLayoutSize()。然后,您可以编写一个将该接口的实例作为参数的私有方法,并使用它在代码中的正确位置运行单独的方法。您甚至可以使用匿名内部类,因此您不必为每种布局大小类型编写新类。不过,对于一个相当简单的问题来说,这似乎仍然很麻烦。有没有更简单的方法?
【问题讨论】:
-
通过反射,查找
Method类。您还需要对invoke()它的对象的引用。 -
我猜 Java 8 是可能的。
-
或者,更好的是,使用
MethodHandle。 -
不通过
Callable仍然适用吗?Callable中的代码将调用对象上的相关方法。 -
代码重复并不好,但如果人们仅仅因为认为代码重复是一件可怕的事情而竭尽全力,那就更糟了。然后你想出令人费解的不可读的“解决方案”,其中重复的代码被删除,但剩下的代码看起来像是一个疯子写的。
标签: java swing customization layout-manager