Można wykonywać różne rzeczy za pomocą metod JdbcTemplate
, które umożliwiają przekazanie PreparedStatementCreator
. Zawsze można użyć tego do przechwytywania wywołań (być może przy użyciu Proxy
), które spowodowały, że cancel
stało się w osobnym wątku przez jakieś cond
, które stało się .
public Results respondToUseRequest(Request req) {
final AtomicBoolean cond = new AtomicBoolean(false);
requestRegister.put(req, cond);
return jdbcTemplate.query(new PreparedStatementCreator() {
public PreparedStatement createPreparedStatement(Connection conn) {
PreparedStatement stmt = conn.prepareStatement();
return proxyPreparedStatement(stmt, cond);
}
},
new ResultSetExtractor() { ... });
}
Ten canceller
może zostać anulowany po pomyślnym zakończeniu; na przykład
private final static ScheduledExecutorService scheduler =
Executors.newSingleThreadedScheduledExecutor();
PreparedStatement proxyPreparedStatement(final PreparedStatement s, AtomicBoolean cond) {
//InvocationHandler delegates invocations to the underlying statement
//but intercepts a query
InvocationHandler h = new InvocationHandler() {
public Object invoke(Object proxy, Method m, Object[] args) {
if (m.getName().equals("executeQuery") {
Runnable cancel = new Runnable() {
public void run() {
try {
synchronized (cond) {
while (!cond.get()) cond.wait();
s.cancel();
}
} catch (InterruptedException e) { }
}
}
Future<?> f = scheduler.submit(cancel);
try {
return m.invoke(s, args);
} finally {
//cancel the canceller upon succesful completion
if (!f.isDone()) f.cancel(true); //will cause interrupt
}
}
else {
return m.invoke(s, args);
}
}
}
return (PreparedStatement) Proxy.newProxyInstance(
getClass().getClassLoader(),
new Class[]{PreparedStatement.class},
h);
więc teraz kodu, który reaguje na anulowanie użytkownika wyglądałby następująco:
cond.set(true);
synchronized (cond) { cond.notifyAll(); }
Ciekawym przykładem i dziwne, nie do końca pewien, w jaki sposób można zastosować do problemu pod ręką. Oczywiście, to 10-sekundowe auto-anulowanie wymagałoby zastąpienia czymś zewnętrznie wywołanym. – skaffman
Dlaczego musiałby być uruchamiany z zewnątrz? OP nie wspomniał nic o tym, że jest, powiedzmy, zdefiniowanym przez użytkownika –
OP tutaj: faktycznie, chcę, aby anulować zostanie wywołany w odpowiedzi na działanie użytkownika. –