Musisz czekać na asynchroniczne wywołanie api, zanim powrócę z metody w Javie

  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());
        }
    }
}

Ok, starałem się tutaj zawrzeć tylko minimalną ilość kodu. Istnieją trzy interesujące klasy, UserSession to wewnętrzny klej między programem obsługi Jetty a aplikacją QuickFix / j.

LogonHandler odbiera żądanie logowania HTTP i próbuje zalogować użytkownika do sesji aplikacji QuickFix / j.

QuickFix / j wysyła komunikat logowania do serwera FIX, to żądanie logowania / odpowiedź jest asynchroniczne. Żądanie logowania HTTP jest oczywiście synchroniczne. Musimy więc poczekać na odpowiedź z serwera FIX, zanim wrócimy z żądania HTTP. Robię to za pomocą CountDownLatch i tego obiektu UserSession.

Kiedy tworzę obiekt sesji QuickFix / j, tworzę również obiekt UserSession i dodaję go do mapy (co dzieje się w metodzie logowania LogonHandler).

Istnieją dwa wywołania zwrotne w obiekcie aplikacji QuickFix / j, toAdmin () i fromAdmin (). W fromAdmin () sprawdzam, czy komunikat jest odpowiedzią logowania, a jeśli tak, to wywołuję metodę UserSession do odliczania zatrzasku. Podczas debugowania kodu widzę, że metoda fromAdmin () jest trafiona, obiekt UserSession znajduje się na mapie i wywoływana jest metoda countdown (), a zatrzask.getCount () przechodzi od 1 do 0, ale zatrzask. ) Metoda w UserSession oczekuje () nigdy nie zwraca. Zawsze wygasa.

questionAnswers(1)

yourAnswerToTheQuestion