<?xml version="1.0" encoding="UTF-8" ?>
<?xml-stylesheet type="text/xsl" href="http://www.aspnetzone.de/utility/FeedStylesheets/rss.xsl" media="screen"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/" xmlns:wfw="http://wellformedweb.org/CommentAPI/"><channel><title>Roberto's Blog</title><link>http://www.aspnetzone.de/blogs/robertobez/default.aspx</link><description /><dc:language /><generator>CommunityServer 2.1 SP2 (Build: 61120.2)</generator><item><title>MVC Html Helper á la Fluent Interface</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2010/06/08/mvc-html-helpers-fluent-interface.aspx</link><pubDate>Tue, 08 Jun 2010 21:45:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:217589</guid><dc:creator>Roberto</dc:creator><slash:comments>2</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/217589.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=217589</wfw:commentRss><description>&lt;img src="http://11249.2.sihosting.net/Blog/Images/MVCHtmlHelpers.jpg" alt="MVC Html Helpers" style="margin-bottom:5px;margin-top:5px;margin-right:10px;float:left;"&gt;Helferlein können das Leben ganz schön vereinfachen.&lt;br&gt;&lt;br&gt;Das zeigt sich an diversen Beispielen. Wer hat sich noch nicht eine LabeledTextbox oder ein Control in der Art in klassischen Webforms fabriziert, um sich damit eine Menge Tipparbeit zu sparen?&lt;br&gt;&lt;br&gt;Natürlich geht das auch mit &lt;a href="http://www.asp.net/mvc" target="_blank" title="ASP.NET MVC"&gt;MVC&lt;/a&gt;.&lt;br&gt;&lt;br&gt;Ist es nicht cool, wie Telerik und Co. Das machen?&lt;br&gt;&lt;br&gt;&lt;b&gt;&amp;lt;%= Html.MeineSuperAnwendung().MeinMegaControl().DasAllesKann %&amp;gt;&lt;/b&gt;&lt;br&gt;&lt;br&gt;Korrekt! Die eierlegende Wollmilchsau unter den eigenen Controls kann das!&lt;br&gt;&lt;br&gt;&lt;b&gt;Wie das geht?&lt;/b&gt;&lt;br&gt;&lt;br&gt;Ganz einfach: Man nehme eine statische Klasse, die das Control instanziert und zurückgibt.&lt;br&gt;&lt;br&gt;Die &lt;a href="http://en.wikipedia.org/wiki/Fluent_interface" target="_blank" title="Fluent interface"&gt;fluent Schreibweise&lt;/a&gt; ist nichts anderes als eine Methode, die das eigene Objekt zurückgibt.&lt;br&gt;&lt;br&gt;Gegeben eine Klasse Button mit folgender Methode:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; Button IchBinEinFluentDing()&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;;&lt;br /&gt;}&lt;br /&gt;var button &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; Button().IchBinEinFluentDing().IchBinEinFluentDing()&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Klingt nicht nur simple, ist es auch. Gut, nicht ganz, ein Zwischenschritt wird noch benötigt.&lt;br&gt;&lt;br&gt;Die „Container“ Klasse, die die einzelnen Controls zurückgeben kann. Somit wird nicht auf ein Controltyp beschränkt.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;class&lt;/span&gt; MyHtmlHelpers&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;private&lt;/span&gt; HtmlHelper _helper;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; MyHtmlHelpers(HtmlHelper helper)&lt;br /&gt;    {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._helper &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; helper;&lt;br /&gt;    }&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; CoolButton CoolButton(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; text)&lt;br /&gt;    {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; CoolButton(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._helper, text);&lt;br /&gt;    }&lt;br /&gt;    ...&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Wie bereits verraten, die &lt;b&gt;statische Klasse&lt;/b&gt; (Extension), die die Helpers zurück gibt:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;static&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;class&lt;/span&gt; MyProductExtension&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;static&lt;/span&gt; MyHtmlHelpers MyProduct(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt; HtmlHelper helper)&lt;br /&gt;    {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; MyHtmlHelpers (helper);&lt;br /&gt;    }&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Lange Rede – kurzer Sinn:&lt;br&gt;&lt;br&gt;&lt;b&gt;Html.MyProduct().CoolButton()&lt;/b&gt;&lt;br&gt;&lt;br&gt;Gut, die Schreibweise ist klar, aber das Ganze macht noch keinen Sinn. Der Button muss her!&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;class&lt;/span&gt; CoolButton&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;private&lt;/span&gt; IDictionary&amp;lt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;&amp;gt; _attributes;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;private&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; _text;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;private&lt;/span&gt; HtmlHelper _helper;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; CoolButton(HtmlHelper helper, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; text)&lt;br /&gt;    {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._text &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; text;&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._helper &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; helper;&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._attributes &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; Dictionary&amp;lt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;&amp;gt;();&lt;br /&gt;    }&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Ein Text für den Button, das html helper objekt - das später benötigt wird und das Kernstück – die Attribute. Das ganze Ding soll ja letztendlich flexibel sein.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; CoolButton SetAttribute(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; key, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; value)&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;if&lt;/span&gt; (!&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._attributes.ContainsKey(key))&lt;br /&gt;    {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._attributes.Add(key, value.ToString());&lt;br /&gt;    }&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Natürlich soll der Button auch navigieren können – Angaben á la MVC + Id:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; CoolButton RouteTo(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; controller, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; action, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; itemId)&lt;br /&gt;{&lt;br /&gt;    var urlHelper &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; UrlHelper(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._helper.ViewContext.RequestContext);&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; url;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;if&lt;/span&gt; (itemId == &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;null&lt;/span&gt;)&lt;br /&gt;    {&lt;br /&gt;        url &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; urlHelper.Action(action, controller);&lt;br /&gt;    }&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;else&lt;/span&gt;&lt;br /&gt;    {&lt;br /&gt;        url &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; urlHelper.Action(action, controller, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; { id &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; itemId });&lt;br /&gt;    }&lt;br /&gt;            &lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._attributes.Add(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"onclick"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;.Format(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"window.location='{0}';return false;"&lt;/span&gt;, url));&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Natürlich macht es nicht immer Sinn ein onclick event zu haben, aber es soll ja letztendlich der Sinn der Attribute erklärt werden.&lt;br&gt;&lt;br&gt;Es können weitere solcher Methoden folgen, bspw. Um die css Klasse oder ein client event zu setzen.&lt;br&gt;&lt;br&gt;Last but not least – &lt;b&gt;Rendern&lt;/b&gt;. &lt;br&gt;Aufgrund des Attribute Dictionarys recht einfach: Stück für Stück zusammensetzen.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; Render()&lt;br /&gt;{&lt;br /&gt;    var attributeBuilder &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; StringBuilder();&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;foreach&lt;/span&gt; (KeyValuePair&amp;lt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;&amp;gt; attribute &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;in&lt;/span&gt; _attributes)&lt;br /&gt;    {&lt;br /&gt;        attributeBuilder.AppendFormat(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;" {0}=\"{1}\""&lt;/span&gt;, attribute.Key, attribute.Value);&lt;br /&gt;    }&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;.Format(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"&amp;lt;button {0}&amp;gt;{1}&amp;lt;/button&amp;gt;"&lt;/span&gt;, attributeBuilder.ToString(), &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;._text);&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Für jedes KeyValuePair des Dictionarys wird folgendes Konstrukt erstellt: Key=“Value“, onclick=“TuWas();“&lt;br&gt;&lt;br&gt;Das Ganze sieht dann in etwa so aus:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;%= Html.MyProduct().CoolButton(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Edit"&lt;/span&gt;)&lt;br /&gt;                        .SetAttribute(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"class"&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"editButton"&lt;/span&gt;)&lt;br /&gt;                        .RouteTo(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"User"&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Edit"&lt;/span&gt;, Model.User.Id)&lt;br /&gt;                        .Render() %&amp;gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Man kann darüber streiten, ob eine Fluent Schreibweise wie eben beschrieben Sinn macht, ob sie sauber ist, oder ob sie nur ein Workaround ist, bis etwas Neues Erfunden wird?&lt;br&gt;&lt;br&gt;In ASP.NET Webforms Zeiten hab ich mir teilweise meine Controls per Render Überschreibung mit dem Writer selbst erzeugt (Fragt jetzt nicht warum – ich wollte einfach die Kontrolle über das Ding haben ;-) ). Das ist ja ebenfalls suboptimal und im Gegensatz dazu sind HTML Helpers ein Genuss :-)&lt;br&gt;&lt;br&gt;Anbei gibt’s wie immer ein etwas ausführlicheres Beispiel.&lt;br&gt;&lt;br&gt;
&lt;div class="DownloadBox"&gt;
&lt;a href="http://aspnetzone.de/files/folders/217590/download.aspx" title="ASP.NET MVC Html Helpers sample"&gt;ASP.NET MVC Html Helpers sample&lt;/a&gt;
&lt;/div&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=217589" width="1" height="1"&gt;</description><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Asp.Net/default.aspx">Asp.Net</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/mvc/default.aspx">mvc</category></item><item><title>Basta! Spring 2010</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2010/03/01/basta-spring-2010.aspx</link><pubDate>Mon, 01 Mar 2010 15:37:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:216260</guid><dc:creator>Roberto</dc:creator><slash:comments>0</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/216260.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=216260</wfw:commentRss><description>&lt;img style="margin-right:5px;margin-bottom:5px;margin-top:5px;float:left;" alt="Basta Spring 2010" src="http://11249.2.sihosting.net/Blog/Images/Basta.gif"&gt;Ich habe es dieses Jahr trotz der weiten Entfernung endlich auf die &lt;a href="http://it-republik.de/dotnet/basta10spring/" target="_blank" title="Basta Spring 2010"&gt;Basta! Spring&lt;/a&gt; geschafft.&lt;br&gt;In der Zeit der Hauptkonferenz vom 23 - 25 Februar gab es über 100 verschiedene Sessions aus denen man frei wählen konnte.&lt;br&gt;&lt;br&gt;Leider war es nicht möglich auf jeder gleichzeitig zu sein - trotzdem haben sich alle besuchten Sessions gelohnt!&lt;br&gt;&lt;br&gt;Es waren viele bekannte Leute bzw. Sprecher dort um Vorträge zu den unterschiedlichsten Themen im .Net Bereich zu halten.&lt;br&gt;&lt;br&gt;Einige besonders interessante Themen waren funktionale Sprachen wie F#, Software Architektur, OR Mapper wie NHibernate oder Entity Framework 4, TDD und BDD, Silverlight 4, WIF, Paralellisierung und viele weitere mehr.&lt;br&gt;&lt;br&gt;Wer etwas verpasst hat oder mehr dazu wissen möchte, kann gerne auf &lt;a title="Jürgen Gutsch" target="_blank" href="http://www.aspnetzone.de/blogs/juergengutsch/archive/tags/BASTA_2100_/default.aspx"&gt;Jürgens Blog&lt;/a&gt; vorbeischauen. Trotz der knappen Zeit ist es ihm gelungen zu jeder Session einen Blog Eintrag zu schreiben.&lt;br&gt;&lt;br&gt;Die Veranstaltung fand im &lt;a href="http://www.maritim.de/de/hotels/deutschland/rhein-main-hotel-darmstadt"&gt;Maritim Rhein Hotel&lt;/a&gt; in Darmstadt statt.&lt;br&gt;&lt;br&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/MaritimHotel.jpg"&gt;&lt;br&gt;&lt;br&gt;Für Verpflegung wurde ausreichend gesorgt - sowohl gutes Essen als auch ein Bierchen am Abend zur letzten "Night Session" waren dabei ;-)&lt;br&gt;&lt;br&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Basta1.jpg" alt="Basta Konferenz Saal" style="margin:3px;float:left;"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Basta2.jpg" alt="Basta Konferenz Saal" style="margin:3px;"&gt;&lt;br&gt;&lt;br&gt;Leider war, wie Jürgen schon erwähnt hat, die Abschluss Veranstaltung etwas zu knapp. Es waren nicht mehr viele Teilnehmer dort.&lt;br&gt;Zum Abschluss haben &lt;a href="http://www.aspnetzone.de/blogs/juergengutsch/"&gt;Jürgen&lt;/a&gt;, &lt;a href="http://www.aspnetzone.de/blogs/peterbucher/"&gt;Peter&lt;/a&gt;, &lt;a href="http://webenliven-space.de/dotnetblog/"&gt;Gregor&lt;/a&gt;, Robert und ich nochmals beim Bierchen die Basta revue passiert.&lt;br&gt;&lt;br&gt;Im Großen und Ganzen gesehen war die Teilnahme auf jeden Fall Wert.&lt;br&gt;Dabei habe ich viele interessante Leute kennen gelernt und klarerweiße auch viele neue Inputs geholt.&lt;br&gt;&lt;br&gt;Jetzt wird erst einige Zeit vergehen, bis man das gesammelte Material auch verarbeiten und umsetzten kann.&lt;br&gt;&lt;br&gt;Wenn die Möglichkeit besteht, war es sicher nicht meine letzte Basta!&lt;br&gt;&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=216260" width="1" height="1"&gt;</description><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Basta/default.aspx">Basta</category></item><item><title>Abstraktion</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2010/03/01/Software-Abstraktion.aspx</link><pubDate>Mon, 01 Mar 2010 08:37:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:216240</guid><dc:creator>Roberto</dc:creator><slash:comments>1</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/216240.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=216240</wfw:commentRss><description>&lt;img alt="Abstraktion" src="http://11249.2.sihosting.net/Blog/Images/Abstraktion.jpg" style="margin-right:5px;margin-bottom:5px;margin-top:5px;float:left;"&gt;Am 13. Oktober 2008 haben &lt;a href="http://www.peterbucher.ch/" target="_blank" title="Peter Bucher"&gt;Peter Bucher&lt;/a&gt; und &lt;a target="_blank" title="Golo Roden" href="http://www.goloroden.de/"&gt;Golo Roden&lt;/a&gt; unter dem Titel "Noch Fragen, Bucher? Ja, Roden!" angekündigt, jeweils zum ersten eines jeden Monats einen Kommentar zu einem vorab gemeinsam gewählten Thema verfassen zu wollen. Heute, am 1. März 2010, ist es nun wieder so weit, und das Thema für diesen Monat lautet:&lt;br&gt;&lt;br&gt;&lt;b&gt;Abstraktion&lt;/b&gt;&lt;br&gt;&lt;br&gt;Die beiden haben mich netterweise gefragt ob ich dieses Mal als "Gast-Mitstreiter" dabei bin, und so haben wir drei uns unabhängig voneinander im Vorfeld unsere Gedanken gemacht, wie wir diesem Thema gegenüberstehen. &lt;a target="_blank" href="http://www.aspnetzone.de/blogs/peterbucher/" title="Peter Buchers Blog"&gt;Peters&lt;/a&gt; und &lt;a target="_blank" href="http://www.des-eisbaeren-blog.de/" title="Golo Roden Blog"&gt;Golos&lt;/a&gt; Kommentare finden sich zeitgleich in ihren Blogs, folgend nun meine Meinung zu diesem Thema:&lt;br&gt;&lt;br&gt;Das Prinzip der &lt;a target="_blank" href="http://en.wikipedia.org/wiki/Abstraction_%28computer_science%29" title="Abstraktion"&gt;Abstraktion&lt;/a&gt; ist im Grunde nichts anderes als das Weglassen von Einzelheiten und das Überführen auf etwas Allgemeines oder Einfacheres (Generalisieren). So jedenfalls die Definition.&lt;br&gt;&lt;br&gt;Auch in der &lt;a target="_blank" href="http://de.wikipedia.org/wiki/Objektorientierte_Programmierung"&gt;Objektorientierten Programmierung&lt;/a&gt; findet dieses Prinzip Einsatz. Durch die Abstraktion von Aktionen oder Eigenschaften wird versucht, Redundanzen zu vermeiden.&lt;br&gt;&lt;br&gt;Außerdem vereinfacht uns Abstraktion das Leben durch Verringerung der Komplexität.&lt;br&gt;&lt;br&gt;Beispiele für Abstraktion&lt;br&gt;&lt;ul&gt;&lt;li&gt;Auslagerung von Codezeilen in eine benannte (zentralere) Funktion&lt;/li&gt;&lt;li&gt;Einführung von generischen Klassen&lt;/li&gt;&lt;li&gt;Unterklassen&lt;/li&gt;&lt;li&gt;Interfaces&lt;/li&gt;&lt;li&gt;Vererbung&lt;/li&gt;&lt;li&gt;Modularisierung der Anwendung&lt;/li&gt;&lt;li&gt;…&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;Grundsätzlich wird in der Software Programmierung zwischen zwei Arten der Abstraktion unterschieden:&lt;br&gt;&lt;br&gt;&lt;b&gt;Funktionale Abstraktion (Control abstraction)&lt;/b&gt;&lt;br&gt;&lt;br&gt;In der funktionalen Abstraktion wird eine Anwendung an Hand von funktionalen Merkmalen in individuelle Komponenten aufgebrochen, wobei die einzelnen Komponenten jeweils eine logisch-semantisch zusammengehörige Einheit bilden.&lt;br&gt;&lt;br&gt;Bspw. so eine drei Schichten Architektur&lt;br&gt;&lt;br&gt;&lt;img alt="Funktionale Abstraktion" src="http://11249.2.sihosting.net/Blog/Images/FunktionaleAbstraktion.jpg"&gt;&lt;br&gt;&lt;br&gt;Jede der einzelnen Schichten erfüllt eine bestimmte Funktion und auf diese sollte sie sich beschränken. Wie das intern abgehandelt wird, interessiert einzig und allein diese selbst.&lt;br&gt;&lt;br&gt;Die Grafik zeigt die Präsentationsschicht (bspw. Website), die auf die Business Logik zugreift um bspw. eine Auflistung aller Kunden zu bekommen. Wenn ein Entwickler das grafische UI programmiert, sollte er die Möglichkeiten der Business Schicht verstehen, aber nicht deren internen Aufbau.&lt;br&gt;&lt;br&gt;Die Business Schicht wiederum dient als Schnittstelle zur Datenschicht. Sie beinhaltet lediglich Services, die die gewünschten Daten aus der Datenschicht holen und zur Verfügung stellen. In welcher Art und Weiße die Daten geholt werden, ist, wie weiter oben bereits erwähnt, der Präsentation egal.&lt;br&gt;&lt;br&gt;Woher die Daten kommen, ob und welche Datenbank verwendet wird, welcher OR Mapper im Einsatz ist, … interessiert nur die Datenschicht.&lt;br&gt;Alle Schichten sind voneinander unabhängig. Dies senkt die Komplexität der Anwendung und vereinfacht das ändern zentraler Funktionalitäten.&lt;br&gt;&lt;br&gt;Wenn gesagt wird: Belästige mich nicht mit Details – wird versucht zu abstrahieren.&lt;br&gt;&lt;br&gt;&lt;b&gt;Datenabsraktion (Data abstraction)&lt;/b&gt;&lt;br&gt;&lt;br&gt;Das klassische Beispiel hierfür sind mehrere Entitäten&lt;br&gt;&lt;br&gt;&lt;img alt="Daten Abstraktion" src="http://11249.2.sihosting.net/Blog/Images/DataAbstraction.jpg"&gt;&lt;br&gt;&lt;br&gt;Jede dieser einzelnen Entitäten beinhaltet Standartfelder wie Id, Erstell- und Änderungsdatum.&lt;br&gt;&lt;br&gt;Es wäre suboptimal diese getrennt in den einzelnen Entitäten einzufügen. Viel besser ist eine etwas mehr zentralere Klasse (Entity), die bestimmte, für alle gemeinsame Eigenschaften beinhaltet um&amp;nbsp; somit ein wenig Redundanz zu vermeiden.&lt;br&gt;&lt;br&gt;&lt;i&gt;In C# werden mit dem „abstract“- Modifizier(in vb.Net MustInherit) zentrale Klassen erstellt, sprich sie dienen nur als Basisklassen und müssen implementiert werden.&lt;/i&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Wie viel Abstraktion ist zu viel Abstraktion?&lt;/b&gt;&lt;br&gt;&lt;br&gt;Je detailreicher die Strukturen bzw. Modelle werden (je weniger Abstraktion sie verwenden), desto besser ist die Akzeptanz. Je weniger Details die einzelnen Modelle der Anwendung beinhalten (je mehr Abstraktion sie verwenden), desto größer ist die Flexibilität und Wiederverwendbarkeit.&lt;br&gt;&lt;br&gt;Das Abstraktionsprinzip besagt, Abstraktion sollte dann verwendet werden, wenn damit Redundanzen (üblich im Code) vermieden werden können.&lt;br&gt;Die Softwareentwicklung strebt ja bekanntlich nach immer höheren Stufen der Abstraktion. Aber mehr ist nicht immer besser. Wer ganze Stockwerke aus Abstraktionsebenen baut, vergisst, dass diese nicht nur von Maschinen betreten werden.&lt;br&gt;&lt;br&gt;Mehrere Abstraktionen bedeuten meist einen höheren Aufwand und sind außerdem schwieriger zu verstehen.&lt;br&gt;&lt;br&gt;Zu Beginn eines Projektes sind Abstraktionen teilweise schnell implementiert, allerdings müssen sich diese erst bewähren: &lt;br&gt;Wird in einem späteren Projektzyklus klar, dass eine Abstraktion nicht korrekt ist, können sehr hohe Änderungskosten anfallen – ein Alptraum für jeden Entwickler.&lt;br&gt;&lt;br&gt;Um ein neues Problem zu lösen gerät mancher Entwickler in die Versuchung eine zusätzliche Abstraktionsschicht einzubauen. Abstraktionen an sich sind nicht das Problem, sondern vielmehr die (zu komplexe oder falsche) Denkweise des Entwicklers, der sich damit zusätzliche Probleme einfängt.&lt;br&gt;&lt;br&gt;&lt;u&gt;Ein Beispiel dazu:&lt;/u&gt;&lt;br&gt;&lt;br&gt;Zwei Entitäten Kunden und Benutzer sind gegeben. Jetzt fällt einem Entwickler ein, dass beides Personen sind und beide das Feld Name haben. Es ist eine Basisklasse „Person“ gegeben und das „Name“ Feld wird dorthin verschoben.&lt;br&gt;Jetzt kommt ein weiterer Entwickler, der der Meinung ist, dass das Feld „Name“ bei bestimmten Kunden eigentlich die Kundennummer ist und abstrahiert das Feld zu „Kundennummer“.&lt;br&gt;Was zurück bleibt ist eine nicht mehr klare Bedeutung der einzelnen Felder.&lt;br&gt;&lt;br&gt;&lt;b&gt;Mein Fazit&lt;/b&gt;&lt;br&gt;&lt;br&gt;Wie bei so vielem im Leben ist auch hier eine gesunde Mischung gefragt.&lt;br&gt;Abstraktion gehört zu den Urbausteinen der Softwareprogrammierung. Heute wäre es nahezu undenkbar eine Anwendung ohne zu entwickeln.&lt;br&gt;&lt;br&gt;Wie viel Abstraktion verwendet wird, hängt von vielen Faktoren ab. Unter anderem den Anforderungen, der Erfahrung des Teams, Erweiterbarkeit, etc. &lt;br&gt;Wenn bspw. von Anfang an vorgesehen wird, dass ein Kunde ein Verkäufer werden kann, sollte vielleicht auch vorgesehen werden, dass dieser zu einem Mars Mensch mutieren kann.&lt;br&gt;Klar, diese „neverending abstraction“ ist immer eine Frage des Aufwands.&lt;br&gt;&lt;br&gt;Ich kenne die Frage sehr wohl, was man wie in Zukunft machen könnte oder vorsehen sollte. In den meisten Fällen hat sich herausgestellt – Eierlegende Wollmilchsäue sind nur selten die Beste Lösung. Abstraktion ist dafür gedacht, die Komplexität einer Anwendung zu verringern und nicht in den Sternen zu treiben.&lt;br&gt;&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=216240" width="1" height="1"&gt;</description><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Asp.Net/default.aspx">Asp.Net</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Abstraktion/default.aspx">Abstraktion</category></item><item><title>DI / IoC Container LightCore Teil 3: Registrierung von Generics</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-generics.aspx</link><pubDate>Sun, 21 Feb 2010 17:18:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:216097</guid><dc:creator>Roberto</dc:creator><slash:comments>2</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/216097.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=216097</wfw:commentRss><description>&lt;img alt="LightCore Generics Registration" src="http://11249.2.sihosting.net/Blog/Images/LightCoreGeneric.gif" style="margin-right:5px;margin-bottom:5px;margin-top:5px;float:left;"&gt;Im dritten Teil der Post Serie wird die bisher angewandte drei Schichten Struktur etwas erweitert, sodass diese durch das hinzufügen von generischen Klassen etwas mehr flexibel wird.&lt;br&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/01/16/inversion-of-control-di-ioc-container-lightcore.aspx" title="LightCore - Einführung"&gt;1. LightCore – Einführung&lt;/a&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-xml-module.aspx" title="LightCore - Registrierung über Xml Module"&gt;2. LightCore – Registrierung über Xml Module&lt;/a&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-generics.aspx" title="LightCore - Registrierung von Generics"&gt;3. LightCore – Registrierung von Generics&lt;/a&gt;&lt;br&gt;&lt;br&gt;&lt;a href="http://msdn.microsoft.com/de-de/library/sz6zd40f.aspx"&gt;Generics&lt;/a&gt; sind nichts anderes als Klassen, bei denen der Nutzer die Datentypen, die der Typ verarbeiten soll, vorgeben kann.&lt;br&gt;&lt;br&gt;Es kann durchaus Sinn machen, bei Projekten die Repositories generisch zu halten, da sich diese immer auf dieselbe Datenquelle beziehen(bspw. bei Verwendung des NHibernate OR-Mappers) und sie sich lediglich von den angeforderten Typen unterscheiden.&lt;br&gt;&lt;br&gt;LightCore bietet &lt;u&gt;zwei Möglichkeiten&lt;/u&gt;, generische Typen zu registrieren.&lt;br&gt;&lt;br&gt;&lt;b&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Lock.png" style="float:left;margin-top:-7px;"&gt;&amp;nbsp; Closed Types&lt;/b&gt;&lt;br&gt;&lt;br&gt;Eine generische Klasse, die schon einen Typparameter hat. &lt;br&gt;IRepository&amp;lt;User&amp;gt; zu Repository&amp;lt;User&amp;gt; registrieren.&lt;br&gt;&lt;br&gt;Die Registrierung und Auflösung sieht in dem Fall so aus:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;builder.Register&amp;lt;IRepository&amp;lt;User&amp;gt;, Repository&amp;lt;User&amp;gt;();&lt;br /&gt;container.Resolve&amp;lt;IRepository&amp;lt;User&amp;gt;&amp;gt;();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Der &lt;u&gt;Nachteil&lt;/u&gt; dieser Variante liegt darin, dass jedes Repository getrennt registriert werden muss (eines für den User, eines für die UserGroup…). Dies wär in dem Fall suboptimal, da sie von den Funktionen her identisch sind.&lt;br&gt;&lt;br&gt;&lt;b&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Open.png" style="float:left;margin-top:-7px;"&gt;&amp;nbsp; Open Types&lt;/b&gt;&lt;br&gt;&lt;br&gt;Eine generische Klasse, die keinen bestimmten Parameter hat.&lt;br&gt;Der Kontrakt, in dem Fall IRepository&amp;lt;&amp;gt; hätte die Implementation Repository&amp;lt;&amp;gt;.&lt;br&gt;&lt;br&gt;Der Vorteil dabei ist die Auflösung (on-the-fly) auf einem beliebigen Typparameter.&lt;br&gt;So wäre IRepository&amp;lt;User&amp;gt; und IRepository&amp;lt;UserGroup&amp;gt;.&lt;br&gt;&lt;br&gt;In dem Fall muss das Repository nur &lt;u&gt;ein einziges Mal&lt;/u&gt; registrieren werden und kann mit beliebigem Typ auflöst werden.&lt;br&gt;&lt;br&gt;Wenn das Projekt ein generisches Repository mit viele verschiedene Typen dazu hat, ist der open type die erste Wahl.&lt;br&gt;&lt;br&gt;Als Beispiel wird wieder eine Solution mit einer klassischen drei Schichten Architektur verwendet.&lt;br&gt;&lt;br&gt;Um das Ganze noch etwas zu vertiefen wird eine vierte Schicht „Core“ hinzugefügt. Diese beinhaltet die zentralen Interfaces wie bspw. IRepository oder IService.&lt;br&gt;&lt;br&gt;&lt;img alt="LightCore Generics Registration Project Tree" src="http://11249.2.sihosting.net/Blog/Images/LightCoreDemoGenerics.jpg"&gt;&lt;br&gt;&lt;br&gt;Das Repository muss wie oben beschrieben nur einmal registriert werden:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;container.Register(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;typeof&lt;/span&gt;(IRepository&amp;lt;&amp;gt;), &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;typeof&lt;/span&gt;(Repository&amp;lt;&amp;gt;));&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Jetzt könnten alle gewünschten Typen (Im Beispiel user und usergroup) instanziiert werden:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;container.resolve&amp;lt;IRepository&amp;lt;User&amp;gt;&amp;gt;();&lt;br /&gt;container.resolve&amp;lt;IRepository&amp;lt;UserGroup&amp;gt;&amp;gt;();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Fazit:&lt;/b&gt;&lt;br&gt;&lt;br&gt;Durch die Unterstützung von generischen Klassen ist Lightcore für mich interessant gewordern und ist bereits in einer Anwendung im Einsatz. Es funktioniert bis dato Problemfrei und ohne merkbare Performance Verluste. Es bietet genau das, was ich zur Zeit benötige. Somit bin ich gespannt auf dessen zukünftige Erweiterungen.&lt;br&gt;&lt;br&gt;Anbei wie immer das Beispiel nicht vergessen! :-)&lt;br&gt;&lt;br&gt;
&lt;div class="DownloadBox"&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216101/download.aspx" title="Download LightCore Demo: Einführung"&gt;LightCore Demo: Einführung&lt;/a&gt;&lt;br&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216102/download.aspx" title="Download LightCore Demo: Registrierung über Xml Modul"&gt;LightCore Demo: Registerirung über Xml Module&lt;/a&gt;&lt;br&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216103/download.aspx" title="Download LightCore Demo: Registrierung von Generics"&gt;LightCore Demo: Registrierung von Generics&lt;/a&gt;&lt;br&gt;
&lt;/div&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=216097" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/216097.ashx" length="63987" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Asp.Net/default.aspx">Asp.Net</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/LightCore/default.aspx">LightCore</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/IoC/default.aspx">IoC</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Generics/default.aspx">Generics</category></item><item><title>DI / IoC Container LightCore Teil 2: Registrierung über Xml Modul</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-xml-module.aspx</link><pubDate>Sun, 21 Feb 2010 16:02:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:216096</guid><dc:creator>Roberto</dc:creator><slash:comments>5</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/216096.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=216096</wfw:commentRss><description>&lt;img alt="LightCore Xml Registration module" src="http://11249.2.sihosting.net/Blog/Images/LightCoreXml.gif" style="margin-right:5px;margin-bottom:5px;margin-top:5px;float:left;"&gt;Wie bereits im &lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/01/16/inversion-of-control-di-ioc-container-lightcore.aspx"&gt;ersten Teil&lt;/a&gt; erwähnt, wird es einer mehr Schichten Architektur immer wichtiger, Abhängigkeiten zu minimieren. Dabei einen ersten Schritt zu setzen, ist die Implementierung eines IoC Containers. Dieser hilft dabei, den Code in den diversen Projekten unabhängiger zu halten.&lt;br&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/01/16/inversion-of-control-di-ioc-container-lightcore.aspx" title="LightCore - Einführung"&gt;1. LightCore – Einführung&lt;/a&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-xml-module.aspx" title="LightCore - Registrierung über Xml Module"&gt;2. LightCore – Registrierung über Xml Module&lt;/a&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-generics.aspx" title="LightCore - Registrierung von Generics"&gt;3. LightCore – Registrierung von Generics&lt;/a&gt;&lt;br&gt;&lt;br&gt;Mich persönlich hat &lt;a href="http://www.aspnetzone.de/blogs/peterbucher/" title="Peter Bucher" target="_blank"&gt;Peter Bucher&lt;/a&gt; mit seinem neusten Projekt &lt;a target="_blank" href="http://lightcore.peterbucher.ch/"&gt;LightCore&lt;/a&gt; darauf aufmerksam gemacht. Anfangs war dies nur als Unit Testing Spielwiese gedacht, hat sich aber zu einem konkurrenzfähigen Micro Kernel entwickelt. Dessen weitere Entwicklung wird in Zukunft per Community gesteuert. Der Source Code wurde veröffentlicht und ist auf jeden Fall einen Blick Wert! (Stichwort: Codequalität)&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;Zurück zum Thema. Viele werden sich denken, was bringt mir das Ganze eigentlich?&lt;br&gt;&lt;ul&gt;&lt;li&gt;Abhängigkeiten verringern und Kontrolle der Abhängigkeiten (Interface / Basisklasse - also Abstraktion) und dadurch auf die Instanziierung (Die sonst auf den konkreten Typen geht), rausziehen.&lt;/li&gt;&lt;li&gt;Dadurch bessere Testbarkeit und Aufbrechung der Anwendung in mehr Einzelteile, wodurch die Komplexität sinkt.&lt;/li&gt;&lt;li&gt;Außerdem globale Konfigurierbarkeit, was genutzt werden soll um Änderungen an vielen, unnötigen Stellen vorzunehmen.&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;Nun wird anhand eines Beispiels wird gezeigt, wie man Lightcore in einer drei Schichten Architektur einsetzen bzw. integrieren kann.&lt;br&gt;Gleichzeitig wird die Registrierung über das integrierte Xml Modul erklärt&lt;br&gt;&lt;br&gt;Was eine drei Schichten Architektur (Tier 3 layer) ist, erklärt &lt;a target="_blank" href="http://code-inside.de/blog/"&gt;Robert Mühsig&lt;/a&gt; in &lt;a target="_blank" href="http://code-inside.de/blog/2008/07/09/howto-3-tier-3-schichten-architektur/"&gt;seinem Post&lt;/a&gt;.&lt;br&gt;&lt;br&gt;Das folgende Beispiel wird eine Solution mit mehreren Projekten beinhalten, die eine Anwendung mit mehreren Schichten darstellt.&lt;br&gt;&lt;br&gt;Die Klassische Struktur sieht wie folgt aus:&lt;br&gt;&lt;ul&gt;&lt;li&gt;Data Access Layer&lt;/li&gt;&lt;li&gt;Business Layer&lt;/li&gt;&lt;li&gt;Presentation Layer&lt;/li&gt;&lt;/ul&gt;Folglich könnten die Projekte so benannt werden:&lt;br&gt;&lt;br&gt;&lt;b&gt;Data&lt;/b&gt;: beinhaltet eine &lt;u&gt;User&lt;/u&gt;&lt;b&gt; &lt;/b&gt;Klasse sowie das dazugehörige &lt;u&gt;Repository&lt;/u&gt;.&lt;br&gt;&lt;br&gt;&lt;b&gt;Service&lt;/b&gt;: beinhaltet einen &lt;u&gt;User Service&lt;/u&gt; mit dem
 Repository als Member, der per Constructor Injection injiziert wird.&lt;br&gt;&lt;br&gt;&lt;b&gt;Web&lt;/b&gt;:
 Die Präsentation&lt;br&gt;&lt;br&gt;&lt;img alt="LightCore Xml Registration Project Tree" src="http://11249.2.sihosting.net/Blog/Images/LightCoreDemoPartTwo.jpg"&gt;&lt;br&gt;&lt;br&gt;Im &lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/01/16/inversion-of-control-di-ioc-container-lightcore.aspx"&gt;ersten Teil des Posts&lt;/a&gt; wurde gezeigt, wie man Kontrakte einfach per code (global.asax) registrieren kann. Der erste Schritt wird sein, die Registrierung in die &lt;b&gt;Web/App.config&lt;/b&gt; bzw. einer eigenen Konfigurationsdatei auszulagern.&lt;br&gt;&lt;br&gt;Wer achte gibt sieht, dass beim &lt;a href="http://lightcore.peterbucher.ch/download.aspx"&gt;heruntergeladenen Paket&lt;/a&gt; unter Anderem eine XML Schema Datei (LightCore.xsd), sowie eine Beispiel Konfiguration (SampleConfiguration.xml) dabei ist.&lt;br&gt;Nun die Datei LightCore.xsd dem aktuellen Web Projekt hinzufügen.&lt;br&gt;&lt;br&gt;Es kann eine eigene Konfigurationsdatei erstellt werden, oder aber einfach den entsprechenden Block in die Web/App.config hinzufügen.&lt;br&gt;Wer sich für letzteres entscheidet, den Block, der innerhalb der SampleConfiguration.xml zu finden ist, einfach in den &amp;lt;configuration&amp;gt; Abschnitt packen.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;configuration&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;    …&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;LightCoreConfiguration&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;xmlns&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="clr-namespace:LightCore.Configuration;assembly=LightCore.Configuration"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Als erstes kommt die Einstellung über den genutzten Lebenszyklus, also ob Singleton, Transient oder HttpRequest.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;LightCoreConfiguration.TypeAliases&amp;gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;TypeAlias&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Alias&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="HttpRequest"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Type&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="LightCore.Integration.Web.HttpRequestLifecycle, LightCore.Integration.Web"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;br /&gt;&amp;lt;/LightCoreConfiguration.TypeAliases&amp;gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Jetzt die eigentliche &lt;b&gt;Registrierung&lt;/b&gt;.&lt;br&gt;Einfache Registrierungen ohne Gruppen kommen in die LightCoreConfiguration.Registrations Sektion:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;LightCoreConfiguration.Registrations&amp;gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Registration&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Name&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="MyRepository"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ContractType&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="MyAssembly.IUserRepository, MyAssembly"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ImplementationType&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="MyAssembly.Repository, MyAssembly"&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;br /&gt;&amp;lt;/LightCoreConfiguration.Registrations&amp;gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Sollte selbsterklärend sein, Namensangabe um den registrierten Kontrakt wiederzufinden, der Kontrakttyp sowie die Klasse mit jeweiliger Angabe des Assembly Namens. Optionale Argumente für den Konstruktor können über Arguments=“Value1, Value2“ mitgegeben werden. Wenn die LightCore.xsd im Projekt eingebunden wurde, sollte auch IntelliSense zur Verfügung stehen!&lt;br&gt;&lt;br&gt;Ist eine gruppierte Registrierung gewollt, um bspw. die Repositories zu sammeln, gibt es auch hierfür eine Sektion:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;LightCoreConfiguration.RegistrationGroups&amp;gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;RegistrationGroup&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Name&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="Repositories"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;        &amp;lt;RegistrationGroup.Registrations&amp;gt;&lt;br /&gt;            &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Registration&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Name&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="MyRepository"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ContractType&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="MyAssembly.IUserRepository, MyAssembly"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ImplementationType&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="MyAssembly.Repository, MyAssembly"&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;br /&gt;        &amp;lt;/RegistrationGroup.Registrations&amp;gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;/&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;RegistrationGroup&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;&amp;lt;/LightCoreConfiguration.RegistrationGroups&amp;gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Jetzt wo die Registrierung der Repositories gemacht wurde, muss der Container erstellt und das &lt;b&gt;Konfigurationsmodul&lt;/b&gt; zugewiesen werden.&lt;br&gt;&lt;br&gt;Dies sollte in der Globalen Anwendungsdatei (global.asax) in der Application_Start methode gemacht werden:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;protected&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;void&lt;/span&gt; Application_Start(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; sender, EventArgs e)&lt;br /&gt;{&lt;br /&gt;    var builder &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; ContainerBuilder();&lt;br /&gt;    var module &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; XamlRegistrationModule();&lt;br /&gt;    builder.RegisterModule(module);&lt;br /&gt;    _container &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; builder.Build();&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Wenn gewollt, kann die Konfigurationsdatei im Konstruktor des XamlRegistrationModules angegeben werden:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var module &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; XamlRegistrationModule(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"~/MyConfiguration.xml"&lt;/span&gt;)&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die Objekte können wie gehabt mit resolve aufgelöst werden&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var myRepository &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; _container.Resolve&amp;lt;IRepository&amp;gt;(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"MyRepository"&lt;/span&gt;);&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Fazit: Welche Variante ist nun besser?&lt;/b&gt;&lt;br&gt;&lt;br&gt;Registrierung per Code hat den Vorteil, dass weniger Fehler auftreten können.&lt;br&gt;Da es kompiliert werden muss, wird überprüft ob der Kontrakt implementiert ist oder nicht.&lt;br&gt;&lt;br&gt;Somit ist es intuitiver und Refactoring-sicher. Wenn eine Klasse umbenannt wird – wird sie auch hier übernommen.&lt;br&gt;&lt;br&gt;Allerdings hat die Registrierung per Code einen riesen Haken: Es ist eine harte – also feste – Referenz.&lt;br&gt;Wenn eine Assembly, aus welchem Grund auch immer, ausgetauscht wird, muss das komplette Projekt neu kompiliert werden.&lt;br&gt;&lt;br&gt;Sollte das bei einer Produktivumgebung der Fall sein, kann das ein großer Nachteil sein. Dadurch muss ein komplettes Deployment gemacht werden.&lt;br&gt;&lt;br&gt;Was also „besser“ ist, hängt vom Anwendungsfall ab.&lt;br&gt;Ich persönlich finde die „per Code“ Variante bei nicht allzu vielen Klassen übersichtlich.&lt;br&gt;Eine gut durchdachte Anwendungsstruktur von Anfang an ist natürlich Voraussetzung :-) &lt;br&gt;&lt;br&gt;Anbei gibt’s wie immer das oben gezeigte Beispiel als Download.&lt;br&gt;&lt;br&gt;
