Introduction

For one of my projects I needed to show the 30 last messages to the user. I was not interested in keeping these messages. So I decided to use a queue. The messages come from different sources and different threads.

And the different threads bit is important.

The queue

Here is what can happen if you use the not threadsafe queue.

Let’s take this little example.

Imports System.Threading

Module Module1
    Private _queue As Queue(Of String)
    Private _count As Integer

    Sub Main()
        Console.WriteLine("Not concurrent threads")
        _count = 0
        Dim t As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessage(x))))
        Dim t2 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessage(x))))
        Dim t3 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessage(x))))
        Dim t4 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessage(x))))
        t.Start()
        t2.Start()
        t3.Start()
        t4.Start()
        Do While _count < 4

        Loop
        _count = 1
        For Each q In _queue.Reverse.ToList
            Console.WriteLine(_count & ". " & q)
            _count += 1
        Next
        Console.ReadLine()
    End Sub

    Public Sub Addmessages(ByVal action As Action(Of String))
        For x = 1 To 1000000
            action.Invoke("test")
        Next
        _count += 1
    End Sub

    Public Sub AddMessage(message As String)
        If _Queue Is Nothing Then
            _queue = New Queue(Of String)(1000000)
        End If
        If Not String.IsNullOrEmpty(message) Then
            _Queue.Enqueue(message)
        End If
        Dim result As String
        If _queue.Count > 10 Then
            _queue.Dequeue()
        End If
    End Sub

End Module```
That is 4 threads that add 1000000 messages to the queue at an alarming rate. And I only want to keep the last 10 messages in my queue.

The result. 

> 128683. test
  
> 128684. test
  
> 128685. test
  
> 128686. test
  
> 128687. test
  
> 128688. test
  
> 128689. test
  
> 128690.
  
> 128691. test
  
> 128692. test
  
> 128693. test
  
> 128694. test
  
> 128695. test
  
> 128696. test
  
> 128697. test
  
> 128698. test
  
> 128699. test
  
> 128700. test
  
> 128701. test
  
> 128702. test
  
> 128703. test
  
> 128704. test

Those are only the last ones. There are 128704 elements in our queue and some are empty. Not sure how I get the empty ones but I guess it must be somewhere between an enqueue and a dequeue. 

Anyway the results are not what we desire. 

And of course now I could start adding blocks here and there to make sure the enqueue and dequeue are atomic operations and what not but Since .Net 4.0 there are Concurent collections and they are threadsafe by design. Lucky us.

## The concurrent queue

We can simply change our code to this.

```vbnet
Imports System.Collections.Concurrent
Imports System.Threading

Module Module1
    Private _concurrentQueue As ConcurrentQueue(Of String)
    Private _count As Integer

    Sub Main()
        Console.WriteLine("concurrent threads")
        _count = 0
        Dim t5 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessageConcurrently(x))))
        Dim t6 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessageConcurrently(x))))
        Dim t7 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessageConcurrently(x))))
        Dim t8 As New Thread(New ParameterizedThreadStart(Sub() Addmessages(Sub(x) AddMessageConcurrently(x))))
        t5.Start()
        t6.Start()
        t7.Start()
        t8.Start()
        Do While _count < 4

        Loop
        _count = 1
        For Each q In _concurrentQueue.Reverse.ToList
            Console.WriteLine(_count & ". " & q)
            _count += 1
        Next
        Console.ReadLine()
    End Sub

    Public Sub Addmessages(ByVal action As Action(Of String))
        For x = 1 To 1000000
            action.Invoke("test")
        Next
        _count += 1
    End Sub

    Public Sub AddMessageConcurrently(message As String)
        If _concurrentQueue Is Nothing Then
            _concurrentQueue = New ConcurrentQueue(Of String)()
        End If
        If Not String.IsNullOrEmpty(message) Then
            _concurrentQueue.Enqueue(message)
        End If
        Dim result As String
        If _concurrentQueue.Count > 10 Then
            _concurrentQueue.TryDequeue(result)
        End If
    End Sub

End Module

And now the result is.

concurrent threads

  1. test
  1. test
  1. test
  1. test
  1. test
  1. test
  1. test
  1. test
  1. test
  1. test

And that is exactly what I expected.

Conclusion

There is a namespace in .Net since 4.0 that contains threadsafe collections. And it is there for you to use. I would however not overuse it since I expect the blocking to have an effect on performance.