<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>build broken &#187; Assembly</title>
	<atom:link href="http://blog.aztec-project.org/tag/assembly/feed/" rel="self" type="application/rss+xml" />
	<link>http://blog.aztec-project.org</link>
	<description>but we can fix it! Sometimes :)</description>
	<lastBuildDate>Fri, 11 Jun 2010 12:52:34 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=</generator>
		<item>
		<title>Wozu dient Reflection</title>
		<link>http://blog.aztec-project.org/2009/06/24/wozu-dient-reflection/</link>
		<comments>http://blog.aztec-project.org/2009/06/24/wozu-dient-reflection/#comments</comments>
		<pubDate>Wed, 24 Jun 2009 10:13:06 +0000</pubDate>
		<dc:creator>Thomas Christian</dc:creator>
				<category><![CDATA[How-To]]></category>
		<category><![CDATA[Assemblies]]></category>
		<category><![CDATA[Assembly]]></category>
		<category><![CDATA[C#]]></category>
		<category><![CDATA[Reflection]]></category>

		<guid isPermaLink="false">http://blog.aztec-project.org/?p=280</guid>
		<description><![CDATA[Über Reflection kann man zur Laufzeit relativ einfach an alle Informationen eines Assembly herankommen. So ist es möglich, dass man von einem Assembly den Namen, Klassen, Methoden mit ihren Parametern, Eigenschaften und Rückgabewerte, usw. auslesen kann. Aber nicht nur das Auslesen ist möglich, sondern auch das Setzen.

Jetzt stellt sich für den einen oder anderen vielleicht die Frage, warum man zur Laufzeit auf diese Daten zugreifen kann. Jede Assembly verfügt über Module in denen sich die einzelnen Typen(Klassen) befinden. Für jedes Modul gibt es Metadaten, in denen die einzelnen Typen beschrieben werden. Die Assembly selbst verfügt noch über ein Manifest, in dem sich alle Informationen zum Assembly (z.B. Name, Version) befinden. Wenn man über Reflection auf Assemblies zugreift, dann werden diese Metadaten und das Manifest dafür herangezogen. Die Metadaten verfügen zwar noch über die Informationen, welche Modifizierer für Methoden, Properties, usw. verwendet wurden, aber der MSIL-Code (die Zwischensprache, die vom Endcompiler in die plattformspezifische Sprache übersetzt wird) nicht mehr. Die Modifizierer sind nur für den Compiler relevant und werden nach dem Kompilieren nicht mehr berücksichtigt. Somit ist es auch möglich über Reflection Properties, Methoden, usw. zu verwenden, die nicht öffentlich sind.

Für dieses Beispiel habe ich eine kleine Konsolen-Anwendung geschrieben (TestApplication), welche eine Referenz auf das Assembly TestProject.dll hat. Dieses Assembly hat zwei von mir erstellte Typen (Klassen) TestClass und SecondTestClass. In den beiden Klassen steckt keine Logik. Sie sind ausschließlich für dieses Beispiel erstellt worden, um ein paar Daten zu modifizieren und um zu zeigen wie die Informationen aus dem Assembly ausgelesen und verändert werden können und dass es möglich ist an nicht-öffentliche Methoden heranzukommen.  <a href="http://blog.aztec-project.org/2009/06/24/wozu-dient-reflection/">weiter lesen...</a>]]></description>
			<content:encoded><![CDATA[<h2><span style="color: #ffffff;">.</span></h2>
<h2>Vorbereitung</h2>
<p>Über Reflection kann man zur Laufzeit relativ einfach an alle Informationen eines Assembly herankommen. So ist es möglich, dass man von einem Assembly den Namen, Klassen, Methoden mit ihren Parametern, Eigenschaften und Rückgabewerte, usw. auslesen kann. Aber nicht nur das Auslesen ist möglich, sondern auch das Setzen.</p>
<p>Jetzt stellt sich für den einen oder anderen vielleicht die Frage, warum man zur Laufzeit auf diese Daten zugreifen kann. Jede Assembly verfügt über Module in denen sich die einzelnen Typen(Klassen) befinden. Für jedes Modul gibt es Metadaten, in denen die einzelnen Typen beschrieben werden. Die Assembly selbst verfügt noch über ein Manifest, in dem sich alle Informationen zum Assembly (z.B. Name, Version) befinden. Wenn man über Reflection auf Assemblies zugreift, dann werden diese Metadaten und das Manifest dafür herangezogen. Die Metadaten verfügen zwar noch über die Informationen, welche Modifizierer für Methoden, Properties, usw. verwendet wurden, aber der MSIL-Code (die Zwischensprache, die vom Endcompiler in die plattformspezifische Sprache übersetzt wird) nicht mehr. Die Modifizierer sind nur für den Compiler relevant und werden nach dem Kompilieren nicht mehr berücksichtigt. Somit ist es auch möglich über Reflection Properties, Methoden, usw. zu verwenden, die nicht öffentlich sind.</p>
<p>Für dieses Beispiel habe ich eine kleine Konsolen-Anwendung geschrieben (TestApplication), welche eine Referenz auf das Assembly TestProject.dll hat. Dieses Assembly hat zwei von mir erstellte Typen (Klassen) TestClass und SecondTestClass. In den beiden Klassen steckt keine Logik. Sie sind ausschließlich für dieses Beispiel erstellt worden, um ein paar Daten zu modifizieren und um zu zeigen wie die Informationen aus dem Assembly ausgelesen und verändert werden können und dass es möglich ist an nicht-öffentliche Methoden heranzukommen.</p>
<p><span style="text-decoration: underline;"><strong>TestClass.cs</strong></span></p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<pre style="margin: 0px;"><span style="color: blue;">using</span> System.Security.Cryptography;</pre>
<pre style="margin: 0px;"><span style="color: blue;">using</span> System.Text;</pre>
<pre style="margin: 0px;"><span style="color: blue;">namespace</span> TestProject {</pre>
<pre style="margin: 0px;"><span style="color: blue;">class</span> <span style="color: #2b91af;">TestClass</span> {</pre>
<pre style="margin: 0px;">    <span style="color: blue;">public</span> <span style="color: blue;">string</span> Name { <span style="color: blue;">get</span>; <span style="color: blue;">set</span>; }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">public</span> <span style="color: blue;">string</span> Surname { <span style="color: blue;">get</span>; <span style="color: blue;">set</span>; }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">private</span> <span style="color: blue;">string</span> Password { <span style="color: blue;">get</span>; <span style="color: blue;">set</span>; }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">public</span> <span style="color: blue;">string</span> GetFullName() {</pre>
<pre style="margin: 0px;">        <span style="color: blue;">return</span> <span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"{0} {1}"</span>, Name ?? <span style="color: blue;">string</span>.Empty, Surname ?? <span style="color: blue;">string</span>.Empty).Trim();</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">public</span> <span style="color: blue;">void</span> SetPasswort(<span style="color: blue;">string</span> password){</pre>
<pre style="margin: 0px;">        Password = password;</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">private</span> <span style="color: blue;">static</span> <span style="color: blue;">string</span> CalcPassword(<span style="color: blue;">string</span> password){</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">MD5</span> md5 = <span style="color: #2b91af;">MD5</span>.Create();</pre>
<pre style="margin: 0px;">        <span style="color: blue;">byte</span>[] data = md5.ComputeHash(<span style="color: #2b91af;">Encoding</span>.Default.GetBytes(password));</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">StringBuilder</span> hashPassword = <span style="color: blue;">new</span> <span style="color: #2b91af;">StringBuilder</span>();</pre>
<pre style="margin: 0px;">        <span style="color: blue;">foreach</span> (<span style="color: blue;">byte</span> b <span style="color: blue;">in</span> data){</pre>
<pre style="margin: 0px;">            hashPassword.Append(b);</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">        <span style="color: blue;">return</span> hashPassword.ToString();</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">public</span> <span style="color: blue;">static</span> <span style="color: blue;">string</span> DoSomething(<span style="color: blue;">string</span> value) {</pre>
<pre style="margin: 0px;">        <span style="color: green;">// do something</span></pre>
<pre style="margin: 0px;">        <span style="color: blue;">return</span> value;</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">}</pre>
</div>
<p><span style="text-decoration: underline;"><strong>SecondTestClass.cs</strong></span></p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<pre style="margin: 0px;"><span style="color: blue;">namespace</span> TestProject {</pre>
<pre style="margin: 0px;">    <span style="color: blue;">public</span> <span style="color: blue;">class</span> <span style="color: #2b91af;">SecondTestClass</span> {</pre>
<pre style="margin: 0px;">        <span style="color: blue;">public</span> <span style="color: blue;">int</span> Addition(<span style="color: blue;">int</span> a, <span style="color: blue;">int</span> b) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">return</span> a + b;</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">        <span style="color: blue;">public</span> <span style="color: blue;">int</span> Multiply(<span style="color: blue;">int</span> a, <span style="color: blue;">int</span> b) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">return</span> a * b;</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">        <span style="color: blue;">public</span> <span style="color: blue;">double</span> Divide(<span style="color: blue;">int</span> a, <span style="color: blue;">int</span> b) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">if</span> (b == 0) <span style="color: blue;">return</span> 0;</pre>
<pre style="margin: 0px;">            <span style="color: blue;">return</span> a / b;</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">}
<span style="color: #ffffff;">.
.</span></pre>
</div>
<h2>Auslesen von Assembly- Type-Informationen</h2>
<p>Mit der Methode Thread.GetDomain().GetAssemblies() bekommt man alle referenzierten Assemblies der aktuellen Domäne. Damit die TestProject-Assembly auch in dieser Domäne vorhanden ist, muss vor dem Aufruf dieser Methode eine Instanz von einem Objekt dieses Assemblies erzeugt werden.</p>
<p>Um sich nun alle Assemblies der aktuellen Domäne anzeigen zu lassen, könnte man folgendes schreiben:</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">SecondTestClass</span> secondTestClass = <span style="color: blue;">new</span> <span style="color: #2b91af;">SecondTest</span><span style="color: #2b91af;"> </span><span style="color: #2b91af;">Class</span>();</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span>[] assemblies = <span style="color: #2b91af;">Thread</span>.GetDomain().GetAssemblies();</pre>
<pre style="margin: 0px;">    <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> assembly <span style="color: blue;">in</span> assemblies) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name
                                                          , assembly.GetName().Version));</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px;">}
<img class="size-full wp-image-281 aligncenter" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/1.png" alt="1" width="549" height="120" /></pre>
</div>
<p><span style="color: #ffffff;">.</span></p>
<p>Nun könnte man noch alle zur Verfügung stehenden Typen(Klassen) aus der TestProject-Assembly anzeigen.</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">SecondTestClass</span> secondTestClass = <span style="color: blue;">new</span> <span style="color: #2b91af;">SecondTestClass</span>();</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span>[] assemblies = <span style="color: #2b91af;">Thread</span>.GetDomain().GetAssemblies();</pre>
<pre style="margin: 0px;">    <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> assembly <span style="color: blue;">in</span> assemblies) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name
<pre style="margin: 0px;">                                                          , assembly.GetName().Version));</pre>
</pre>
<pre style="margin: 0px;">        <span style="color: blue;">if</span> (assembly.GetName().Name.Equals(<span style="color: #a31515;">"TestProject"</span>)) {</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">Type</span>[] types = assembly.GetTypes();</pre>
<pre style="margin: 0px;">            <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> type <span style="color: blue;">in</span> types) {</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++{0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px; text-align: left;">}

<img class="size-full wp-image-284 aligncenter" title="2" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/2.png" alt="2" width="533" height="130" /></pre>
</div>
</div>
<p><span style="color: #ffffff;">.</span></p>
<p>Wenn man nun allerdings Informationen eines bestimmten Types haben möchte, so werden diese Informationen aus den Metadaten des Moduls geladen, in dem sich der Type befindet. Um zum Beispiel alle öffentlichen Methoden eines Types anzuzeigen, könnte man folgendes schreiben:</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">SecondTestClass</span> secondTestClass = <span style="color: blue;">new</span> <span style="color: #2b91af;">SecondTestClass</span>();</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span>[] assemblies = <span style="color: #2b91af;">Thread</span>.GetDomain().GetAssemblies();</pre>
<pre style="margin: 0px;">    <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> assembly <span style="color: blue;">in</span> assemblies) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name
                                                          , assembly.GetName().Version));</pre>
<pre style="margin: 0px;">        <span style="color: blue;">if</span> (assembly.GetName().Name.Equals(<span style="color: #a31515;">"TestProject"</span>)) {</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">Type</span>[] types = assembly.GetTypes();</pre>
<pre style="margin: 0px;">            <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> type <span style="color: blue;">in</span> types) {</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">MethodInfo</span>[] methodInfos = type.GetMethods();</pre>
<pre style="margin: 0px;">                <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> methodInfos) {</pre>
<pre style="margin: 0px;">                    <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0}"</span>, info.Name);</pre>
<pre style="margin: 0px;">                }</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px;">}</pre>
</div>
<p align="center">
<p style="text-align: center;"><img class="size-full wp-image-285 aligncenter" title="3" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/3.png" alt="3" width="629" height="383" /></p>
<p><span style="color: #ffffff;">.</span></p>
<p>Es fällt auf, dass auch die Methoden ToString, Equals, GetHashCode und GetType in beiden Klassen angezeigt werden, obwohl diese in den Klassen überhaupt nicht deklariert wurden. Das liegt daran, dass alle Typen von <span style="color: blue;">object </span>ableiten und <span style="color: blue;">object</span> diese Methoden implementiert. Was noch auffällt ist, dass nur öffentliche Methoden angezeigt werden. Möchte man sich auch die privaten Methoden anzeigen lassen, so muss man der Methode type.GetMethods Parameter übergeben. Diese Methode verlangt BindingFlags, die man mit dem Bitweise ODER-Operator (|) miteinander verknüpfen kann. Um sich nun öffentliche und private Methoden anzeigen zu lassen, muss der Aufruf der Methode type.GetMethods folgendermaßen lauten:</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black; -moz-background-clip: -moz-initial; -moz-background-origin: -moz-initial; -moz-background-inline-policy: -moz-initial;">
<pre style="margin: 0px;"><span style="color: #2b91af;">MethodInfo</span>[] methodInfos = type.GetMethods(<span style="color: #2b91af;">BindingFlags</span>.Public | <span style="color: #2b91af;">BindingFlags</span>.NonPublic | <span style="color: #2b91af;">BindingFlags</span>.Instance);</pre>
</div>
<p>Dieser Aufruf gibt an, dass man alle öffentlichen und nicht-öffentlichen Instanz-Methoden bekommen möchte. Instanz-Methoden bedeutet, dass es sich um Methoden handelt, die nur mit dem Instanzieren eines Objektes zur Verfügung stehen. D.h. es werden keine static-Methoden angezeigt. Es gibt noch weitere BindingFlags, auf die ich hier allerdings nicht weiter eingehen möchte.</p>
<p>Das ganze könnte man natürlich noch in der Form weiter treiben, dass man sich die kompletten Methodensignaturen mit Modifizierer, Schlüsselwort und Rückgabewert anzeigen lässt:</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">SecondTestClass</span> secondTestClass = <span style="color: blue;">new</span> <span style="color: #2b91af;">SecondTestClass</span>();</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span>[] assemblies = <span style="color: #2b91af;">Thread</span>.GetDomain().GetAssemblies();</pre>
<pre style="margin: 0px;">    <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> assembly <span style="color: blue;">in</span> assemblies) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name, assembly.GetName().Version));</pre>
<pre style="margin: 0px;">        <span style="color: blue;">if</span> (assembly.GetName().Name.Equals(<span style="color: #a31515;">"TestProject"</span>)) {</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">Type</span>[] types = assembly.GetTypes();</pre>
<pre style="margin: 0px;">            <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> type <span style="color: blue;">in</span> types) {</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">MethodInfo</span>[] methodInfos = type.GetMethods(<span style="color: #2b91af;">BindingFlags</span>.Public | <span style="color: #2b91af;">BindingFlags</span>.NonPublic
                                                                               | <span style="color: #2b91af;">BindingFlags</span>.Instance
<pre style="margin: 0px;">                                                                               | <span style="color: #2b91af;">BindingFlags</span>.Static);</pre>
</pre>
<pre style="margin: 0px;">                <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> methodInfos) {</pre>
<pre style="margin: 0px;">                    <span style="color: blue;">string</span> modifier = GetModifier(info);</pre>
<pre style="margin: 0px;">                    <span style="color: blue;">string</span> keyword = GetKeyword(info);</pre>
<pre style="margin: 0px;">                    <span style="color: blue;">string</span> parameters = GetParameters(info);</pre>
<pre style="margin: 0px;">                    <span style="color: blue;">if</span> (!<span style="color: blue;">string</span>.IsNullOrEmpty(keyword)) {</pre>
<pre style="margin: 0px;">                        keyword = <span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"{0} "</span>, keyword);</pre>
<pre style="margin: 0px;">                    }</pre>
<pre style="margin: 0px;">                    <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0} {1}{2}{3}({4})"</span>, modifier, keyword
                                                                 , info.ReturnParameter, info.Name, parameters);</pre>