&lt;div class="DownloadBox"&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216101/download.aspx" title="Download LightCore Demo: Einführung"&gt;LightCore Demo: Einführung&lt;/a&gt;&lt;br&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216102/download.aspx" title="Download LightCore Demo: Registrierung über Xml Modul"&gt;LightCore Demo: Registerirung über Xml Module&lt;/a&gt;&lt;br&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216103/download.aspx" title="Download LightCore Demo: Registrierung von Generics"&gt;LightCore Demo: Registrierung von Generics&lt;/a&gt;&lt;br&gt;
&lt;/div&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=216096" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/216096.ashx" length="82873" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Asp.Net/default.aspx">Asp.Net</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/LightCore/default.aspx">LightCore</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/IoC/default.aspx">IoC</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Xml/default.aspx">Xml</category></item><item><title>DI / IoC Container LightCore Teil 1: Einführung</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2010/01/16/inversion-of-control-di-ioc-container-lightcore.aspx</link><pubDate>Sat, 16 Jan 2010 16:03:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:215449</guid><dc:creator>Roberto</dc:creator><slash:comments>4</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/215449.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=215449</wfw:commentRss><description>&lt;img src="http://11249.2.sihosting.net/Blog/Images/LightCore1.gif" style="margin-right:5px;margin-bottom:5px;margin-top:5px;float:left;" alt="LightCore DI/Ioc Container"&gt;In einer mehrschichtigen Architektur (Oft verwendet die&lt;a title="How to tier 3 Schichten Architektur" target="_blank" href="http://code-inside.de/blog/2008/07/09/howto-3-tier-3-schichten-architektur/"&gt; 3-Tier/3 Schichten Architektur&lt;/a&gt;) wird es immer wichtiger Abhängigkeiten zu minimieren. &lt;br&gt;In großen Projekten kann es leicht vorkommen, dass ein Chaos von Abhängigkeiten entsteht. &lt;br&gt;Nicht alle Abhängigkeiten lassen sich vermeiden, aber viele sind unnötig. Vor allem falsche können die Entwicklung eines Projektes aufhalten.&lt;br&gt;&lt;br&gt;In den diversen Schichten(Data, Web, …)&amp;nbsp; den Code unabhängig voneinander zu halten, hilft dem gesamten Prozess ungemein. &lt;br&gt;Eine Technik, um die Kopplung so gering wie möglich zu halten ist ein Inversion of Control Container. (IoC Container) &lt;br&gt;&lt;br&gt;Mehr dazu in einem Artikel von Martin Fowler: &lt;a title="Inversion of control" target="_blank" href="http://martinfowler.com/articles/injection.html"&gt;Inversion of Control Containers and the Dependency Injection pattern&lt;/a&gt;&lt;br&gt;&lt;br&gt;Genauer eingegangen wird auf den frischen IoC Container „&lt;a title="LightCore" target="_blank" href="http://lightcore.peterbucher.ch/"&gt;LightCore&lt;/a&gt;“ von &lt;a target="_blank" href="http://www.peterbucher.ch"&gt;Peter Bucher&lt;/a&gt;. &lt;br&gt;Wie das Wort schon sagt, ist dieser lightweight und kann mit Performance und Einfachheit punkten, muss aber nicht in Sachen Features einpacken. &lt;br&gt;&lt;br&gt;Eine Feature List von Lightcore finder sich &lt;a target="_blank" href="http://lightcore.peterbucher.ch/features.aspx"&gt;hier&lt;/a&gt;.&lt;br&gt;Weiter Argumente die für LightCore sprechen finden sich in einem &lt;a target="_blank" title="LightCore im Vergleich" href="http://www.des-eisbaeren-blog.de/post/2010/01/01/LightCore.aspx"&gt;Blogeintrag&lt;/a&gt; von &lt;a title="Golo Roden" target="_blank" href="http://www.des-eisbaeren-blog.de/"&gt;Golo Roden&lt;/a&gt;.&lt;br&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/01/16/inversion-of-control-di-ioc-container-lightcore.aspx" title="LightCore - Einführung"&gt;1. LightCore – Einführung&lt;/a&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-xml-module.aspx" title="LightCore - Registrierung über Xml Module"&gt;2. LightCore – Registrierung über Xml Module&lt;/a&gt;&lt;br&gt;&lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2010/02/21/ioc-container-lightcore-registration-generics.aspx" title="LightCore - Registrierung von Generics"&gt;3. LightCore – Registrierung von Generics&lt;/a&gt;&lt;br&gt;&lt;br&gt;Der Erste Teil der Einführung wird ein einfaches Beispiel mit LightCore behandeln. &lt;br&gt;Um etwas tiefer in die Materie einzusteigen wird im zweiten Teil gezeigt, wie man mit LightCore die Unabhängigkeit einer 3-Schichten Architektur verbessern kann, wie man mit dem integrierten Registrations-Modul über Xml bzw. XAML arbeitet und wie man Generics registrieren kann.&lt;br&gt;&lt;br&gt;&lt;b&gt;Einbinden von Lightcore&lt;/b&gt;&lt;br&gt;&lt;br&gt;Wer auf die Entwicklungsversion zugreifen will, muss sich diese mit &lt;a title="Subversion SVN" href="http://subversion.tigris.org/"&gt;Subversion&lt;/a&gt; laden.&lt;br&gt;Diese wurde vor allem laut den Prinzipien von CCD (&lt;a title="Clean code developers" target="_blank" href="http://www.clean-code-developer.de/"&gt;Clean code development&lt;/a&gt;) und mit einer kompletten Unit Test Abdeckung entwickelt – somit auf jeden Fall einen Blick Wert!&lt;br&gt;&lt;br&gt;Für das folgende Beispiel reichen die DLLs, die es ebenfalls &lt;a title="Download lightcore" target="_blank" href="http://lightcore.peterbucher.ch/download.aspx"&gt;hier zum Download&lt;/a&gt; gibt.&lt;br&gt;&lt;br&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/LightCore2.jpg"&gt;&lt;br&gt;&lt;br&gt;Als Erstes müssen die gewünschten Kontrakte (Schnittstellen) registriert werden. Dies kann über zwei Varianten erfolgen:&lt;br&gt;&lt;ul&gt;&lt;li&gt;Registrierung über Code&lt;/li&gt;&lt;li&gt;Registrierung über eine XML Konfigurations- Datei&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;Wenn die Registrierung codeseitig gewünscht ist, sollte dies in der Globalen Anwendungsdatei (Global.asax) erfolgen&lt;br&gt;&lt;br&gt;Der Container wird als static member deklariert&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;private&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;static&lt;/span&gt; IContainer _container;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;In der Methode Application_Start wird der IoC Container initialisiert.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;protected&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;void&lt;/span&gt; Application_Start(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; sender, EventArgs e)&lt;br /&gt;{&lt;br /&gt;    var builder &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; ContainerBuilder();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Es gibt mehrere, optionale Features, die bei der Registrierung angegeben werden können. Für das Beispiel wird ein Logger registriert.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Einfache Registrierung&lt;/span&gt;&lt;br /&gt;builder.Register&amp;lt;ILogger, Logger&amp;gt;();&lt;br /&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Angabe eines Namen, um eine klare Trennung zu behalten&lt;/span&gt;&lt;br /&gt;builder.Register&amp;lt;ILogger, Logger&amp;gt;().WithName(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"MyLogger"&lt;/span&gt;);&lt;br /&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Angabe einer Gruppe, um bestimmte Kontrakte zu sammeln&lt;/span&gt;&lt;br /&gt;builder.Register&amp;lt;ILogger, Logger&amp;gt;().WithGroup(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"MyGroup"&lt;/span&gt;);&lt;br /&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Angabe von Argumenten für den Konstruktoraufruf&lt;/span&gt;&lt;br /&gt;builder.Register&amp;lt;ILogger, Logger&amp;gt;().WithArguments(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Test Argument"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;true&lt;/span&gt;);&lt;br /&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Gemischte Angabe&lt;/span&gt;&lt;br /&gt;builder.Register&amp;lt;ILogger, Logger&amp;gt;().WithName(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"MyLogger"&lt;/span&gt;).WithArguments(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Test Argument"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;true&lt;/span&gt;);&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Wenn alles registriert ist, den Container noch erstellen:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;_container &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; builder.Build();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die Registrierten Kontrakte können manuell über resolve aufgelöst werden&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Einfach&lt;/span&gt;&lt;br /&gt;var myLogger &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; _container.Resolve&amp;lt;ILogger&amp;gt;();&lt;br /&gt;&lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;//Mit Namen&lt;/span&gt;&lt;br /&gt;var myLogger &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; _container.Resolve&amp;lt;ILogger&amp;gt;(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"MyLogger"&lt;/span&gt;);&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Lifecycle&lt;/b&gt;&lt;br&gt;&lt;br&gt;Der Lebenszyklus besagt, ob bei jedem Anfordern einer Instanz über &amp;lt;Container&amp;gt;.Resolve&amp;lt;Icontract&amp;gt;() ein neues Objekt erstellt wird (Identisch zum new-Operator), oder das Objekt wiederverwendet werden kann.&lt;br&gt;&lt;br&gt;Für die Wiederverwendung von Instanzen werden Folgende Lebenszyklen unterstützt, wobei auch weitere Lebeszyklen geschrieben werden können:&lt;br&gt;&lt;ul&gt;&lt;li&gt;Transient&lt;/li&gt;&lt;li&gt;Singleton&lt;/li&gt;&lt;li&gt;ThreadSingleton&lt;/li&gt;&lt;li&gt;HttpRequest&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;Als Standard wird der transient Lebenszyklus verwendet. Dies besagt, dass bei jedem Aufruf eine neue Instanz der Klasse erstellt wird. Singleton würde bedeuten, dass eine einzige Instanz pro Anwendung erzeugt wird.&lt;br&gt;&lt;br&gt;Der Lebenszyklus wird einmal pro Kernel wie folgt gesetzt:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;builder.DefaultControlledBy&amp;lt;TransientLifecycle&amp;gt;();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Klarer weiße kann man den Lebenszyklus auch pro Registration angeben:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;builder.Register&amp;lt;ILogger, Logger&amp;gt;().ControlledBy&amp;lt;SingletonLifecycle&amp;gt;();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die Lebenszyklen können ohne Probleme beliebig erweitert werden.&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Das Ergebnis&lt;/b&gt;&lt;br&gt;&lt;br&gt;Angenommen wir haben den Logger registriert und möchten nun die Instanz in einer normalen Page Klasse&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; partial &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;class&lt;/span&gt; _Default : System.Web.UI.Page&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; ILogger Logger&lt;br /&gt;    { &lt;br /&gt;        get; &lt;br /&gt;        set;&lt;br /&gt;    }&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;In der Page Load wird der logger nun benötigt:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;protected&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;void&lt;/span&gt; Page_Load(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; sender, EventArgs e)&lt;br /&gt;{&lt;br /&gt;    ILogger logger &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;.Logger;&lt;br /&gt;    logger.log(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Page Loaded!"&lt;/span&gt;);&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Und wie von Magie haben wir eine Instanz der Klasse ohne direkte Erstellung auf der aktuellen Seite.&lt;br&gt;&lt;br&gt;Was ist passiert?&lt;br&gt;Ein http Modul in Lightcore macht automatisch ein resolve auf die Eigenschaften, die registriert sind. Somit entfällt die manuelle Auflösung und man ist komplett vom Dependency Injection Container unabhängig.&lt;br&gt;&lt;br&gt;Damit das auch klappt, muss das Modul in der Web config im Abschnitt &amp;lt;httpModules&amp;gt; hinzugefügt werden:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;httpModules&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;    ...&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;add&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;name&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="LightCoreDependencyInjectionModule"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;type&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="LightCore.Integration.Web.DependencyInjectionModule, LightCore.Integration.Web, Version=1.0.0.0, Culture=neutral"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;/&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;httpModules&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Mit ein bisschen Kreativität kann diese Vorgehensweise in allen möglichen Stellen eingesetzt werden – und schon ist man ein klein wenig Unabhängiger!&lt;br&gt;&lt;br&gt;Der zweite Teil des Posts wird mit folgenden Themen in Kürze folgen: Registration über Web/App Config oder eigene Konfigurationsdatei, Einsatz in einer 3 Schichten Architektur und Registrierung von Generics!&lt;br&gt;&lt;br&gt;Anbei gibt’s wie immer das oben angeführte Beispiel!&lt;br&gt;&lt;br&gt;
&lt;div class="DownloadBox"&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216101/download.aspx" title="Download LightCore Demo: Einführung"&gt;LightCore Demo: Einführung&lt;/a&gt;&lt;br&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216102/download.aspx" title="Download LightCore Demo: Registrierung über Xml Modul"&gt;LightCore Demo: Registerirung über Xml Module&lt;/a&gt;&lt;br&gt;
&lt;a href="http://www.aspnetzone.de/files/folders/216103/download.aspx" title="Download LightCore Demo: Registrierung von Generics"&gt;LightCore Demo: Registrierung von Generics&lt;/a&gt;&lt;br&gt;
&lt;/div&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=215449" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/215449.ashx" length="48217" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Asp.Net/default.aspx">Asp.Net</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/LightCore/default.aspx">LightCore</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/IoC/default.aspx">IoC</category></item><item><title>QUnit – Javascript Unit Test Framework</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2009/12/02/jQuery-javascript-qunit-unit-test-framework.aspx</link><pubDate>Wed, 02 Dec 2009 21:40:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:214913</guid><dc:creator>Roberto</dc:creator><slash:comments>2</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/214913.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=214913</wfw:commentRss><description>Wie Peter Bucher bereits in seinem &lt;a target="_blank" href="http://www.aspnetzone.de/blogs/peterbucher/archive/2009/11/01/wieviel-sinn-machen-unittests.aspx"&gt;Eintrag&lt;/a&gt; erwähnte, macht es überall dort Sinn Unit Tests einzubauen, wo das Verfahren leicht einzubauen ist, ohne sich dadurch das Leben schwer zu machen.&lt;br&gt;&lt;br&gt;Nicht nur für Asp.Net, sondern auch für Javascript gibt es diverse Test Suiten. &lt;br&gt;Nicht gewusst? Na dann wird’s Zeit! Letztendlich ist es nicht schön, wenn auf einer Webseite immer diese lästigen Javascript Fehler auftreten und somit bestimmte Funktionen der Seite nicht mehr funktionieren&lt;br&gt;&lt;br&gt;In diesem Beitrag wird auf &lt;a target="_blank" href="http://docs.jquery.com/QUnit"&gt;QUnit&lt;/a&gt; näher eingegangen. Diese wird bspw. zum Testen des &lt;a target="_blank" href="http://www.jquery.com"&gt;JQuery&lt;/a&gt; Frameworks verwendet, ist „easy-to-use“ und kann eigentlich alles, was man von einem einfachen Testing Framework erwartet.&lt;br&gt;&lt;br&gt;Zum starten braucht es nicht viel: Einfach die &lt;a target="_blank" href="http://github.com/jquery/qunit/raw/master/qunit/qunit.js"&gt;qunit.js&lt;/a&gt; und &lt;a target="_blank" href="http://github.com/jquery/qunit/raw/master/qunit/qunit.css"&gt;qunit.css&lt;/a&gt; in eine einfache HTML Seite einbinden und los geht’s!&lt;br&gt;&lt;br&gt;Im Body Tag der Seite das folgende Element hinzufügen:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;ol id=&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"qunit-tests"&lt;/span&gt;&amp;gt;&amp;lt;/ol&amp;gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Wie alles beginnt:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;test&lt;/span&gt;(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Das ist mein erster Test"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; () {&lt;br /&gt;    ok(true, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Dieser Test war erfolgreich"&lt;/span&gt;);&lt;br /&gt;});&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die Ausgabe wird nun wie folgt aussehen:&lt;br&gt;&lt;br&gt;&lt;i&gt;Das ist mein erster Test (&lt;span class="Red"&gt;0&lt;/span&gt;, &lt;span class="Green"&gt;1&lt;/span&gt;, 1)&lt;/i&gt;&lt;br&gt;&lt;br&gt;Die Zahlen in den Klammern sagen aus, wie viele von den insgesamt durchgeführten „&lt;a target="_blank" href="http://de.wikipedia.org/wiki/Assertion_%28Informatik%29"&gt;Assertions&lt;/a&gt;“&amp;nbsp; erfolgreich und fehlerhaft waren:&lt;br&gt;&lt;br&gt;&lt;i&gt;(&lt;span class="Red"&gt;Anzahl Fehlgeschlagen&lt;/span&gt;, &lt;span class="Green"&gt;Anzahl Erfolgreich&lt;/span&gt;, Anzahl Insgesamt)&lt;/i&gt;&lt;br&gt;&lt;br&gt;Neben der „ok assertion“ gibt es noch zwei weitere: &lt;b&gt;equals &lt;/b&gt;und &lt;b&gt;same&lt;/b&gt;&lt;br&gt;Equals ist grundsätzlich nichts anderes als ein Vergleich zweier Werte, die als Parameter der Funktion übergeben werden:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;test&lt;/span&gt;(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Der zweite Test"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; () {&lt;br /&gt;    expect(2);&lt;br /&gt;    equals(true, false, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Fehlerhaft"&lt;/span&gt;);&lt;br /&gt;    equals(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Wert"&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Wert"&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Erfolgreich"&lt;/span&gt;);&lt;br /&gt;});&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Wem jetzt das &lt;b&gt;expect &lt;/b&gt;aufgefallen ist, es ist die Anzahl der zu erwartenden Assertions.&lt;br&gt;&lt;br&gt;Um Objekte zu vergleichen wird die „same“ Funktion angeboten. Hierfür ein einfaches Objekt „User“ mit der Property „Name“&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;test&lt;/span&gt;(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Der 'same' Test"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; () {&lt;br /&gt;    expect(2);&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; user1 &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; new User(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Name1"&lt;/span&gt;);&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; user2 &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; new User(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Name2"&lt;/span&gt;);&lt;br /&gt;    ok(true, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"User wurden erstellt"&lt;/span&gt;);&lt;br /&gt;    same(user1, user2, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"user1 und user2 sind identisch"&lt;/span&gt;);&lt;br /&gt;});&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&amp;nbsp;&lt;br&gt;Der Test sollte nun eine erfolgreiche und eine fehlgeschlagene Assertion ausgeben, da die User zwar erfolgreich erstellt wurden, aber nicht den gleichen Namen haben.&lt;br&gt;&lt;br&gt;Zudem bietet QUnit einige Callbacks die überschrieben werden können, wie bspw. log um die Fehler zu loggen, testStart, testDone…&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;QUnit.&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;log&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;(result, &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;message&lt;/span&gt;) {&lt;br /&gt;    alert(&lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;message&lt;/span&gt;);&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Ein weiterer Punkt bei QUnit ist die Unterteilung in &lt;b&gt;Modulen&lt;/b&gt;. Der Aufruf erfolgt über module( name, [lifecycle] ), wobei lifecycle die setup und teardown callbacks darstellt, die bei jedem Test im aktuellen Modul ausgeführt werden. Dies kann verwendet werden, um bspw. bestimmte Test- Daten für jeden Test zu erstellen.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;module(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Modul abc"&lt;/span&gt;, {&lt;br /&gt;    setup: &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; () {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;.user &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; new User(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"TestUser"&lt;/span&gt;);&lt;br /&gt;    },&lt;br /&gt;    teardown: &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; () {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;.user &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; null;&lt;br /&gt;    }&lt;br /&gt;});&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;test&lt;/span&gt;(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"User Test"&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; () {&lt;br /&gt;    expect(1);&lt;br /&gt;    equals(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;.user.&lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;name&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"TestUser"&lt;/span&gt;);&lt;br /&gt;});&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Nun ist Kreativität gefragt! Wie JQuery das macht sieht man bei den Beispielen in der &lt;a target="_blank" href="http://docs.jquery.com/QUnit"&gt;QUnit Dokumentation&lt;/a&gt; im unteren Bereich der Seite.&lt;br&gt;&lt;br&gt;Anbei gibt’s wie immer ein Beispiel dazu!&lt;br&gt;Viel Spass beim Testen :-)&lt;br&gt;&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=214913" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/214913.ashx" length="895" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/jQuery/default.aspx">jQuery</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Javascript/default.aspx">Javascript</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Testing/default.aspx">Testing</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/QUnit/default.aspx">QUnit</category></item><item><title>Dynamic Objects in C# 4.0</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2009/11/24/DLR-AspNet-Dynamic-Objects-csharp-4.aspx</link><pubDate>Tue, 24 Nov 2009 00:00:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:214739</guid><dc:creator>Roberto</dc:creator><slash:comments>1</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/214739.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=214739</wfw:commentRss><description>Eine neues Feature in C# 4 ist die „Dynamic Language Runtime“ (DLR). Mit dem „&lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/dd264741%28VS.100%29.aspx"&gt;dynamic&lt;/a&gt;“ Schlüsselwort ist es möglich zur Laufzeit Properties und Methoden an Objekten zu binden. Dabei gibt es allerdings keinerlei Überprüfung ob diese auch existieren – dafür erhöht sich die Flexibilität der Anwendung.&lt;br&gt;Eigene Klassen können durch die Implementierung des &lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/system.dynamic.dynamicobject%28VS.100%29.aspx"&gt;DynamicObject &lt;/a&gt;und durch die Überschreibung der gewünschten Methoden erstellt werden.&lt;br&gt;&lt;br&gt;&lt;b&gt;Ein praktisches Beispiel dazu:&lt;/b&gt;&lt;br&gt;Wer kennt das Problem nicht, es gibt eine Javascript Komponente die im Header der Seite einiger Einstellungen bedarf.&amp;nbsp; Diese möchte man aber nicht per Javascript, sondern per Asp.Net setzen. Ist ja kein Problem, aber jede einzelne Property Serverseitig abzubilden kostet viel Zeit, insbesondere wenn die Komponente ständig weiterentwickelt wird.&lt;br&gt;&lt;br&gt;Das Ausgabe sollte in JavaScript etwa so aussehen:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; myObject &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; new JavaScriptComponent();&lt;br /&gt;myObject.width &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; 400;&lt;br /&gt;myObject.show &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; false;&lt;br /&gt;…&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Dynamic to the rescue!&lt;/b&gt;&lt;br&gt;Wie oben beschreiben ist es möglich an dynamischen Objekten beliebige Properties und Methoden zu binden.&lt;br&gt;Dafür einfach eine Klasse ClientScriptCreator die von &lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/system.dynamic.dynamicobject%28VS.100%29.aspx"&gt;DynamicObject &lt;/a&gt;ableitet.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;class&lt;/span&gt; ClientScriptCreator : DynamicObject&lt;/span&gt;&lt;/code&gt; &lt;br&gt;&lt;br&gt;Um die dynamischen Properties zu speichern wird ein &lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/xfhwa508.aspx"&gt;Dictionary &lt;/a&gt;verwendet&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Dictionary&amp;lt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;&amp;gt; _members;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Für die Benennung der clientseitigen Klasse und des Objekts werden zwei weitere Member deklariert&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; _clientClassName;&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; _clientObjectName;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die drei Member werden nun im Konstruktor der Klasse übergeben/gesetzt&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; ClientScriptCreator(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; clientClassName, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; clientObjectName)&lt;br /&gt;{&lt;br /&gt;    _members &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; Dictionary&amp;lt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;,&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;&amp;gt;();&lt;br /&gt;    _clientClassName &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; clientClassName;&lt;br /&gt;    _clientObjectName &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; clientObjectName;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Mit der Überschreibung der Methode &lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/system.dynamic.dynamicobject.trysetmember%28VS.100%29.aspx"&gt;TrySetMember&lt;/a&gt; werden die Properties (falls noch nicht vorhanden) in das Dictionary gespeichert.&lt;br&gt;Diese liefert einen &lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/system.dynamic.setmemberbinder%28VS.100%29.aspx"&gt;SetMemberBinder&lt;/a&gt; und ein Objekt mit dem Wert.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;override&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;bool&lt;/span&gt; TrySetMember(SetMemberBinder binder, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; value)&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;if&lt;/span&gt; (!_members.ContainsKey(binder.Name))&lt;br /&gt;    {&lt;br /&gt;        _members.Add(binder.Name, value);&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;true&lt;/span&gt;;&lt;br /&gt;    }&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;else&lt;/span&gt;&lt;br /&gt;    {&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;false&lt;/span&gt;;&lt;br /&gt;    }&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Ähnlich dazu gibt es die &lt;a target="_blank" href="http://msdn.microsoft.com/en-us/library/system.dynamic.dynamicobject.trygetmember%28VS.100%29.aspx"&gt;TryGetMember&lt;/a&gt; Methode, welche den gesetzten Wert (falls vorhanden) zurückgibt&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;override&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;bool&lt;/span&gt; TryGetMember(GetMemberBinder binder, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;out&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; result)&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;if&lt;/span&gt; (_members.ContainsKey(binder.Name))&lt;br /&gt;    {&lt;br /&gt;        result &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; _members[binder.Name];&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;true&lt;/span&gt;;&lt;br /&gt;    } &lt;br /&gt;    result &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;null&lt;/span&gt;;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;false&lt;/span&gt;;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Nun ist die Klasse im Stande dynamisch Properties zu setzen und zurück zu geben.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;dynamic clientScriptCreator &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; ClientScriptCreator();&lt;br /&gt;clientScriptCreator.PropertyOne &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"'Some string'"&lt;/span&gt;;&lt;br /&gt;clientScriptCreator.PropertyTwo &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"false"&lt;/span&gt;;&lt;br /&gt;clientScriptCreator.PropertyThree &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"4711"&lt;/span&gt;;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Das client Script muss nun zusammengesetzt werden um in den Header geschrieben zu werden. &lt;br&gt;Dafür eine Methode GetDefaultScript die den Dictionary Member durchläuft und diesen in ein &amp;lt;script type=“test/javascript“&amp;gt; Konstrukt wiedergibt.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;private&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; GetDefaultScript()&lt;br /&gt;{&lt;br /&gt;    StringBuilder sb &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;new&lt;/span&gt; StringBuilder();&lt;br /&gt;    sb.Append(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"&amp;lt;script type=\"text/javascript\"&amp;gt;"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; Environment.NewLine);&lt;br /&gt;    sb.Append(_clientObjectName &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;" = new "&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; _clientClassName &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"();"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; Environment.NewLine);&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;foreach&lt;/span&gt; (KeyValuePair&amp;lt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt;, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;&amp;gt; kvp &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;in&lt;/span&gt; _members)&lt;br /&gt;    {&lt;br /&gt;        sb.Append(_clientObjectName &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"."&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; kvp.Key &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;" = "&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; kvp.Value &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;";"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; Environment.NewLine);&lt;br /&gt;    }&lt;br /&gt;    sb.Append(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"&amp;lt;/script&amp;gt;"&lt;/span&gt;);&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; sb.ToString();&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Das Script soll anschließend in einem Literal Control im head Bereich der Seite stehen&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;head&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;/&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;asp:Literal&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ID&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="ClientScriptLiteral"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;runat&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="server"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;/&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;head&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;
&lt;br&gt;Was noch fehlt ist der Aufruf der Methode, die das client Script zurückgibt.&lt;br&gt;Dafür die TryInvokeMember überschreiben, damit dynamische Methoden aufgerufen werden können. In dem folgenden Abschnitt wird auf die Methode „ToDefaultScript“ überprüft – wenn diese vorkommt, wird das Script zurückgeliefert:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;override&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;bool&lt;/span&gt; TryInvokeMember(InvokeMemberBinder binder, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt;[] args, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;out&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; result)&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;if&lt;/span&gt; (binder.Name == &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"ToDefaultScript"&lt;/span&gt;)&lt;br /&gt;    {&lt;br /&gt;        result &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; GetDefaultScript();&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;true&lt;/span&gt;;&lt;br /&gt;    }&lt;br /&gt;    result &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;null&lt;/span&gt;;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;false&lt;/span&gt;;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;In der Page_Load der Seite erfolgt der Aufruf wie folgt&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ClientScriptLiteral.Text &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; clientScriptCreator.ToDefaultScript();&lt;/span&gt;&lt;/code&gt;&lt;br&gt;
&lt;br&gt;Das Ganze ist nun wirklich dynamisch. Wird die JavaScript Komponente weiter entwickelt, braucht man sich nicht mehr darum zu kümmern und kann die Objekte wo gewünscht einfach erweitern.&lt;br&gt;&lt;br&gt;Anbei gibt’s noch das Beispiel als Visual Studio 2010 Projekt dazu.&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=214739" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/214739.ashx" length="14397" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Dynamic/default.aspx">Dynamic</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Asp.Net/default.aspx">Asp.Net</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/C_2300_+4/default.aspx">C# 4</category></item><item><title>ASP.NET Update Panel, Callback und jQuery Webservice - Drei Ajax Techniken im Überblick</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2009/11/02/asp-net-Ajax-UpdatePanel-Callback-jQuery-Webservice.aspx</link><pubDate>Mon, 02 Nov 2009 21:05:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:214333</guid><dc:creator>Roberto</dc:creator><slash:comments>5</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/214333.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=214333</wfw:commentRss><description>Dass mit diversen Techniken clientseitig über Javascript serverseitige Funktionen aufgerufen werden können ist nichts Neues.&lt;br&gt;Doch welche Technik überzeugt und welche passt am besten? In diesem Artikel werden drei grundlegende Vorgehensweisen erläutert und miteinander verglichen: &lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2009/11/02/asp-net-Ajax-UpdatePanel-Callback-jQuery-Webservice.aspx#UpdatePanel"&gt;Das Updatepanel&lt;/a&gt;, &lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2009/11/02/asp-net-Ajax-UpdatePanel-Callback-jQuery-Webservice.aspx#Callback"&gt;das Callback&lt;/a&gt; und der &lt;a href="http://www.aspnetzone.de/blogs/robertobez/archive/2009/11/02/asp-net-Ajax-UpdatePanel-Callback-jQuery-Webservice.aspx#Webservice"&gt;clientseitige Webservice Aufruf&lt;/a&gt;. Dazu gibt es jeweils ein kurzes Beispiel. &lt;br&gt;&lt;br&gt;&lt;b&gt;&lt;a name="UpdatePanel"&gt;Das Updatepanel&lt;/a&gt; – Segen oder Fluch?&lt;/b&gt;&lt;br&gt;&lt;br&gt;Immer wieder wird über folgende Wörter gestolpert: „böses Asp.Net Updatepanel…“. Doch warum wird es dann oft verwendet und vor allem: warum ist es böse?&lt;br&gt;Grundsätzlich entspricht das Updatepanel dem traditionellen Postbackmodell – eine Anforderung ist immer noch ein vollständiges Postback. Das Formular wird inklusive Viewstate an den Server gesendet. Der Großteil des Lebenszyklus der Seite bleibt erhalten und das Rendern bleibt dem Server überlassen. &lt;br&gt;&lt;br&gt;&lt;i&gt;„Es ist aber immer noch schneller als ein komplettes Postback!“&lt;/i&gt;&lt;br&gt;&lt;br&gt;Das stimmt tatsächlich - der Unterschied und Vorteil liegt also nicht in der Anforderung, sondern in der Antwort des Servers.&lt;br&gt;Nur bestimmte Teile der Seite bzw. des &lt;a target="_blank" href="http://msdn.microsoft.com/de-de/library/ms972427.aspx"&gt;Viewstates&lt;/a&gt; werden zurückgegeben und ausgetauscht (Partielles Rendering).&lt;br&gt;&lt;br&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/UpdatePanel.jpg" alt="Update Panel"&gt;&lt;br&gt;&lt;b&gt;Fazit:&lt;/b&gt;&lt;br&gt;&lt;br&gt;Diese Variante ist mit Sicherheit die einfachste. Dadurch, dass sich der auszutauschende Inhalt innerhalb eines Containers befindet, kann das UpdatePanel als relativ elegant bezeichnen werden. Allerdings leidet die Performance unter der Datenmenge die von Client zu Server gesendet wird (gesamte Formulardaten inkl. Viewstate). Das UpdatePanel ist nicht clientseitig beeinflussbar und somit auf Serverseitige Steuerung angewiesen. Die Features leiden darunter.&lt;br&gt;&lt;br&gt;&lt;table style="width:200px;" align="" cellpadding="0" cellspacing="0"&gt;
&lt;tr&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Einfach &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Elegant &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Performant &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;
&lt;/table&gt;
&lt;br&gt;&lt;b&gt;Beispiel:&lt;/b&gt;&lt;br&gt;&lt;br&gt;Um die partiellen UpdatePanel Aktualisierungen nutzen zu können muss der Seite ein ScriptManager hinzugefügt werden&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;asp:ScriptManager&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ID&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="ScriptManager1"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;runat&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="server"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die Deklaration des UpdatePanels siehst dann wie folgt aus (mehr Infos gibt es &lt;a target="_blank" title="ASP.NET Update Panel" href="http://www.asp.net/Ajax/Documentation/Live/overview/UpdatePanelOverview.aspx"&gt;hier&lt;/a&gt;)&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;asp:UpdatePanel&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ID&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="UpdatePanel1"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;runat&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="server"&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ContentTemplate&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;asp:Label&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ID&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="Message"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Text&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="This is a update panel…"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;runat&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="server"&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;asp:Button&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ID&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="Save"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Text&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="Save"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;runat&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;="server"&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;/&amp;gt;&lt;/span&gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;/&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ContentTemplate&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;/&lt;/span&gt;&lt;span style="color: Maroon;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;asp:UpdatePanel&lt;/span&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;&lt;a name="Callback"&gt;Das Callback&lt;/a&gt; – Keine Angst, es beißt nicht!&lt;/b&gt;&lt;br&gt;&lt;br&gt;Das Callback ruft Servercode mittels Clientscript auf, ohne dadurch(zwingend) einen Postback zu verursachen. Klingt auf Anhieb super – aber ohne Postback gibt’s auch keine Formular Daten, d.h. die Anforderung besteht aus Parametern die per Javascript manuell angegeben werden. &lt;br&gt;Wie das Updatepanel nützt auch das Callback das &lt;a title="XmlHttpRequest Objekt" target="_blank" href="http://de.wikipedia.org/wiki/XMLHttpRequest"&gt;XmlHttp Objekt&lt;/a&gt; intern zum Aufruf der Serverseitigen Methoden. &lt;br&gt;Die gesendeten Daten sind sehr gering. Allerdings wird der Lebenszyklus der Seite normal durchlaufen. Das kann sich bei einer größeren Seite durchaus bemerkbar machen, denn sie wird wie deren Controls und Members normal erstellt.&lt;br&gt;Rückgabe des Callbacks können per json Serialisierte Daten oder auch normaler HTML Text sein.&lt;br&gt;&lt;br&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Callback.jpg" alt="Callback"&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Fazit:&lt;/b&gt;&lt;br&gt;&lt;br&gt;In bestimmten Fällen kann es durchaus Sinn machen, Servercode aufzurufen ohne dadurch ein Postback zu verursachen oder Teile der Seite neu zu rendern.&amp;nbsp; Der Datentransfer ist sehr gering – allerdings wird der Lebenszyklus der Seite trotzdem durchlaufen. Der Einbau ist durch die Implementierung zweier Interfaces relativ einfach und sauber.&lt;br&gt;&lt;br&gt;&lt;table style="width:200px;" align="" cellpadding="0" cellspacing="0"&gt;&lt;tr&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Einfach &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Elegant &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Performant &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;/table&gt;&lt;br&gt;&lt;b&gt;Beispiel:&lt;/b&gt;&lt;br&gt;&lt;br&gt;Damit das Callback funktioniert müssen folgende &lt;a title="Interfaces" target="_blank" href="http://msdn.microsoft.com/de-de/library/87d83y5b%28VS.80%29.aspx"&gt;Interfaces &lt;/a&gt;implementiert werden:&lt;br&gt;&lt;ul&gt;&lt;li&gt;ICallback&lt;/li&gt;&lt;li&gt;ICallbackEventHandler&lt;/li&gt;&lt;/ul&gt;Diese können sowohl in ein Control, als auch in eine normale Seite implementiert werden.&lt;br&gt;Das Interface hat zwei Methoden: &lt;br&gt;&lt;br&gt;RaiseCallbackEvent wird für den clientseitigen Aufruf über eine Javascript Funktion benötigt&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;void&lt;/span&gt; RaiseCallbackEvent(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; eventArgument)&lt;br /&gt;{&lt;br /&gt;  &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;//do something&lt;/span&gt;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;GetCallbackResult wird aufgerufen, wenn RaiseCallbackEvent durch ist. Die Rückgabe würd asynchrony an die Javascript Funktion zurück gegeben.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; GetCallbackResult() &lt;br /&gt;{&lt;br /&gt;  &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"something"&lt;/span&gt;;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Im Load oder Init Event werden folgende clientseitige Methoden registriert:&lt;br&gt;&lt;br&gt;&lt;ul&gt;&lt;li&gt;CallServer(arg, context) – ruft die serverseitige Methode auf&lt;/li&gt;&lt;li&gt;ReceiveServerData(arg, context) – bekommt das Ergebnis durch den „arg“ Parameter von GetCallbackResult()&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;protected&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;void&lt;/span&gt; Page_Load(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;object&lt;/span&gt; sender, EventArgs e)&lt;br /&gt;{&lt;br /&gt;    ClientScriptManager scriptManager &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; Page.ClientScript;&lt;br /&gt;    String callbackReference &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; scriptManager.GetCallbackEventReference(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"arg"&lt;/span&gt;,&lt;br /&gt;    &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"ReceiveServerData"&lt;/span&gt;, &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;""&lt;/span&gt;);&lt;br /&gt;    String callbackScript &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"function CallServer(arg, context) {"&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; callbackReference &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"; }"&lt;/span&gt;;        &lt;br /&gt;    scriptManager.RegisterClientScriptBlock(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;.GetType(),&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"CallServer"&lt;/span&gt;, callbackScript, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;true&lt;/span&gt;);&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die 2 Javascript Funktionen sehen wie folgt aus&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; ReceiveServerData(arg, context)&lt;br /&gt;{&lt;br /&gt;  alert(arg);&lt;br /&gt;} &lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt; CallSrv()&lt;br /&gt;{&lt;br /&gt;  CallServer('get something', '');&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;&lt;a name="Webservice"&gt;Scriptservice&lt;/a&gt; – jQuery sei Dank!&lt;/b&gt;&lt;br&gt;&lt;br&gt;Durch &lt;a title="jQuery" target="_blank" href="http://jquery.com/"&gt;jQuery&lt;/a&gt; wurde es relativ einfach einen Webservice über Javascript aufzurufen. Wie bereits in einem &lt;a title="Übergabe komplexer Objekte per Json Serialisierung von jQuery zu ASP.NET Webservice" target="_blank" href="http://aspnetzone.de/blogs/robertobez/archive/2009/08/31/komplexe-objekte-per-json-serialisierung-von-jquery-zu-aspnet-webservice.aspx"&gt;letzten Eintrag beschrieben&lt;/a&gt;, werden über einen minimalen HTTP Post Request an den Server die benötigten Daten zurückgesendet. Diese sind in der Regel um einiges geringer als der komplette HTML Code dazu. Diese „reine Ajax Variante“ bringt das Problem mit sich, dass das Rendering auf dem Client stattfindet und dafür gibt es nur zwei simple Funktionen:&lt;br&gt;&lt;ul&gt;&lt;li&gt;innerHTML, das den kompletten Inhalt eines Elements austauscht&lt;/li&gt;&lt;li&gt;DOM APIs, die mit Tags und Attributen arbeiten&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;&lt;br&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Webservice.jpg" alt="Webservice"&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Fazit:&lt;/b&gt;&lt;br&gt;&lt;br&gt;In der Praxis ist eine reine Ajax Vorgehensweise meist besser als das UpdatePanel Prinzip. Das Rendering findet auf dem Client statt und der Server schickt nur die benötigten Daten zurück. Dies ist zwar performant, dafür aber aufwendiger zu integrieren. Wer mit jQuery arbeitet, sollte sich auf jeden Fall über diese Möglichkeit Gedanken machen.&lt;br&gt;&lt;br&gt;&lt;table style="width:200px;" align="" cellpadding="0" cellspacing="0"&gt;&lt;tr&gt;&lt;td&gt;Einfach &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Elegant &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;tr&gt;&lt;td&gt;Performant &lt;/td&gt;&lt;td&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;img src="http://11249.2.sihosting.net/Blog/Images/Star.gif"&gt;&lt;/td&gt;&lt;/tr&gt;&lt;/table&gt;&lt;br&gt;&lt;b&gt;Beispiel&lt;/b&gt;:&lt;br&gt;&lt;br&gt;Ein kurzer Auszug aus einem Beispiel. &lt;a title="Übergabe komplexer Objekte per Json Serialisierung von jQuery zu ASP.NET Webservice" target="_blank" href="http://aspnetzone.de/blogs/robertobez/archive/2009/08/31/komplexe-objekte-per-json-serialisierung-von-jquery-zu-aspnet-webservice.aspx"&gt;(In voller Länge)&lt;/a&gt;&lt;br&gt;&lt;br&gt;Die Webservice methode&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;[WebMethod]&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;string&lt;/span&gt; GetTestValue()&lt;br /&gt;{&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Test value"&lt;/span&gt;;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Der Javascript Aufruf&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;$.ajax({&lt;br /&gt;    type        : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"POST"&lt;/span&gt;,&lt;br /&gt;    url         : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Webservices.asmx/GetTestValue"&lt;/span&gt;,&lt;br /&gt;    data        : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;""&lt;/span&gt;,&lt;br /&gt;    contentType : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"application/json; charset=utf-8"&lt;/span&gt;,&lt;br /&gt;    dataType    : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"json"&lt;/span&gt;,&lt;br /&gt;    success     : &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( Result ) {&lt;br /&gt;        &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Result.d enthält die Rückgabe&lt;/span&gt;&lt;br /&gt;        $(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"#MyContainer"&lt;/span&gt;).html(Result.d);&lt;br /&gt;    }&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Welche Technik nun die passende ist, hängt immer stark vom Anwendungsfall ab - jede Technik hat ihre Stärken und Schwächen.&lt;br&gt;Grundsätzlich gilt - wenn Post Daten, Control States o.Ä. benötigt werden (bspw. bei einem größeren Eingabe-Formular), kann das UpdatePanel verwenden werden.&lt;br&gt;Wird darauf verzichtet, sollte das Callback bzw. der Webservice Aufruf in Betracht gezogen werden!&lt;br&gt;&lt;br&gt;
&lt;div class="HinweisBox"&gt;
&lt;b&gt;Update:&lt;/b&gt;&lt;br&gt;Einen Vergleich zu Ajax.Pro von &lt;a target="_blank" href="http://weblog.drescher-hackel.de"&gt;René Drescher-Hackel&lt;/a&gt; findet sich &lt;a target="_blank" href="http://weblog.drescher-hackel.de/38D42556-AFF4-482B-AF53-8F6CD2D47ABE/Update+ohne+Panel+mit+AJAX.PRO.content.aspx"&gt;hier&lt;/a&gt;!&lt;/div&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=214333" width="1" height="1"&gt;</description><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Ajax/default.aspx">Ajax</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Callback/default.aspx">Callback</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Update+Panel/default.aspx">Update Panel</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/jQuery/default.aspx">jQuery</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Javascript/default.aspx">Javascript</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Webservice/default.aspx">Webservice</category></item><item><title>jQuery Plugin selbstgemacht - Dynamisches laden von ASP.NET Controls</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2009/09/02/jquery-plugin-dynamisches-laden-von-asp-net-controls.aspx</link><pubDate>Wed, 02 Sep 2009 21:25:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:212717</guid><dc:creator>Roberto</dc:creator><slash:comments>5</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/212717.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=212717</wfw:commentRss><description>Wer kennt sie nicht... die zauberhaften ajax Kekse.&lt;br&gt;&lt;br&gt;Doch was steckt dahinter?&lt;br&gt;&lt;br&gt;In diesem Beitrag werde ich anhand eines Beispiels erklären, wie man ein &lt;a target="_blank" href="http://plugins.jquery.com/"&gt;jQuery Plugin&lt;/a&gt; dafür erstellen kann.&lt;br&gt;&lt;br&gt;Die Vorgehensweiße:&lt;br&gt;Über
einen clientseitigen Funktionsaufruf werden wir den Inhalt eines HTML
Containers mit dem eines klassischen ASP.NET Controls ersetzen.&lt;br&gt;Das alles passiert mit einem &lt;a target="_blank" href="http://aspnetzone.de/blogs/robertobez/archive/2009/08/31/komplexe-objekte-per-json-serialisierung-von-jquery-zu-aspnet-webservice.aspx"&gt;Webservice Aufruf per jQuery&lt;/a&gt; der den HTML Code zurückliefert.&lt;br&gt;&lt;br&gt;&lt;b&gt;Schritt 1: Das Plugin&lt;br&gt;&lt;/b&gt;&lt;br&gt;Eine kurze Einführung in jQuery Plugins.&lt;br&gt;In jQuery werden Objekte mit $.fn.&amp;lt;Name&amp;gt; = function() {} erweitert.&lt;br&gt;Allerdings sollte der "$" Alias vermieden werden, da Konflikte mit anderen Javascript Bibliotheken(bspw. &lt;a target="_blank" href="http://prototypejs.org/"&gt;prototype.js&lt;/a&gt;) entstehen können. &lt;br&gt;Die am häufigsten benutzte Vorgehensweise sieht so aus:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;($){&lt;br /&gt;    &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// --- Plugin kommt hier&lt;/span&gt;&lt;br /&gt;})(jQuery);&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Das neue Plugin nennt sich ascxLoder. Als Parameter die Pfadangabe und eine Sammlung mit Einstellungen.&lt;br&gt;Natürlich werden auch Standard Werte benötigt, damit ewige Parameter Übergaben vermieden werden.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;$.fn.ascxLoader &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( path, params ) {&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; defaults &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; {&lt;br /&gt;        showLoadingScreen &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; true&lt;br /&gt;    };&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; config &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; $.extend(defaults, params);&lt;br /&gt;    &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// --- Mit $(this) können wir auf den Container zugreifen.&lt;/span&gt;&lt;br /&gt;    &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// --- über config.&amp;lt;Value&amp;gt; die gewünschte Einstellung.&lt;/span&gt;&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Der Aufruf erfolgt dann über:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;$(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"#MyContainer"&lt;/span&gt;).ascxLoader(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"~/Control.ascx"&lt;/span&gt;, { showLoadingScreen: false });&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;&lt;b&gt;Schritt 2: Der Webservice Aufruf&lt;/b&gt;&lt;br&gt;&lt;br&gt;Damit das Plugin jetzt auch den gewünschten Effekt bringt, bauen wir einen einfachen &lt;a target="_blank" href="http://aspnetzone.de/blogs/robertobez/archive/2009/08/31/komplexe-objekte-per-json-serialisierung-von-jquery-zu-aspnet-webservice.aspx"&gt;Webservice Request&lt;/a&gt; ein.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;($){&lt;br /&gt;$.fn.ascxLoader &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( path, params ) {&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; defaults &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; {&lt;br /&gt;        path: &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Control.ascx"&lt;/span&gt;,&lt;br /&gt;        showLoadingScreen: true&lt;br /&gt;    };    &lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; config &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; $.extend(defaults, params);&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; obj &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; $(&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;);&lt;br /&gt;    path &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; path || config.path;&lt;br /&gt;    &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// --- der Webservice Aufruf mit Pfad übergabe:&lt;/span&gt;&lt;br /&gt;    $.ajax({&lt;br /&gt;        type        : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"POST"&lt;/span&gt;,&lt;br /&gt;        url         : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Webservices.asmx/GetControl"&lt;/span&gt;,&lt;br /&gt;        data        : '{&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Path"&lt;/span&gt;:&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"' + path + '"&lt;/span&gt;}',&lt;br /&gt;        contentType : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"application/json; charset=utf-8"&lt;/span&gt;,&lt;br /&gt;        dataType    : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"json"&lt;/span&gt;,&lt;br /&gt;        success     : &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( result ) {&lt;br /&gt;            &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// result.d enthält den HTML code für unseren Container&lt;/span&gt;&lt;br /&gt;            obj.html( result.d );&lt;br /&gt;        }&lt;br /&gt;    });&lt;br /&gt;}&lt;br /&gt;})(jQuery);&lt;/span&gt;&lt;/code&gt;&lt;br&gt;
&lt;br&gt;
&lt;i&gt;Hinweis&lt;/i&gt;: "path = path || config.path" bedeutet: falls path undefined ist, nimm config.path&lt;br&gt;&lt;br&gt;&lt;b&gt;Schritt 3: Der Webservice&lt;/b&gt;&lt;br&gt;&lt;br&gt;Nun zum Webservice, der uns das gewünschte Control als HTML code zurückgibt.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;WebMethod()&amp;gt; _&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Function&lt;/span&gt; GetControl( Path &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;String&lt;/span&gt; ) &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;String&lt;/span&gt;&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Dim&lt;/span&gt; page        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;New&lt;/span&gt; Page()&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Dim&lt;/span&gt; userControl &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; UserControl &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; Page.LoadControl( Path )&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Dim&lt;/span&gt; output      &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;New&lt;/span&gt; System.IO.StringWriter()&lt;br /&gt;    page.Controls.&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Add&lt;/span&gt;( userControl )&lt;br /&gt;    HttpContext.Current.Server.Execute( page, output, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;false&lt;/span&gt; )       &lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;return&lt;/span&gt; output.ToString()&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;End&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Function&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Über folgenden Funktionsaufruf lässt sich das Control neu laden.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;   $(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Container"&lt;/span&gt;).ascxLoader(&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"~/Controls/Control.ascx"&lt;/span&gt;);&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;
War doch gar nicht so schwierig oder?&lt;br&gt;
&lt;br&gt;Anbei gibt es ein Beispiel zum selber ausprobieren.&lt;br&gt;
Dieses beinhaltet neben dem eben gezeigten Beispiel auch die
Möglichkeit, einer Anforderungen Parameter mitzugeben und so je nach
Aufruf ein geändertes Control zurückzugeben.&lt;br&gt;&lt;br&gt;
&lt;div class="HinweisBox"&gt;
&lt;b&gt;Update:&lt;/b&gt;&lt;br&gt;Eine gelungene und erweiterte Version des Plugins, von &lt;a target="_blank" href="http://www.zerbit.de"&gt;Kristof Zerbe&lt;/a&gt; findet sich &lt;a target="_blank" href="http://www.zerbit.de/artikel/usercontrol_dynamisch_laden_update_und_jquery_plugin.aspx"&gt;hier&lt;/a&gt;!&lt;/div&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=212717" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/212717.ashx" length="80136" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/jQuery/default.aspx">jQuery</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Javascript/default.aspx">Javascript</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Webservice/default.aspx">Webservice</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Plugin/default.aspx">Plugin</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Control/default.aspx">Control</category></item><item><title>Übergabe komplexer Objekte per Json Serialisierung von jQuery zu ASP.NET Webservice</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2009/08/31/komplexe-objekte-per-json-serialisierung-von-jquery-zu-aspnet-webservice.aspx</link><pubDate>Mon, 31 Aug 2009 21:00:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:212578</guid><dc:creator>Roberto</dc:creator><slash:comments>5</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/212578.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=212578</wfw:commentRss><description>Die Übergabe komplexer Objekte an einen Webservice per Json Serialisierung ist grundsätzlich relativ einfach, hin und wieder jedoch etwas tricky.&lt;br&gt;&lt;br&gt;&lt;b&gt;Warum jQuery und warum Json?&lt;/b&gt;&lt;br&gt;Ich bin ein großer Fan von &lt;a href="http://jquery.com/"&gt;jQuery&lt;/a&gt;. Der Webservice Aufruf erfordert keine Einbindung eines ScriptManagers und ist in weniger als 10 Zeilen Code geschrieben.&lt;br&gt;Die Datenmenge durch Json Serialisierung ist sehr gering. Bspw. könnte ein Grid mit 1000 Datensätzen etwa 200KB an Daten enthalten.&lt;br&gt;Json Serialisierte Webservice Methoden erfordern zwingend:&lt;br&gt;&lt;ul&gt;&lt;li&gt;Einen HTTP Post request&lt;/li&gt;&lt;li&gt;Den Header: "Content-Type: application/json; charset=utf-8"&lt;br&gt;&lt;/li&gt;&lt;/ul&gt;dies hat allerdings den Sicherheits- Vorteil, dass die Methode nicht über &amp;lt;script src="http://www.domain.de/Webservice.asmx/Methode"&amp;gt; ansprechbar ist, da die Browser das einfügen von eigenen Headern bei einem script Include nicht erlauben. Mehr zum Thema in einem ausführlichen &lt;a target="_blank" href="http://weblogs.asp.net/scottgu/archive/2007/04/04/json-hijacking-and-how-asp-net-ajax-1-0-mitigates-these-attacks.aspx"&gt;Beitrag&lt;/a&gt; von Scott Guthrie.&lt;br&gt;&lt;br&gt;&lt;b&gt;Los gehts!&lt;/b&gt;&lt;br&gt;Was wird benötigt?&lt;br&gt;&lt;ul&gt;&lt;li&gt;eine aktuelle &lt;a target="_blank" href="http://jquery.com/"&gt;jQuery Library&lt;/a&gt;&lt;br&gt;&lt;/li&gt;&lt;li&gt;Einen Json Serializer wie bspw. &lt;a target="_blank" href="http://www.json.org/json2.js"&gt;Json2&lt;/a&gt;&lt;/li&gt;&lt;/ul&gt;Damit der Webservice per jQuery ansprechbar ist, reicht es, folgendes &lt;a target="_blank" href="http://msdn.microsoft.com/de-de/library/system.web.script.services.scriptserviceattribute.aspx"&gt;Attribut&lt;/a&gt; hinzuzufügen:&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;System.Web.Script.Services.ScriptService()&amp;gt; _&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Die Einbindung des Webservices über Scriptmanager wird nicht benötigt.&lt;br&gt;&lt;br&gt;Zum Empfang von Daten liefert eine Webservice Methode bspw. eine List( Of &amp;lt;T&amp;gt; ) zurück.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;WebMethod()&amp;gt; _&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Function&lt;/span&gt; GetCustomers() &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; List( Of Customers )&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Return&lt;/span&gt; Customers&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;End&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Function&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Diese Methode wird per jQuery angesprochen&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;$.ajax({&lt;br /&gt;    type        : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"POST"&lt;/span&gt;,&lt;br /&gt;    url         : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Webservices.asmx/GetCustomers"&lt;/span&gt;,&lt;br /&gt;    data        : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;""&lt;/span&gt;,&lt;br /&gt;    contentType : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"application/json; charset=utf-8"&lt;/span&gt;,&lt;br /&gt;    dataType    : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"json"&lt;/span&gt;,&lt;br /&gt;    success     : &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( Result ) {&lt;br /&gt;        &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Result.d enthält die Customers liste&lt;/span&gt;&lt;br /&gt;        $.each( Result.d, &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( Key, Value ) {&lt;br /&gt;            alert( Value.Name );&lt;br /&gt;            &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// ...&lt;/span&gt;&lt;br /&gt;        }&lt;br /&gt;    }&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Zum senden der Daten&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;&amp;lt;WebMethod()&amp;gt; _&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Public&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Function&lt;/span&gt; SetCustomer( &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;ByVal&lt;/span&gt; Customer &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; Customer ) &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Boolean&lt;/span&gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Dim&lt;/span&gt; CustomerName &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;As&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;String&lt;/span&gt; &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; Customer.Name&lt;br /&gt;    &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;' ....&lt;/span&gt;&lt;br /&gt;    &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Return&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;True&lt;/span&gt;&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;End&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Function&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;In Javascript kann man mit &lt;a target="_blank" href="http://www.json.org/json2.js"&gt;Json2&lt;/a&gt; Objekte einfach über JSON.Stringify( &amp;lt;Objekt&amp;gt; ) serialisieren.&lt;br&gt;&lt;br&gt;&lt;code&gt;&lt;span style="color: Black;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;Customer &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;() {&lt;br /&gt;   &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;this&lt;/span&gt;.Name;&lt;br /&gt;}&lt;br /&gt;&lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;var&lt;/span&gt; CurrentCustomer &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; new Customer();&lt;br /&gt;    CurrentCustomer.Name &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;=&lt;/span&gt; &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"TestName"&lt;/span&gt;;&lt;br /&gt;&lt;br /&gt;$.ajax({&lt;br /&gt;    type        : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"POST"&lt;/span&gt;,&lt;br /&gt;    url         : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Webservices.asmx/SetCustomer"&lt;/span&gt;,&lt;br /&gt;    data        : '{&lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Customer"&lt;/span&gt;:' &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; JSON.stringify( CurrentCustomer ) &lt;span style="color: Red;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;+&lt;/span&gt; '}',&lt;br /&gt;    contentType : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"application/json; charset=utf-8"&lt;/span&gt;,&lt;br /&gt;    dataType    : &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"json"&lt;/span&gt;,&lt;br /&gt;    success     : &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;function&lt;/span&gt;( Result ) {&lt;br /&gt;        &lt;span style="color: Green;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;// Result.d enthält eine boolschen Wert&lt;/span&gt;&lt;br /&gt;        &lt;span style="color: Blue;background-color: Transparent;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;if&lt;/span&gt;( Result.d ) {&lt;br /&gt;            alert( &lt;span style="color: #666666;background-color: #e4e4e4;font-family: Courier New;font-size: 14px;font-weight: normal;"&gt;"Daten erfolgreich gespeichert."&lt;/span&gt; );&lt;br /&gt;        }&lt;br /&gt;    }&lt;br /&gt;}&lt;/span&gt;&lt;/code&gt;&lt;br&gt;&lt;br&gt;Anbei gibts noch ein ausführlicheres Beispiel dazu.&lt;br&gt;&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=212578" width="1" height="1"&gt;</description><enclosure url="http://www.aspnetzone.de/blogs/robertobez/attachment/212578.ashx" length="64205" type="application/zip" /><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Json/default.aspx">Json</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/jQuery/default.aspx">jQuery</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Javascript/default.aspx">Javascript</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Webservice/default.aspx">Webservice</category><category domain="http://www.aspnetzone.de/blogs/robertobez/archive/tags/Serialisierung/default.aspx">Serialisierung</category></item><item><title>Besser spät als nie!</title><link>http://www.aspnetzone.de/blogs/robertobez/archive/2009/08/30/besser-spaet-als-nie.aspx</link><pubDate>Sat, 29 Aug 2009 22:12:00 GMT</pubDate><guid isPermaLink="false">ce930855-ae9b-4fa4-8077-06a76071cc6a:212579</guid><dc:creator>Roberto</dc:creator><slash:comments>1</slash:comments><comments>http://www.aspnetzone.de/blogs/robertobez/comments/212579.aspx</comments><wfw:commentRss>http://www.aspnetzone.de/blogs/robertobez/commentrss.aspx?PostID=212579</wfw:commentRss><description>Jeder Anfang ist schwer :-) Aber irgendwann muss man ihn machen!&lt;br&gt;&lt;br&gt;Dieser Blog wird Themen rund um .Net, Ajax und Javascript beinhalten.&lt;br&gt;&lt;br&gt;Ein Dankeschön geht an:&lt;br&gt;&lt;ul&gt;&lt;li&gt;&lt;a href="http://www.aspnetzone.de/blogs/peterbucher/"&gt;Peter Bucher&lt;/a&gt; für die über Monate gebrachte Überzeugungskraft einen Blog zu starten.&lt;br&gt;&lt;/li&gt;&lt;li&gt;&lt;a href="http://www.aspnetzone.de/blogs/stefanfalz/"&gt;Stefan Falz&lt;/a&gt; für das Hosting des Blogs und der Möglichkeit einen Blog bei &lt;a href="http://www.aspnetzone.de/"&gt;ASP.NET Zone&lt;/a&gt; zu bekommen!&lt;/li&gt;&lt;li&gt;Jeden der diesen Beitrag liest und vielleicht in Zukunft mal vorbeischaut :)&lt;/li&gt;&lt;/ul&gt;Also auf gutes Gelingen!&lt;br&gt;&lt;img src="http://www.aspnetzone.de/aggbug.aspx?PostID=212579" width="1" height="1"&gt;</description></item></channel></rss>