Condividi tramite


Maximize the use of CPU with parallel extensions (+ some WPF stuff)

Since this is my 40th post to this blog I decided to go back to square one… or post one actually :-) I’m going to create Windows Presentation Foundation (WPF) application that solves the Knight’s Tour puzzle. I actually didn’t know about this puzzle before I bought book called Puzzles for Programmers and Pros. That book had interesting puzzle that lead in to this post. So here we go!

I said that I’m going to create WPF application for my UI. You might ask why not the “good old” Windows Forms application...? Well for these simple reasons:

1.   I don’t like to write code to OnPaint / MainForm_Paint methods.
2. I wanted to define my user interface and then just say in code “hey knight go there” and it should just draw the UI with the knight in the correct position. But the defined UI must be also scalable.
3. WPF doesn’t have same barriers than Windows Forms does => It’s the face of future applications!

I don’t probably have to explain my reason #1 for you if you have experienced the same that I have :-) You’ll end up writing the UI code a lot and that’s not what you’re trying to do. You’re trying to solve puzzle and you are suddenly focusing for the UI code. That’s wrong approach. Therefore reason #2 goes hand-in-hand with #1.

So let’s look the the UI of the running application:

WPF UI

It’s the view of the classic chess board and I decided to take shortcut when creating the knight. I decided to use gray circle instead (or ellipse actually) :-) And for the layout management I just took the easy approach by using Grid and defining Columns and Rows. Here is the XAML for the UI:

 12345678910111213141516171819202122232425262728293031323334353637
 <Window x:Class="Window1"   xmlns="https://schemas.microsoft.com/winfx/2006/xaml/presentation"   xmlns:x="https://schemas.microsoft.com/winfx/2006/xaml"   Title="Knight's Tour" MinHeight="100" MinWidth="100"      Width="300" Height="300" Loaded="Window_Loaded">  <Grid x:Name="Board">        <Grid.ColumnDefinitions>      <ColumnDefinition />      <ColumnDefinition />      <ColumnDefinition />      <ColumnDefinition />      <ColumnDefinition />      <ColumnDefinition />      <ColumnDefinition />      <ColumnDefinition />    </Grid.ColumnDefinitions>    <Grid.RowDefinitions>      <RowDefinition />      <RowDefinition />      <RowDefinition />      <RowDefinition />      <RowDefinition />      <RowDefinition />      <RowDefinition />      <RowDefinition />    </Grid.RowDefinitions>    <Ellipse x:Name="Knight" Fill="Gray" Panel.ZIndex="1"         Grid.Column="{Binding Path=KnightX}"          Grid.Row="{Binding Path=KnightY}" />        <Rectangle x:Name="A8" Fill="White" />    <Rectangle x:Name="B8" Grid.Column="1" />    <!-- Etc... -->  </Grid></Window>

You probably noticed the interesting part of the XAML... and that’s the lines 30 and 31 where Binding is defined. It means that these values coming from the public properties of the DataContext. So let’s look at the code behind that XAML and let’s discuss the binding little bit more:

 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748
 using System;using System.ComponentModel;using System.Diagnostics;using System.Windows;using System.Windows.Media;using System.Windows.Shapes;public partial class Window1 : Window, INotifyPropertyChanged{  private int knightX = 0;  public int KnightX  {    get { return knightX; }    set    {      knightX = value;      NotifyPropertyChanged("KnightX");    }  }  private int knightY = 0;  public int KnightY  {    get { return knightY; }    set    {      knightY = value;      NotifyPropertyChanged("KnightY");    }  }  private void Window_Loaded(object sender, RoutedEventArgs e)  {    Knight.DataContext = this;    // Now moving the Knight is easy!    KnightX = 3;    KnightY = 3;  }  public event PropertyChangedEventHandler PropertyChanged;  private void NotifyPropertyChanged(String info)  {    if (PropertyChanged != null)    {      PropertyChanged(this, new PropertyChangedEventArgs(info));    }  }  //...

You might already noticed that my window also implements INotyfyPropertyChanged interface. And my two properties actually call NotifyPropertyChanged method when they are changed. So what’s this all about? Well if you don’t do this your values will be updated for the first time and after that they don’t actually get “bubbled” up to the ellipse anymore... unless you implement the notify mechanisms yourself. This is quite important and you should probably read more information about it on MSDN.

For the actual solving part I just used classic “old recursion” to solve the puzzle. And this is the part where we finally are going to the title of my post...

Classic one worker thread approach gives “fairly easy to implement but sub-optimal” solution. And you might ask why? And to answer this question I’m going so show you picture of task manager:

CPU1 

This picture was taken when my solver was running in “full speed ahead” –mode. And guess what... I’m not impressed! I’m actually just using single CPU (see the third box where green line has reached the roof)!!! So my worker thread approach is far from optimal resource usage.

Okay... What can I do then? I could do multiple threads and handle them manually but that’s again writing a lot of code that doesn’t have anything to do with the actual solving!? So if I would chosen Windows Forms + manual handling of multiple threads I would have a lot of code and just small fraction of that would actually do work that I was originally planning to do.

This is where Parallel Extensions(download)comes into the game! It’s additional library (System.Threading.dll) sitting on top of .NET Framework 3.5 and it’s currently in CTP phase. But I still highly recommend you to check it out if you want easily get more horse power to your algorithms.

I analyzed my code and noticed part where I could do things differently:

 1234567891011
 // My code was this:foreach (int location in startLocations){  // Calculations here!}// And I changed it to this:Parallel.ForEach<int>(startLocations, (location) =>{  // Calculations here!});

So I changed code in line 2 to be the code at line 8. What was the result at the task manager then:

CPU2 

Well I believe that I managed to get better use of the available horse power :-) It shows of course in the results:

“foreach”: ~50 solved solutions in ~5 minutes

”Parallel.ForEach”: ~450 solved solutions in < 5 minutes

To summarize... I just changed 1 line of code and I was able to get unbelievable results from it! So if you’re doing something similar then I recommend checking out the parallel extensions first before doing “own custom solution” for that.

I originally thought that I would go little bit deeper into the details of my solver but this post ended up too long even without it so maybe I’ll pass this time... But I’ll include video clip that shows the UI of the application when it’s solving.

Anyways... Happy hacking!

J

P.S. I’m also interested in F# and I’m probably going to do something fun with that too.

KnightsTour.wmv

Comments

  • Anonymous
    August 26, 2008
    PingBack from http://hoursfunnywallpaper.cn/?p=3163

  • Anonymous
    August 26, 2008
    The comment has been removed

  • Anonymous
    October 13, 2008
    Great article, thanks for the info!