Worksheet: J6 | CS 2113 Software Engineering - Spring 2025

Worksheet: J6

Submit a file called worksheet-J6.md to BB for this assignment.

Note

Attempt to answer these questions before running the code. This will improve your ability to analyize and reason about code without an IDE or compiler. This skill we be helpful on the exams.

Questions

  1. Compile the two files below and run the main method in the SimpleThreadExample.

    $ javac *.java
    $ java SimpleThreadExample
    
    /* SimpleThreadExample.java */
    public class SimpleThreadExample {
    
        public static void main(String[] args) {
        
            
            Messages msg = new Messages();
            Thread morningThread = new GreetingsThread(msg, 0);
            Thread afternoonThread = new GreetingsThread(msg, 1);
    
            morningThread.start();
            afternoonThread.start();
            //MARK
            
        }
    }
    
    /* Messages.java */
    public class Messages {
        public String morning;
        public String afternoon;
        public String evening;
        public String night;
    
        public Messages() {
          morning = "Good morning!";
          afternoon = "Good afternnon!";
          evening = "Good evening!";
          night = "Good night!";
        }
    
        public String getMessage(int choice) {
          switch(choice) {
             case 0:
               return this.morning;
             case 1:
               return this.afternoon;
             case 3:
               return this.evening;
             case 4:
               return this.night;
             default:
               return "";        
          }
        }
     }
    
    /* GreetingsThread.java */
    public class GreetingsThread extends Thread {
    
        private Messages msg;
        private int choice;
        public GreetingsThread(Messages msg, int choice) {
            this.msg = msg;
            this.choice = choice;
        }
    
        public void run() {
            System.out.println("Started the " + msg.getMessage(choice) + " thread...");
                
            //print 10 times
            for (int i = 0; i < 10; i++) {
                System.out.println(msg.getMessage(choice));
            }
    
            System.out.println("Exiting the " + msg.getMessage(choice) + " thread...");
        }
    }
    

    Then:

    • Run the program a few times. Describe the output of this program. Is it consistent?
    • Draw a memory diagram of the program at MARK

    Reveal Solution

  2. Now let’s modify GreetingsThread to add a Thread.sleep() in the run() method. Recompile the GreetingsThread class as below.

    Run the program multiple times. Does the output change in any way? Does one thread always finish first, or does the order change?

    /* GreetingsThread.java */
    public class GreetingsThread extends Thread {
        private Messages msg;
        private int choice;
    
        public GreetingsThread(Messages msg, int choice) {
            this.msg = msg;
            this.choice = choice;
        }
    
        public void run() {
            System.out.println("Started the " + msg.getMessage(choice) + " thread...");
    
            for (int i = 0; i < 10; i++) {
                System.out.println(msg.getMessage(choice));
    
                try {
                    // Sleep for 1 second (1000 milliseconds)
                    Thread.sleep(1000); 
                } catch (InterruptedException e) {
                    System.out.println("Interrupted while sleeping...");
                }
            }
    
            System.out.println("Exiting the " + msg.getMessage(choice) + " thread...");
        }
    }
    

    Reveal Solution

  3. Now let’s add a System.out.println() at the end of the main method. Recompile the program with this addition, continuing from above. Explain how it is possible that the main method is complete but the program is still producing output.

    /* SimpleThreadExample.java */
    public class SimpleThreadExample {
        public static void main(String[] args) {
                
            Messages msg = new Messages();
            Thread morningThread = new GreetingsThread(msg, 0);
            Thread afternoonThread = new GreetingsThread(msg, 1);
    
            morningThread.start();
            afternoonThread.start();
    
            System.out.println("Main method exiting...");
        }
    }
    

    Reveal Solution

  4. Finally let’s add a thread.join() to join the morningThread for 5 seconds before starting the afternoonThread. Recompile and rerun. Then describe the output of this program. Explain how attempting to join the first thread for 5 seconds affects the output of this program.

    /* SimpleThreadExample.java */
    public class SimpleThreadExample {
        public static void main(String[] args) {
            Messages msg = new Messages();
            Thread morningThread = new GreetingsThread(msg, 0);
            Thread afternoonThread = new GreetingsThread(msg, 1);
    
            morningThread.start();
    
            try {
                System.out.println("Joining the morning thread for 5 seconds...");
                morningThread.join(5000);
            } catch (InterruptedException e) {
                System.out.println("Interrupted while joining a thread...");
            }
    
            afternoonThread.start();
    
            System.out.println("Main method exiting...");
        }
    }
    

    Reveal Solution

  5. What does join() do compared to join(10)?

    Reveal Solution

  6. What are all the possible outputs of running the code below?

    import java.util.*;
    
    public class Ex0 {
      public static void main(String[] args) {
        Thread t = new Foo();
    
        t.start();
        int x = 5 * 5;
        System.out.println(x);
      }
    }
    
    public class Foo extends Thread {
      public void run() {
        int x = 7 * 7;
        System.out.println(x);
      }
    }
    

    Reveal Solution

  7. What is the output of the code below? Is it deterministic?

    import java.util.*;
    
    public class Ex1 {
      public static Random rand = new Random();
    
      public static void printSlow(String s, String t) {
        for (int i = 0; i < s.length(); i++) {
          try {
            Thread.sleep(rand.nextInt(1000));
          } catch (Exception e) {}
          System.out.println(t + s.charAt(i));
        }
      }
    
      public static void main(String[] args) {
        String s = "Mississippi";
        Thread t = new Foo(s, "   ");
    
        t.start();
        printSlow(s, " ");
      }
    }
    
    public class Foo extends Thread {
      private String msg, tab;
    
      public Foo(String s, String t) {
        this.msg = s;
        this.tab = t;
      }
    
      public void run() {
        Ex1.printSlow(this.msg, this.tab);
      }
    }
    

    Reveal Solution

  8. When adding into a linked list with the code below, why is a race condition possible here?

          tail.next = new Node(s, null);
          tail = tail.next;
    

    Reveal Solution

  9. How do you solve the problem above?

    Reveal Solution

  10. Why is the synchronized keyword necessary in the method below to avoid race conditions? The method is only one line of code, but the value is shared amongst multiple threads.

        public void increment() {
            value++;
    

    Reveal Solution