gpt4 book ai didi

java - 在我从 Java 中的方法返回之前需要等待异步 api 回调

转载 作者:IT老高 更新时间:2023-10-28 20:21:33 25 4
gpt4 key购买 nike

  import java.util.concurrent.CountDownLatch;

import quickfix.Initiator;


public class UserSession {
private final CountDownLatch latch = new CountDownLatch(1);

public String await() {
try {
System.out.println("waiting...");
if (latch.await(5, TimeUnit.SECONDS))
System.out.println("released!");
else
System.out.println("timed out");
return secret;
} catch (InterruptedException e) {
// TODO Auto-generated catch block
System.out.println(e.getMessage());
e.printStackTrace();
}
return null;
}

public void countdown(String s) {
System.out.println("In countdown: "+s+ ". Latch count: "+latch.getCount());
secret = s;
latch.countDown();
System.out.println("Latch count: "+latch.getCount());
}
}


public class LogonHandler extends AbstractHandler {

public void handle(String target, Request baseRequest, HttpServletRequest request, HttpServletResponse response)
throws IOException, ServletException
{
Map<String,String[]> query = request.getParameterMap();

if (query.containsKey("method")) {
if (query.get("method")[0].compareTo(method) == 0) {
baseRequest.setHandled(true);
response.getWriter().println(logon(query));
}
}
else
baseRequest.setHandled(false);
}

private String logon(Map<String,String[]> query) {
if (query.containsKey("username") && query.containsKey("password") && query.containsKey("sendercompid")) {

app.mapUser(query.get("sendercompid")[0], new UserSession(query.get("username")[0], query.get("password")[0]));

SessionID session = new SessionID(new BeginString("FIX.4.4"), new SenderCompID(query.get("sendercompid")[0]), new TargetCompID("PARFX"));

try {
ThreadedSocketInitiator tsi = new ThreadedSocketInitiator(app, app.getFileStoreFactory(), settings, app.getLogFactory(), app.getMessageFactory());
UserSession userSession = new UserSession(query.get("username")[0], query.get("password")[0]);
userSession.setInitiator(tsi);

tsi.start();
return userSession.await();
} catch (ConfigError e) {
// TODO Auto-generated catch block
e.printStackTrace();
return e.toString();
}
}
return "fail";
}
}


public class QuickfixjApplication implements Application {
private Map<String,UserSession> users = new HashMap<String,UserSession>();

public void mapUser(String s, UserSession u) {
users.put(s, u);
}

public void toAdmin(Message message, SessionID sessionId) {

try {
if (message.getHeader().getField(new StringField(MsgType.FIELD)).valueEquals(Logon.MSGTYPE)) {
UserSession user = users.get(sessionId.getSenderCompID());
message.setField(new Username(user.getUsername()));
message.setField(new Password(user.getPassword()));
}
} catch (FieldNotFound e) {
e.printStackTrace();
}
}

public void fromAdmin(Message message, SessionID sessionId)
throws FieldNotFound, IncorrectDataFormat, IncorrectTagValue, RejectLogon {

if (message.getHeader().getField(new StringField(MsgType.FIELD)).valueEquals(Logon.MSGTYPE)) {
System.out.println(message.toString());
UserSession user = users.get(sessionId.getSenderCompID());
user.countdown(message.toString());
}
}
}

好的,我已尝试在此处仅包含最少的代码。有三个有趣的类,UserSession 是 Jetty 处理程序和 QuickFix/j 应用程序之间的内部粘合剂。

LogonHandler 接收 HTTP 登录请求并尝试将用户登录到 QuickFix/j 应用程序 session 。

QuickFix/j 正在向 FIX 服务器发送登录消息,此登录请求/响应是异步的。 HTTP 登录请求当然是同步的。因此,我们必须等待 FIX 服务器的回复,然后才能从 HTTP 请求返回。我使用 CountDownLatch 和这个 UserSession 对象来做到这一点。

当我创建 QuickFix/j session 对象时,我还创建了一个 UserSession 对象并将其添加到 map 中(这发生在 LogonHandler 登录方法中)。

QuickFix/j 应用程序对象中有两个回调,toAdmin() 和 fromAdmin()。在 fromAdmin() 中,我检查消息是否是登录响应,如果是,我调用 UserSession 的方法来倒计时锁存器。在调试代码时,我看到 fromAdmin() 方法被命中,在 map 中找到 UserSession 对象并调用 countdown() 方法,并且 latch.getCount() 从 1 变为 0,但是 latch.await( ) UserSession await() 中的方法永远不会返回。它总是超时。

最佳答案

您可以像这样使用 CountDownLatch:

public class LogonHandler implements Handler {
private final CountDownLatch loginLatch = new CountDownLatch (1);

private boolean callbackResults;

public void serverResponseCallback(boolean result) {
callbackResults = result;
loginLatch.countDown ();
}

public boolean tryLogon(Credentials creds) throws InterruptedException {
SomeServer server = new SomeServer(address);
server.tryLogon (creds.getName (), creds.getPass ());
loginLatch.await ();
return callbackResults;
}
}

如果您想将等待时间限制为例如 5 秒,则使用以下代码代替 loginLatch.await ():

if (loginLatch.await (5L, TimeUnit.SECONDS))
return callbackResults;
else
return false; // Timeout exceeded

关于java - 在我从 Java 中的方法返回之前需要等待异步 api 回调,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14929322/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com