我非常熟悉Scheme和Haskell等功能语言。我正在努力解决Java中的问题而苦苦挣扎,可能是因为我仍处于功能性思维模式中。
我想写:
public void doQueryAndStoreData(String query, <? extends Collection> storeIn) {
/* make a jdbc query, get ResultSet */
ResultSet rset = ...;
ProcessResultSet proc = new ProcessResultSet();
proc.process(rset, storeIn);
/* clean up */
}
使用如下界面:
private interface IProcessResultSet<C> {
public void process(ResultSet rset, C storeIn);
}
和一个实现接口的类,如:
private class ProcessResultSet implements IProcessResultSet<? extends Collection> {
public void process(ResultSet rset, Map storeIn) {
/* do something */
}
public void process(ResultSet rset, List storeIn) {
/* do something else */
}
}
以便第一种方法可以根据给定的process
类型调用相应的storeIn
。
在Haskell中,我可以写
class Storeable c a where
store :: a -> c a -> c a
doQueryAndStoreData :: Storeable c a => ResultSet a -> c a -> c a
doQueryAndStoreData (ResultSet rs) coll = foldr store coll rs
并为我要存储Storeable
的任何集合类型提供ResultSet
个实例。
这是Java中的正确方法吗?因为我觉得我有点为了实现这个目标而奋斗。
答案 0 :(得分:2)
不幸的是,你无法做到这一点。编译器必须在编译时知道 此调用绑定的方法。如果要根据对象的运行时类型决定调用的方法,则必须手动检查它:
你能做的最多:
private class ProcessResultSet implements IProcessResultSet<? extends Collection> {
@Override
public void process(ResultSet rset, Collection storeIn) {
if (storeIn instanceof Set) {
return processSet(rset, (Set) storeIn);
} else if (storeIn instanceof List) {
return processList(rset, (List) storeIn);
} else {
throw new IllegalArgumentException("Unimplemented storage type");
}
}
public void processSet(ResultSet rset, Set storeIn) {
/* do something */
}
public void processList(ResultSet rset, List storeIn) {
/* do something else */
}
}
答案 1 :(得分:2)
不,Java不这样做。
您需要执行以下操作:
public <T> void doQueryAndStoreData(
String query,
T storeIn,
ResultSetProcessor<T> processor
) {
或者更有可能:
public void doQueryAndStoreData(
String query,
ResultSetHandler handler // may contain processor and storeIn
) {
我希望我不需要提到SQL注入漏洞是一件坏事。 (Map
不是Java中的Collection
(它在C#中,但C#Collection
不是很有用)。)
答案 2 :(得分:0)
private class ProcessResultSet implements IProcessResultSet<? extends Collection> {
public void process(ResultSet rset, Object storeIn)
{
if ( storeIn instanceof Map)
processMap(rset,(Map) storeIn);
else if (storeIn instanceof List)
processList(rset,(List) storeIn);
else
System.out.println("Unsupported input type.");
}
public void processMap(ResultSet rset, Map storeIn) {
/* do something */
}
public void processList(ResultSet rset, List storeIn) {
/* do something else */
}
}