torsdag 18 oktober 2007

ASP.NET Cache gets flushed, cleared or randomly disappear

Have you been having problems with the ASP.NET cache gets cleared, flushed or simply seems to disappear when running your ASP.NET 2.0 web application? Restarting the IIS seems to help for a while but eventually the cache gets flushed again.

This problem seem to appear especially computers with 2GB of ram or less.

The solution is quite simple, and should be good enough on a developing machine. Simply add the following lines in web.config in the .



 1: <caching> 

 2: <cache disableMemoryCollection = "true"

 3: disableExpiration = "false" 

 4: privateBytesLimit = "0" 

 5: percentagePhysicalMemoryUsedLimit = "90" 

 6: privateBytesPollTime = "00:02:00"/> 

 7: </caching>





There are a few tools that might be helpful when debugging cache.

Reference:

Thanks to Henrik for good google skills.

tisdag 16 oktober 2007

SQLQueryStress

I want to recommend a good tool when you are optimizing your queries.

Often after a trace you will find that only a couple of your queries ends up taking the majority of the cpu.

Sometimes it's as easy to solve the problems by just adding an index, and sometimes there will be other solutions.

Whatever solution you choose, it can be a good thing to test your query.

The SQLQueryStress tool written by Adam Machanic, can be downloaded here http://www.datamanipulation.net/SQLQueryStress/.

SQLQueryStress is very simple although very powerful. You can use parameter substitution so you can execute your queries with different parameters. You can set the number of iterations but also the number of threads (clients).

The output from this is "cpu seconds", "elapsed time", "client seconds", "logical reads", and more.

So when you compare two queries and booth queries takes the same time to execute, you can easily see in this program if one of them use less cpu or maybe less IO over several executions and with different parameters.

måndag 15 oktober 2007

Are you still using the repeater?

When asp.net first was released we quickly adapted to an object oriented way of creating and manage controls, instead of the old procedural asp ways. We started to use literals, datagrid and repeaters.

All the good functionality that came with the data binding controls also introduced performance problems. So – for example - instead of using the datagrid we started to use the repeater. However it’s easy to forget that even the repeaters data binding functionality comes with performance problems.

The repeater uses reflection internally, and for each object (text or image or whatever) in the itemtemplate it will create a new control, and if you have a rather complex repeater it will quickly add up and create a lot of controls. The creation of a control and all it’s rendering means extra cpu time, and also extra work for GC.

I have made some performance tests and found out that the repeater can be 30 times slower than using string concatenation with stringbuilder and then use response.write to ouput the result. And this is only the rendering time, we also have saved a lot of CPU time and therefore the application will scale better when we stress test it.

The problem with this solution is that it will give us headache when we need to add/edit the html or add an attribute of a control or something. What we can do is to create a usercontrol that has a public collection (of some sort), that we can set, and then use inline c# with other custom controls. This solution will still give us the same performance benefits.

Example
<%
for (int i = 0; i < myCollection.Count; i++){
myCustomControl.myAttribute = myCollection[i].SomeProperty;
%>
<div class="<%=myCollection[i].myDivClassProperty %>">some html</div>
<uc:myUc id="myCustomControl" runat="server" />
<%
}
%>

This way we will keep the rest of our code clean and use this optimized version of data rendering where you have a lot of posts to render or is one of the most viewed pages in your application.

fredag 5 oktober 2007

ASP.NET Ajax History control example

This is a short example how to set up the History control from ASP.NET Futures and ASP.NET AJAX.

<cc1:History ID="History1" runat="server"></cc1:History>

<asp:UpdatePanel ID="updPnl" runat="server" UpdateMode="Always" ChildrenAsTriggers="true">

<ContentTemplate>

<asp:MultiView ID="multiView" runat="server">

<asp:View id="viewStep1" runat="server">

...

</asp:View>

<asp:View ID="viewStep1" runat="server">

...

</asp:View>

</asp:MultiView>

</ContentTemplate>

<Triggers>

<asp:AsyncPostBackTrigger ControlID="History1" EventName="Navigate" />

</Triggers>

</asp:UpdatePanel>

Event that is fired when you press the back button.

public void History1_Navigate(object sender, Microsoft.Web.Preview.UI.Controls.HistoryEventArgs args)

{

string viewId = "";

View historyView = viewStep1;

if (args.State.ContainsKey("BookingSteps"))

{

viewId = (string)args.State["BookingSteps"];

}

foreach (View v in multiView.Views)

{

if (v.ID == viewId)

historyView = v;

}

setActiveView(historyView);

}

Set history point each time you change the view.

private void setActiveView(View view)

{

multiView.SetActiveView(view);

History1.AddHistoryPoint("BookingSteps", view.ID);

}

onsdag 3 oktober 2007

List IIS App pool processes

A common issue on a shared IIS server is to distinguish the CPU load between different web application when looking in the task manager.

The solution is to create one application pool for each web application in the IIS. IIS will then launch one individual "w3wp.exe" process with its own ID (PID - Process ID) for each application pool.

To list all running processes, Windows Server has a built in vbs-script to display all running application pools (w3wp processes) located in: "c:\windows\system32\iisapp.vbs".

You run this script from a command console using:
c:\windows\system32\cscript.exe iisapp.vbs

The output is a list of each running w3wp process together with its PID which can be matched to the PID in the task manager.

Using one application pool for each web application also has advantages when looking at performance counters as each counter can be selected for each application pool.