<pre style="margin: 0px;">                }</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px;">}</pre>
<pre style="margin: 0px;"><span style="color: blue;">private</span> <span style="color: blue;">static</span> <span style="color: blue;">string</span> GetModifier(<span style="color: #2b91af;">MethodInfo</span> info) {</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (info.IsPublic) <span style="color: blue;">return</span> <span style="color: #a31515;">"public"</span>;</pre>
<pre style="margin: 0px;">    <span style="color: blue;">return</span> <span style="color: #a31515;">"private"</span>;</pre>
<pre style="margin: 0px;">}</pre>
<pre style="margin: 0px;"><span style="color: blue;">private</span> <span style="color: blue;">static</span> <span style="color: blue;">string</span> GetKeyword(<span style="color: #2b91af;">MethodInfo</span> info) {</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (info.IsStatic) <span style="color: blue;">return</span> <span style="color: #a31515;">"static"</span>;</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (info.IsVirtual) <span style="color: blue;">return</span> <span style="color: #a31515;">"virtual"</span>;</pre>
<pre style="margin: 0px;">    <span style="color: blue;">return</span> <span style="color: blue;">string</span>.Empty;</pre>
<pre style="margin: 0px;">}</pre>
<pre style="margin: 0px;"><span style="color: blue;">private</span> <span style="color: blue;">static</span> <span style="color: blue;">string</span> GetParameters(<span style="color: #2b91af;">MethodInfo</span> info) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">StringBuilder</span> parameters = <span style="color: blue;">new</span> <span style="color: #2b91af;">StringBuilder</span>();</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">ParameterInfo</span>[] parameterInfos = info.GetParameters();</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (parameterInfos != <span style="color: blue;">null</span>) {</pre>
<pre style="margin: 0px;">        <span style="color: blue;">foreach</span> (<span style="color: #2b91af;">ParameterInfo</span> parameterInfo <span style="color: blue;">in</span> parameterInfos) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">if</span> (!<span style="color: blue;">string</span>.IsNullOrEmpty(parameters.ToString())) {</pre>
<pre style="margin: 0px;">                parameters.Append(<span style="color: #a31515;">", "</span>);</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">            parameters.AppendFormat(<span style="color: #a31515;">"{0} {1}"</span>, parameterInfo.ParameterType.Name, parameterInfo.Name);</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: blue;">return</span> parameters.ToString();</pre>
<pre style="margin: 0px;">}</pre>
</div>
<p style="text-align: center;"><img class="size-full wp-image-287 aligncenter" title="4" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/4.png" alt="4" width="605" height="443" /></p>
<p><span style="color: #ffffff;">.</span></p>
<p>Nicht immer ist bereits während der Entwicklung bekannt, aus welchem Assembly man Informationen benötigt. Sodass die Methode Thread.GetDomain().GetAssemblies() nicht verwendet werden kann. Eine weitere Möglichkeit an ein Assembly zu kommen ist es direkt zu laden. Dafür bietet die Klasse Assembly die statische Methode Assembly.LoadFrom zum Laden von Assemblies. Statt dem Laden des Assembly über die aktuelle Domäne, wird das nächste Beispiel so modifiziert, dass die TestProject-Assembly direkt aus einem Verzeichnis heraus geladen wird.</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black; -moz-background-clip: -moz-initial; -moz-background-origin: -moz-initial; -moz-background-inline-policy: -moz-initial;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span> assembly = <span style="color: #2b91af;">Assembly</span>.LoadFrom(<span style="color: #a31515;">@"c:\TestProject.dll"</span>);</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (assembly != <span style="color: blue;">null</span>) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name, assembly.GetName().Version));</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Type</span>[] types = assembly.GetTypes();</pre>
<pre style="margin: 0px;">        <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> type <span style="color: blue;">in</span> types) {</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">MethodInfo</span>[] methodInfos =</pre>
<pre style="margin: 0px;">            type.GetMethods(<span style="color: #2b91af;">BindingFlags</span>.Public | <span style="color: #2b91af;">BindingFlags</span>.NonPublic | <span style="color: #2b91af;">BindingFlags</span>.Instance |</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">BindingFlags</span>.Static);</pre>
<pre style="margin: 0px;">            <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> methodInfos) {</pre>
<pre style="margin: 0px;">                <span style="color: blue;">string</span> modifier = GetModifier(info);</pre>
<pre style="margin: 0px;">                <span style="color: blue;">string</span> keyword = GetKeyword(info);</pre>
<pre style="margin: 0px;">                <span style="color: blue;">string</span> parameters = GetParameters(info);</pre>
<pre style="margin: 0px;">                <span style="color: blue;">if</span> (!<span style="color: blue;">string</span>.IsNullOrEmpty(keyword)) {</pre>
<pre style="margin: 0px;">                    keyword = <span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"{0} "</span>, keyword);</pre>
<pre style="margin: 0px;">                }</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0} {1}{2}{3}({4})"</span>, modifier, keyword, info.ReturnParameter,</pre>
<pre style="margin: 0px;">                info.Name, parameters);</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px;">}</pre>
</div>
<p align="center">
<div id="attachment_288" class="wp-caption aligncenter" style="width: 687px"><img class="size-full wp-image-288" title="5" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/5.png" alt="5" width="677" height="419" /><p class="wp-caption-text">Man sieht, dass die gleiche Ausgabe wie bereits im vorherigen Beispiel generiert wird.</p></div>
<p style="text-align: center;">
<h2><span style="color: #ffffff;">.</span></h2>
<h2>Interaktionen</h2>
<p>Nachdem die Vorgehensweise nun klar sein sollte, möchte ich nun den eigentlich interessanten Teil demonstrieren. Wie kann man nun auf Methoden usw. zugreifen und wie können Werte aus einem Type gelesen und veränder werden? Wenn man die Methode SecondTestClass.Addition(int a, int b) aufrufen möchte, dann braucht man die Methodeninformationen von dem Type SecondTestClass aus dem TestProject-Assembly. D.h. man muss wie in den obigen Beispielen auch über alle Typen iterieren, um an den gesuchten Type SecondTestClass zu kommen. Hat man den gesuchten Type gefunden, so kann man sich von diesem alle Methoden in einem Array zurückgeben lassen. Dieses Array kann dann wieder ganz normal durchlaufen und nach der gewünschten Methode durchsucht werden. Hat man die Methode gefunden so muss zuerst eine Instanz von dem Objekt erzeugt werden und dafür gibt es den Activator, der die statische Methode CreateInstance beinhaltet. Als Parameter übergibt man ihr den Type von dem eine Instanz erzeugt werden soll. Jetzt kann man über die MethodInfo die Methode aufrufen. Dazu muss der Methode die Instanz des Objektes übergeben werden, von welchem die Methode aufgerufen werden soll und zusätzlich noch ein Array von Objekten, welche die zu übergebenen Parameter beinhaltet. Hat die Methode keine Parameter, so muss null übergeben werden. Dabei muss der Rückgabewert in den erwarteten Wert gecastet werden. Das ganze könnte dann folgendermaßen aussehen:</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black; -moz-background-clip: -moz-initial; -moz-background-origin: -moz-initial; -moz-background-inline-policy: -moz-initial;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span> assembly = <span style="color: #2b91af;">Assembly</span>.LoadFrom(<span style="color: #a31515;">@"c:\TestProject.dll"</span>);</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (assembly != <span style="color: blue;">null</span>) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name, assembly.GetName().Version));</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Type</span>[] types = assembly.GetTypes();</pre>
<pre style="margin: 0px;">        <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> type <span style="color: blue;">in</span> types) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">if</span> (type.Name.Equals(<span style="color: #a31515;">"SecondTestClass"</span>)) {</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">MethodInfo</span>[] methodInfos = type.GetMethods();</pre>
<pre style="margin: 0px;">                <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> methodInfos) {</pre>
<pre style="margin: 0px;">                    <span style="color: blue;">if</span> (info.Name.Equals(<span style="color: #a31515;">"Addition"</span>)) {</pre>
<pre style="margin: 0px;">                        <span style="color: blue;">object</span> obj = <span style="color: #2b91af;">Activator</span>.CreateInstance(type);</pre>
<pre style="margin: 0px;">                        <span style="color: blue;">int</span> value = (<span style="color: blue;">int</span>)info.Invoke(obj, <span style="color: blue;">new</span> <span style="color: blue;">object</span>[] { 2, 5 });</pre>
<pre style="margin: 0px;">                        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0}(2 + 5) = {1}"</span>, info.Name, value);</pre>
<pre style="margin: 0px;">                    }</pre>
<pre style="margin: 0px;">                }</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px;">}</pre>
</div>
<p align="center">
<div id="attachment_289" class="wp-caption aligncenter" style="width: 543px"><img class="size-full wp-image-289" title="6" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/6.png" alt="6" width="533" height="143" /><p class="wp-caption-text">Aufruf der Methode Addition</p></div>
<p style="text-align: center;">
<p><span style="color: #ffffff;">.</span></p>
<p>Bei statischen Methoden kann das Erzeugen der Instanz weggelassen werden und für den Parameter des Instanz-Objektes kann null übergeben werden.</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black; -moz-background-clip: -moz-initial; -moz-background-origin: -moz-initial; -moz-background-inline-policy: -moz-initial;">
<pre style="margin: 0px;"><span style="color: blue;">if</span> (type.Name.Equals(<span style="color: #a31515;">"TestClass"</span>)) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">MethodInfo</span>[] methodInfos = type.GetMethods(<span style="color: #2b91af;">BindingFlags</span>.NonPublic | <span style="color: #2b91af;">BindingFlags</span>.Static);</pre>
<pre style="margin: 0px;">    <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> methodInfos) {</pre>
<pre style="margin: 0px;">        <span style="color: blue;">if</span> (info.Name.Equals(<span style="color: #a31515;">"CalcPassword"</span>)) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">string</span> value = (<span style="color: blue;">string</span>)info.Invoke(<span style="color: blue;">null</span>, <span style="color: blue;">new</span> <span style="color: blue;">object</span>[] { <span style="color: #a31515;">"myPassword"</span> });</pre>
<pre style="margin: 0px;">            <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0}(\"myPassword\") = {1}"</span>, info.Name, value);</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">}</pre>
</div>
<p align="center">
<div id="attachment_290" class="wp-caption aligncenter" style="width: 687px"><img class="size-full wp-image-290" title="7" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/7.png" alt="7" width="677" height="155" /><p class="wp-caption-text">Das übergebene Passwort wurde über eine statische Methode gehashed.</p></div>
<p style="text-align: center;">
<p><span style="color: #ffffff;">.</span></p>
<p>Um den Wert einer Property auszulesen, muss anstelle von MethodInfo die PropertyInfo verwendet werden. Aus dieser kann dann über die Methode PropertyInfo.GetValue der aktuelle Wert zurückgegeben werden. Da in diesem Beispiel nach dem Instanzieren des Objektes noch keine Werte zugewiesen wurden, sind die Properties leer.</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black; -moz-background-clip: -moz-initial; -moz-background-origin: -moz-initial; -moz-background-inline-policy: -moz-initial;">
<pre style="margin: 0px;"><span style="color: blue;">static</span> <span style="color: blue;">void</span> Main(<span style="color: blue;">string</span>[] args) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Assembly</span> assembly = <span style="color: #2b91af;">Assembly</span>.LoadFrom(<span style="color: #a31515;">@"C:\ TestProject.dll"</span>);</pre>
<pre style="margin: 0px;">    <span style="color: blue;">if</span> (assembly != <span style="color: blue;">null</span>) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: blue;">string</span>.Format(<span style="color: #a31515;">"-- {0} ver.: {1}"</span>, assembly.GetName().Name, assembly.GetName().Version));</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Type</span>[] types = assembly.GetTypes();</pre>
<pre style="margin: 0px;">        <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> type <span style="color: blue;">in</span> types) {</pre>
<pre style="margin: 0px;">            <span style="color: blue;">if</span> (type.Name.Equals(<span style="color: #a31515;">"TestClass"</span>)) {</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);
<pre style="margin: 0px;"><span style="color: blue;">                object</span> obj = <span style="color: #2b91af;">Activator</span>.CreateInstance(type);</pre>
</pre>
<pre style="margin: 0px;">                <span style="color: #2b91af;">PropertyInfo</span>[] propertyInfos = type.GetProperties(<span style="color: #2b91af;">BindingFlags</span>.Public | <span style="color: #2b91af;">BindingFlags</span>.NonPublic
                                                                                      | <span style="color: #2b91af;">BindingFlags</span>.Instance);</pre>
<pre style="margin: 0px;">                <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> propertyInfos) {</pre>
<pre style="margin: 0px;">                    <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0} = {1}"</span>, info.Name, info.GetValue(obj, <span style="color: blue;">null</span>));</pre>
<pre style="margin: 0px;">                }</pre>
<pre style="margin: 0px;">            }</pre>
<pre style="margin: 0px;">        }</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.ReadLine();</pre>
<pre style="margin: 0px;">}
<img class="aligncenter size-full wp-image-373" title="8" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/8.png" alt="8" width="517" height="203" /></pre>
</div>
<p><span style="color: #ffffff;">.</span></p>
<p>Wenn man nun die Properties mit Werten belegen möchte, so muss man analog zur Methode PropertyInfo.GetValue, die Methode PropertyInfo. SetValue verwenden.</p>
<div style="background: white none repeat scroll 0% 0%; font-family: Courier New; font-size: 10pt; color: black; -moz-background-clip: -moz-initial; -moz-background-origin: -moz-initial; -moz-background-inline-policy: -moz-initial;">
<pre style="margin: 0px;"><span style="color: blue;">if</span> (type.Name.Equals(<span style="color: #a31515;">"TestClass"</span>)) {</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"  ++ {0}"</span>, type.Name);</pre>
<pre style="margin: 0px;">    <span style="color: #2b91af;">PropertyInfo</span>[] propertyInfos = type.GetProperties(<span style="color: #2b91af;">BindingFlags</span>.Public | <span style="color: #2b91af;">BindingFlags</span>.NonPublic
                                                                          | <span style="color: #2b91af;">BindingFlags</span>.Instance);</pre>
<pre style="margin: 0px;">    propertyInfos[0].SetValue(obj, <span style="color: #a31515;">"Peter"</span>, <span style="color: blue;">null</span>);</pre>
<pre style="margin: 0px;">    propertyInfos[1].SetValue(obj, <span style="color: #a31515;">"Müller"</span>, <span style="color: blue;">null</span>);</pre>
<pre style="margin: 0px;">    propertyInfos[2].SetValue(obj, <span style="color: #a31515;">"myPassword"</span>, <span style="color: blue;">null</span>);</pre>
<pre style="margin: 0px;">    <span style="color: blue;">foreach</span> (<span style="color: blue;">var</span> info <span style="color: blue;">in</span> propertyInfos) {</pre>
<pre style="margin: 0px;">        <span style="color: #2b91af;">Console</span>.WriteLine(<span style="color: #a31515;">"    ++ {0} = {1}"</span>, info.Name, info.GetValue(obj, <span style="color: blue;">null</span>));</pre>
<pre style="margin: 0px;">    }</pre>
<pre style="margin: 0px;">}
<img class="aligncenter size-full wp-image-374" title="9" src="http://blog.aztec-project.org/wp-content/uploads/2009/06/9.png" alt="9" width="477" height="167" /></pre>
</div>
<p><strong> </strong></p>
<h2><span style="color: #ffffff;">.</span></h2>
<p><span style="color: #ffffff;">.<br />
</span></p>
<h2>Wozu braucht man jetzt aber nun die Reflection?</h2>
<p>Nun ja, die meisten Programme werden Reflection wohl nie benötigen oder verwenden. Es wird vorallem bei dem Entwickeln von Debuggern, Interpretern, TestTools, Logger oder auch für O/R-Mapper verwendet. Es wird also überall dort gebraucht, wo während der Laufzeit nicht klar ist, welche Assemblies verwendet werden und welche Informationen man aus diesen lesen oder manipulieren möchte.</p>
<h2><span style="color: #ffffff;">.</span></h2>
<h2>kurze Zusammenfassung:</h2>
<p>Wie man gesehen hat, ist der Aufbau einer Assembly mit ihren Typen hierarchisch. Um zb. an einen Parameter einer Methode eines Types zu kommen, braucht man zuerst das Assembly welches den Typ enthält. Von diesem lässt man sich ein Array von Typen geben. Von dem gesuchten Typ lässt man sich dann ein Array von Methoden geben. Von der gesuchten Methode holt man sich nun ein Array von allen Parametern. Dieses Array kann dann nach dem gewünschten Parameter durchsucht werden.</p>
<p>Es gibt natürlich für all diese Methoden auch Methoden, mit denen man sich den Type, die Methodeninformation, die Parameterinformation, usw. direkt mit der entsprechenden Bezeichnung holen kann. Ich wollte hier nur auf die Hierarchie aufmerksam machen.</p>
<p>Wie man in diesen Beispielen erkennen konnte, ist es möglich, ohne Probleme auf alle Informationen von einem Assembly zuzgreifen, sich die Types zu laden, dessen öffentliche und auch nicht-öffentliche Werte zu lesen und zu manipulieren und auch Methoden auszuführen. Man kann sich auch lokale Werte anzeigen lassen, in dem man sich den MethodBody des Objekte geben lässt, von denen man die lokalen Werte ermitteln möchte.</p>
]]></content:encoded>
			<wfw:commentRss>http://blog.aztec-project.org/2009/06/24/wozu-dient-reflection/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Dynamisches Laden von Assemblies</title>
		<link>http://blog.aztec-project.org/2009/06/05/dynamisches-laden-von-assemblies/</link>
		<comments>http://blog.aztec-project.org/2009/06/05/dynamisches-laden-von-assemblies/#comments</comments>
		<pubDate>Fri, 05 Jun 2009 19:57:46 +0000</pubDate>
		<dc:creator>Thomas Christian</dc:creator>
				<category><![CDATA[How-To]]></category>
		<category><![CDATA[Assembly]]></category>
		<category><![CDATA[C#]]></category>
		<category><![CDATA[HowTo]]></category>
		<category><![CDATA[Reflection]]></category>

		<guid isPermaLink="false">http://blog.aztec-project.org/?p=50</guid>
		<description><![CDATA[In diesem Blog-Post möchte ich kurz erläutern, wie ich mit dem Problem des dynamischen Ladens von Assemblies umgegangen bin. Vor einiger Zeit stand ich vor dem Problem, dass ich zur Laufzeit Assemblies austauschen wollte. Grund dafür war, dass ich eine Host-Applikation hatte, die Plug-Ins verwendet. Jetzt wollte ich bestehende Plug-Ins während der Laufzeit austauschen oder [...]]]></description>
			<content:encoded><![CDATA[<p>In diesem Blog-Post möchte ich kurz erläutern, wie ich mit dem Problem des dynamischen Ladens von Assemblies umgegangen bin.</p>
<p>Vor einiger Zeit stand ich vor dem Problem, dass ich zur Laufzeit Assemblies austauschen wollte. Grund dafür war, dass ich eine Host-Applikation hatte, die Plug-Ins verwendet. Jetzt wollte ich bestehende Plug-Ins während der Laufzeit austauschen oder neue Plug-Ins hinzufügen ohne die Host-Applikation zu beenden. Dazu hatte ich ein Verzeichnis in dem sich, außer den Plug-Ins, alle Assemblies befanden. Die Plug-Ins selbst befanden sich in einem eigenen Unterverzeichnis. Selbst wenn die Plug-Ins von der Host-Applikation nicht mehr verwendet wurden, war es nicht möglich, diese Assemblies zu löschen.</p>
<p>Im Normalfall ist es so, dass sobald eine Assembly von einer Applikation verwendet wird, eine Referenz auf diese existiert. Diese Referenz wird leider erst gelöscht, wenn die ganze Applikation beendet wird.</p>
<p>Bei dem dynamischen Laden von Assemblies wird die zu ladende Assembly geöffnet, ausgelesen und dann geschlossen. Aus den ausgelesenen Bytes wird dann mittels Reflection eine Assembly im Arbeitsspeicher erzeugt. Auf die lokale Assembly hängt somit keine Referenz und es ist möglich diese zu löschen.</p>
<p>Meine Umsetzung sieht folgendermaßen aus:</p>
<div style="font-family: Courier New; font-size: 10pt; color: black; background: white;">
<pre style="margin: 0px;"><span style="color: #2b91af;">    2</span> <span style="color: blue;">public</span> IList&lt;Plugin&gt; GetPlugins(<span style="color: blue;">string</span> assemblyName) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    3</span>     Assembly assembly;</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    4</span>     IList&lt;Plugin&gt; pluginList = <span style="color: blue;">new</span> IList&lt;Plugin&gt;();</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    5</span>     <span style="color: blue;">try</span> {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    6</span>         <span style="color: blue;">byte</span>[] byteAssembly = File.ReadAllBytes(assemblyName);</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    7</span>         assembly = Assembly.Load(byteAssembly);</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    8</span>     } <span style="color: blue;">catch</span> (Exception ex) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">    9</span>         log.Error(ex);</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   10</span>     }</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   11</span>     <span style="color: blue;">try</span> {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   12</span>         <span style="color: blue;">if</span> (assembly != <span style="color: blue;">null</span>) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   13</span>             Type[] assemblyTypes = assembly.GetTypes();</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   14</span>             <span style="color: blue;">foreach</span> (Type assemblyTyp <span style="color: blue;">in</span> assemblyTypes) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   15</span>                 <span style="color: blue;">if</span> (<span style="color: blue;">typeof</span>(Plugin).IsAssignableFrom(assemblyTyp)) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   16</span>                     plugin = (Plugin)assembly.CreateInstance(assemblyTyp</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   17</span>                                                                 .FullName);</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   18</span>                     <span style="color: blue;">if</span> (plugin != <span style="color: blue;">null</span>) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   19</span>                         pluginList.Add(plugin);</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   20</span>                     }</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   21</span>                 }</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   22</span>             }</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   23</span>         }</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   24</span>     } <span style="color: blue;">catch</span> (ReflectionTypeLoadException ex) {</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   25</span>         log.Error(ex);</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   26</span>     }</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   27</span>     <span style="color: blue;">return</span> pluginList;</pre>
<pre style="margin: 0px;"><span style="color: #2b91af;">   28</span> }</pre>
</div>
<p><a class="a2a_dd addtoany_share_save" href="http://www.addtoany.com/share_save#url=http%3A%2F%2Fblog.aztec-project.org%2F2009%2F06%2F05%2Fdynamisches-laden-von-assemblies%2F&amp;title=Dynamisches%20Laden%20von%20Assemblies"><img src="http://blog.aztec-project.org/wp-content/plugins/add-to-any/share_save_171_16.png" width="171" height="16" alt="Share"/></a> </p>]]></content:encoded>
			<wfw:commentRss>http://blog.aztec-project.org/2009/06/05/dynamisches-laden-von-assemblies/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>

