- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我目前正在研究可以处理任何网格状结构上的操作的抽象代码。我一直在尝试开发一个框架来代替具体方法,该框架接受 lambda 表达式并提供基于网格元素之间的方向关系的功能,并且在尝试将泛型类型返回到这些方法中时遇到了麻烦。
我有一个像下面这样的工作方法,它接受一个 lambda 函数,该函数在特定方向上查找网格元素,以及一个在每个元素上执行某种方法的 Consumer。
public static <E> void forEachClockwise(Direction d, Function<Direction, ? extends E> f, Consumer<? super E> c){
/*For each of the orthogonal directions, in order:*/
c.accept(f.apply(d));
/*end For*/
}
不必每次调用此方法时都必须传递元素查找函数,我认为最好有一个声明了此类函数的“正交”接口(interface),并重载 forEachClockwise 方法以接受此元素界面。
public static <O extends Orthogonal> void forEachClockwise(Direction d, O center, Consumer<O> c){
forEachClockwise(d, center::findNextByDirection, c);
}
我在这里使用了泛型并进行了声明,以便 Consumer 方法能够访问网格元素的字段而无需进行强制转换。我担心这是一个糟糕的设计选择,因为尽管尝试了一些不同的事情,但我创建的正交接口(interface)和测试类在没有警告和强制转换的情况下还没有编译。
public interface Orthogonal {
public <E extends Orthogonal> E findNextByDirection(Direction a);
}
我尝试过的 findNextByDirection 的两个“有效”实现是:
/* in Coordinate implements Orthogonal: */
@Override
public <E extends Orthogonal> E findNextByDirection(Direction a) {
return (E) (/*The proper Coordinate*/);
}
和
@Override
public Coordinate findNextByDirection(Direction a) {
return /*The proper Coordinate*/;
}
//Warning on return type declaration:
/*Type safety: The return type Element for findNextByDirection(Direction) from the type Element needs unchecked conversion to conform to E from the type Orthogonal*/
最理想的是,我希望有一个 forEachClockwise (和其他方法)可以接受的方法,它返回调用它的同一类的元素,而不进行强制转换。坐标的 findNextByDirection 应该返回一个坐标,地址应该返回一个地址,单元格应该返回单元格等。我看过的其他问题和答案讨论了如何摆脱使用通用返回类型的方法,但我还没有找到有关如何使用此类方法作为 lambda 参数的任何提示。
我还没有尝试过的几件事是创建一个新的 findByDirection 方法,该方法接受某种正交和方向作为参数,或者使用正交中定义的某些方法调用 forEachClockwise 方法,尽管这些方法听起来合理我尝试过什么。我见过的另一种可行的方法是为接口(interface)提供自己的泛型类型,但将每个正交类声明为“MyClass 实现正交”感觉明显错误且违背目的我是否从一开始就完全错误地设计了这个界面,并期望泛型做一些他们没有设计目的的事情?或者是否有更简单的方法让类调用正确的方法并一般返回其自己类型的元素?
完整代码:
package test;
import java.util.function.Consumer;
import java.util.function.Function;
public enum Direction {
NORTH, EAST, SOUTH, WEST;
public static <O extends Orthogonal> void forEachClockwise(Direction d, O center, Consumer<O> c){
forEachClockwise(d, center::findNextByDirection, c);
}
public static <X> void forEachClockwise(Direction d, Function<Direction, ? extends X> f, Consumer<? super X> c){
c.accept(f.apply(d));
forEachExcept(d, f, c);
}
public static <X> void forEachExcept(Direction d, Function<Direction, ? extends X> f, Consumer<? super X> c){
for(int i=0; i<3; i++){
d = Direction.getClockwise(d);
c.accept(f.apply(d));
}
}
public static Direction getClockwise(Direction d){
switch(d){
case EAST:
return SOUTH;
case NORTH:
return EAST;
case SOUTH:
return WEST;
case WEST:
return NORTH;
default:
return null;
}
}
}
interface Orthogonal {
public <E extends Orthogonal> E findNextByDirection(Direction a);
}
class Coordinate implements Orthogonal{
int x;
int y;
public Coordinate(int x, int y){
this.x = x;
this.y = y;
}
public int getX(){
return x;
}
public int getY() {
return y;
}
@Override //Warning on "Coordinate" below. Compiler suggests writing
//entire <E extends Orthogonal> method signature
public Coordinate findNextByDirection(Direction a) {
switch(a){
case NORTH:
return new Coordinate(x+1, y);
case SOUTH:
return new Coordinate(x-1, y);
case EAST:
return new Coordinate(x, y+1);
case WEST:
return new Coordinate(x, y-1);
default:
return null;
}
}
}
最佳答案
让我们假设有两个类实现 Orthogonal
-Coordinate
和Area
。
public <E extends Orthogonal> E findNextByDirection(Direction a);
您在这里声明的是一个模板化方法。该声明基本上说:“给定特定上下文,此方法将始终返回预期类型 - 如果将结果分配给 Coordinate
变量,则该方法将返回 Coordinate
,如果分配给 Area
-它将返回 Area
等等 - 只要预期的是实现 Orthogonal
的类型“
但是,当您尝试在 Coordinate
中实现它时如public Coordinate findNextByDirection(Direction a)
,您不再 promise 返回给定上下文中预期的任何内容 - 您现在只返回 Coordinate
,从而破坏了接口(interface)中声明的契约。更糟糕的是,如果你仔细想想,一开始就没有办法真正履行这份契约(Contract)。
你应该做的是声明你的接口(interface)通用( Orthogonal<E>
)并删除 <E extends Orthogonal>
来自方法声明。
如果是声明class Coordinate implements Orthogonal<Coordinate>
感觉“明显错误”,其实不应该。这是常见的做法,在 SDK 本身中广泛使用(例如,请参阅Comparable
)。
如果您觉得有必要findNextByDirection
返回Orthogonal
,您可以将返回类型声明为 Orthogonal<E>
。您甚至应该能够在 Coordinate
中实现您的方法如public Coordinate findNextByDirection(Direction a)
由于 Java 的类型协变,没有任何警告。
关于java - 通用返回类型和 lambda 函数参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41366170/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!