The Barber Shop Problem in Go

A similar solution was posted here. I wanted to look at this problem because it is a classic example of handling interprocess communication.

In this problem one models a stream of customers coming into a Barber Shop. If the waiting room, or lobby, is full, then they leave, otherwise they take a seat and wait for a haircut. The barber(s) take customers, cut their hair, and then look for the next customer.

In this code one channel is filled randomly with Customers; some of these will wait in the lobby, others will look for another barber. Within the BarberShop function, there is a barbers slice, and a channel named syncBarberChan. When the function runs, barbers are initialized and added to the slice. When a customer is selected and paired with a barber, the barber is taken out of the slice. At this point the haircut occurs for some period of time. When the haircut is over, the barber is added to the syncBarberChan. Next the syncBarberChan is drained for any idle barbers. If a customer is waiting, that newest barber is paired with a customer, otherwise the barber is added back to the slice of barbers.

In this way, the slice holds idle barbers, and the channel is used to signal that a barber is done with a given task.

package main

import (
    "fmt"
    "math/rand"
    "time"
)

const (
    NUM_BARBERS = 2
    LOBBY_SIZE = 3
    HAIRCUT_TIME = 20
)

func main() {
    customers := make(chan *Customer)
    go CustomerProducer(customers)
    go BarberShop(customers)
    time.Sleep(2 * time.Second)
}

type Barber struct {
    Id int
}

func (b Barber) String() string {
    return fmt.Sprintf("%d", b.Id)
}

type Customer struct {
    Id int
}

func (c Customer) String() string {
    return fmt.Sprintf("%d", c.Id)
}

func CustomerProducer(customers chan *Customer) {
    customerCounter := 1
    for {
        rnd := rand.Intn(10)
        time.Sleep(time.Duration(rnd) * time.Millisecond)
        customers <- &Customer{customerCounter}
        customerCounter += 1
    }
}

func CutHair(barber *Barber, customer *Customer, finished chan *Barber) {
    time.Sleep(HAIRCUT_TIME * time.Millisecond)
    fmt.Printf("Barber %s done with customer %s.\n", barber, customer)
    finished <- barber
}

func BarberShop(customers <-chan *Customer) {
    barbers := []*Barber{}
    lobby := []*Customer{}
    syncBarberChan := make(chan *Barber)

    for i := 0; i < NUM_BARBERS; i++ {
        barbers = append(barbers, &Barber{i+1})
    }

    for {
        select {
        // take a customer off of the customer channel
        case customer := <-customers:
            // if there are no barbers, either sit in the lobby or leave
            if len(barbers) == 0 {
                if len(lobby) < LOBBY_SIZE {
                    lobby = append(lobby, customer)
                    fmt.Printf("Customer %s seated in lobby.\n", customer)
                } else {
                    fmt.Printf("Lobby full, customer %s left.\n", customer)
                }
            } else {
                barber := barbers[0]
                barbers = barbers[1:]
                fmt.Printf("Customer %s goes to barber %s.\n", customer, barber)
                go CutHair(barber, customer, syncBarberChan)
            }
            fmt.Printf("Lobby: %+v\n", lobby)
        // take a barber off of the sync barber channel
        case barber := <-syncBarberChan:
            // if there are customers in the lobby, grab one, otherwise take a nap
            if len(lobby) > 0 {
                customer := lobby[0]
                lobby = lobby[1:]
                fmt.Printf("Customer %s goes to barber %s.\n", customer, barber)
                go CutHair(barber, customer, syncBarberChan)
            } else {
                barbers = append(barbers, barber)
                fmt.Printf("Barber %s idle.\n", barber)
            }
            fmt.Printf("Lobby: %+v\n", lobby)
        }
    }
}