¿Cómo puede System.out.format evitar un interbloqueo?

He encontrado que incluir una llamada a System.out.format en el clásicoTutorial de interbloqueo de Java evitará que se produzca un punto muerto, y no puedo entender por qué.

El siguiente código es el mismo que el del tutorial, con la adición demain deSystem.out.format("Hi, I'm %s...no deadlock for you!\n\n", alphonse.getName());

public class Deadlock {
    static class Friend {
        private final String name;

        public Friend(String name) {
            this.name = name;
        }

        public String getName() {
            return this.name;
        }

        public synchronized void bow(Friend bower) {
            System.out.format("%s: %s has bowed to me!\n",
                    this.name, bower.getName());
            bower.bowBack(this);
        }

        public synchronized void bowBack(Friend bower) {
            System.out.format("%s: %s has bowed back to me!\n",
                    this.name, bower.getName());
        }
    }

    public static void main(String[] args) throws InterruptedException {
        final Friend alphonse = new Friend("Alphonse");
        final Friend gaston = new Friend("Gaston");

        System.out.format("Hi, I'm %s...no deadlock for you!\n\n", alphonse.getName());

        new Thread(new Runnable() {
            public void run() { alphonse.bow(gaston); }
        }).start();

        new Thread(new Runnable() {
            public void run() { gaston.bow(alphonse); }
        }).start();
    }
}

Aquí está la salida:

Hi, I'm Alphonse...no deadlock for you!

Alphonse: Gaston has bowed to me!
Gaston: Alphonse has bowed back to me!
Gaston: Alphonse has bowed to me!
Alphonse: Gaston has bowed back to me!

Al eliminar los resultados de la línea ofensiva en el interbloqueo habitual:

Alphonse: Gaston has bowed to me!
Gaston: Alphonse has bowed to me!
... deadlock ...

¿La llamada a System.out.format de alguna manera está cambiando la forma en que los hilos adquieren los bloqueos intrínsecos en los objetos?

Actualizar:

Pude hacer que el sistema se bloquee de nuevo solo con cambiar donde comienzo los hilos en el código:

public static void main(String[] args) throws InterruptedException {
    final Friend alphonse = new Friend("Alphonse");
    final Friend gaston = new Friend("Gaston");

    System.out.format("Hi, I'm %s...no deadlock for you!\n\n", alphonse.getName());

    Thread t1 = new Thread(new Runnable() {
        public void run() { alphonse.bow(gaston); }
    });

    Thread t2 = new Thread(new Runnable() {
        public void run() { gaston.bow(alphonse); }
    });

    t1.start();
    t2.start();
}

Esto plantea la pregunta sobre cómo podemos obtener una mejor visión de cómo se comporta el programador de hilos, pero lo guardaré para un día diferente.

Respuestas a la pregunta(2)

Su respuesta a la pregunta