Wednesday, July 29, 2009

.Net Application Performance Profiling, Performance metrics, Performance Monitoring and Optimization Part 1

Profiling, Performance metrics.

Application performance is a core element of any software's success. People spend all the time writing code and developing solutions but when it comes to performance, raising an eyebrow is the first expression that's is evident on every team members face.

Performance tuning, performance optimization, performance review, profiling are the jargon which people are wary about. Its psychological that any developer would not like to be the person who's code did the most amount of work and contributed extensively towards making the application slow, but an honest introspection is something that gives a definitive edge.


Wisdom in terms of software development comes by retrospection, code review and facts which establish that writing a certain piece of code has definitive measurable impact on performance.

Let me stop this verbal jugglery and try to show you some facts and figures which i came across while performance tuning a critical application.

I had the massive task of improving the start up time and performance of an application that had to be used in a critical situation. Errors, flaws, ignorance, exceptions, exclusions are strictly unacceptable. As a matter of fact they are Lethal to existence.



I started with the CLR profiler and got some startling facts.




Its bang on. It will show u your code's mettle. how much memory is allocated and how does it contribute to the application memory allocation.

a) Use RuntimeMethodHandle, RuntimeFieldHandle etc in place of MethodInfo. Method handles are lightweight, efficient and fast. Check out

b) Try to use BeginInvoke instead of Invoke. The difference being that one is synchronous call and the other is an async call.

c) Try to use Async calls. Eventing is a very good strategy. It may be a little confusing to follow in code but it establishes a decent async method call routine and is considered robust. Sequential flow must be broken down to such eventing mechanisms where after a step an event in raised and the second event would then commence. Beware of updating a User interface on a BeginInvoke method since in WPF only the main thread is the UI thread and any other thread updating the UI is considered a villain.

d) String operations and concatenations are killers. Replace them with string builders.

e) The foreach on a List is slower than the for loop with count. Try to use the FOR loop instead of the FOREACH loop for mission critical places where performance is needed. Etc startups etc.


d) Locking and multi threaded scenarios. For multi threaded scenarios where there are chances for a raise condition or for a deadlock condition we use the LOCK construct in C#. It has to be noted that Lock will slow the application since when two threads simultaneously try to access a resource there would be one thread which succeeds and the other will wait until the first thread leaves the lock. This can be tested in load testing scenarios for eg, try to write to a Hashtable or array with 200 users load on a load test. There will be exceptions that occur.

The solution. Don't use a lock for READING from a hashtable. Use the LOCK for Writing to a HashTable and release it as soon as possible.

eg:

INCORRECT USAGE

Hashtable hs = new Hashtable();
object lockObj = new object()
lock(lockObj)
{
//read the object from hashtable.
object readObject = hs["key"];

//Create new object and write to the hashtable.
hs.Add("writeKey","WriteVALUE");
}

CORRECT USAGE


//We wont use the lock for reading
object readObj = hs["key"]

//We will use the lock for writing.
lock(lockObj)
{
hs.Add("writeKey","WriteValue");
}


f) Remove un-necessary references from your project. Adding references which are not useful or not used anywhere will increase time in loading these assemblies on startup. Make sure that the reference are updated when u remove the "using ----" statement.

g) Remember that loading an assembly in memory is not = JITTIng assembly. Assembly JITTING is a costly affair and takes CPU cycles. Loading an assembly would just load it in memory but JIT is performed when any method in the loaded assembly is called.

h) Make sure that object creation is not costly and time consuming. What happens is that we clutter the constructor of a class with lots of code which have a catastrophic impact on the speed of an application. The application cannot create instance of the class or it may take time instantiating the class hence the application is now slow.


i) For WPF applications, Try not to make extra overheads and wrappers around in-build controls. They can tremendously slow your application. So try to use the controls provided in .net framework as much as possible and try to avoid un-necessary wrappers around them.


j) App domains are an interesting topic of discussion and they are extremely useful for application development. If you are creating an application, which would host other applications, or other assemblies, then it does make a lot of sense to use loader optimization. AppDomain creation is a costly process, assemblies are loaded into the newly created app-domain and the loading takes place in spite of some of the common assemblies already being loaded into the default appdomain. So use LoaderOptimization Attribute.

For WPF applications delete the file named App.XAML and App.Xaml.cs. Now use the following piece of code

public partial class App: Application
{


[STAThread]
[LoaderOptimization(LoaderOptimization.MultiDomainHost)]
static void Main()
{
App app = new App();
app.InitializeComponent();
app.Run();
}


}


k) Use tools like CLR Profiler and .Net Performance analysis wizard to analyze memory allocation. I have tried to use JetBrains DotTrace and found it to be quite good. You could get a trial for 30 days and could buy it if u like it,

l) Although these are some of the ways we could improve the performance of ,net applications it has to been seen that constant review, constant change, constant trials are the only way to improvise on it.


I wish to blog in detail about other aspects of application performance improvement and will do so in the next sessions. If you like or dislike this article, i would like to listen to your feedback and thoughts.

No comments: