Introduction
This is a simple application for generating the code dynamically and compiling it dynamically.
I have given two classes, GenerateClass
, which generates a class on the fly,
and RunClass
,
which compiles the generated class.
For dynamic code generation and code compilation, we have the CodeDom technology.
Before going into the sample, let me give you a short introduction to CodeDom. Some applications need the
generation and or compilation of source code at run time. The .NET Framework SDK provides
a standard mechanism called the Code Document Object Model (CodeDOM) that enables the output
of the source code in multiple programming languages at run time, based on the single model that
represents the code to render. The CodeDom namespace of .NET provides:
- A single model for rendering the source code. Therefore, we can extend the
set of supported languages.
- A way to represent source code in a language independent object model.
- Programs can be dynamically created, compiled and executed at run time.
Details
CodeDom architecture can conceptually represent most programming constructs.
However there are several limitations to the current CodeDom implementation. A
few to mention are variable declaration list, unsafe modifier, aliasing the
namespaces, nested namespaces, and others. To represent the constructs that are
not provided by CodeDom, we can make use of the �Snippet� classes. To talk about
the dynamic code generation and compilation, I have chosen the traditional
example, �Hello World�. I tried to generate a code that will look like the
following.
namespace Samples {
using System;
public class HelloWorld {
public HelloWorld() {
writeHelloWorld();
}
private void writeHelloWorld() {
System.Console.WriteLine("Hello World!!");
}
public static void Main() {
HelloWorld hw = new HelloWorld();
}
}
}
Now let us see how to generate the code. To generate the code,
we need to create IcodeGenerator
and CSharpCodeProvider
. Have
a TextWriter
to write the generated class.
string fileName = "HelloWorld.cs";
TextWriter tw = new StreamWriter(new FileStream(fileName, FileMode.Create));
ICodeGenerator codeGenerator = new CSharpCodeProvider().CreateGenerator();
CSharpCodeProvider cdp = new CSharpCodeProvider();
codeGenerator = cdp.CreateGenerator();
Declare the namespace for the class and adds the includes to the class.
CodeNamespace Samples = new CodeNamespace("Samples");
Samples.Imports.Add(new CodeNamespaceImport("System"));
Create the class.
CodeTypeDeclaration Class1 = new CodeTypeDeclaration("HelloWorld");
Samples.Types.Add(Class1);
Class1.IsClass = true;
The constructor for the created class is
CodeConstructor cc = new CodeConstructor();
cc.Attributes = MemberAttributes.Public;
cc.Statements.Add(mi1);
Class1.Members.Add(cc);
The method writeHelloWorld
can be given as :-
CodeMemberMethod Method1 = new CodeMemberMethod();
Method1.Name = "writeHelloWorld";
Method1.ReturnType = new CodeTypeReference(typeof(void));
CodeMethodInvokeExpression cs1 = new CodeMethodInvokeExpression(
new CodeTypeReferenceExpression("System.Console"),
"WriteLine", new CodePrimitiveExpression("Hello World!!"));
Method1.Statements.Add(cs1);
Class1.Members.Add(Method1);
To create an executable, we should mention the entry point of the application. The
method, Main
is given as :-
CodeEntryPointMethod Start = new CodeEntryPointMethod();
Start.Statements.Add(
new CodeSnippetStatement("HelloWorld hw = new HelloWorld();"));
Class1.Members.Add(Start);
Finally generate the source code file.
codeGenerator.GenerateCodeFromNamespace(Samples, tw, null);
Now, let us see how to compile and run the code at run time.
For this we need the ICodeCompiler
.
ICodeCompiler cc = new CSharpCodeProvider().CreateCompiler();
Set all the compiler parameters that you want. Note if you want
to compile an assembly, set the GenerateExecutable
to false
.
CompilerParameters cp = new CompilerParameters();
cp.ReferencedAssemblies.Add("system.dll");
cp.GenerateExecutable = true;
After setting the parameters, compile the code and get the results. Check
out for errors, if any.
CompilerResults cr = cc.CompileAssemblyFromFile(cp, fileName);
if(cr.Errors.HasErrors)
{
StringBuilder error = new StringBuilder();
error.Append("Error Compiling Expression: ");
foreach (CompilerError err in cr.Errors)
{
error.AppendFormat("{0}\n", err.ErrorText);
}
throw new Exception("Error Compiling Expression: " + error.ToString());
}
And the last step, get the assembly generated and create an instance
to run.
Assembly a = cr.CompiledAssembly;
_Compiled = a.CreateInstance("Samples.HelloWorld");
That's all.