Click here to Skip to main content
15,867,308 members
Articles / Programming Languages / Visual Basic
Article

Runtime Compilation (A .NET eval statement)

Rate me:
Please Sign up or sign in to vote.
4.83/5 (49 votes)
13 Jan 20063 min read 333.8K   7.2K   94   65
An article on a new library, allowing easy, language-independent access to runtime compilation.

Image 1

Introduction

There are several languages out there (primarily interpreted languages) which provide a convenient eval statement or function. These usually accept simple fragments of code and execute them on the fly, often used for mathematical expressions. .NET leaves this useful feature out of its BCL (Base Class Library), and buries access to its compiler under incomplete documentation. I was working on a project of my own when I realized that I needed some form of runtime compilation, and went off on a tangent to build an easy-to-use library (also adding full XML documentation).

Using the code

The library is centered around the Eval class, which contains the static methods that access the compiler.

The methods provided are as follows:

C#
public static AssemblyResults CreateAssembly(ICodeCompiler compiler,
                                          string assemblySource,
                                          CompilerParameters options,
                                          Language language);
public static TypeResults CreateType(ICodeCompiler compiler,
                                     string typeSource,
                                     string typeName,
                                     CompilerParameters options,
                                     Language language);
public static MethodResults CreateMethod(ICodeCompiler compiler,
                                         string methodSource,
                                         string methodName,
                                         CompilerParameters options,
                                         Language language);

public static AssemblyResults CreateVirtualAssembly(ICodeCompiler compiler,
                                         string assemblySource,
                                         bool debug,
                                         Language language,
                                         params string[] references);
public static TypeResults CreateVirtualType(ICodeCompiler compiler,
                                         string typeSource,
                                         string typeName,
                                         Language language,
                                         bool debug,
                                         params string[] references);
public static MethodResults CreateVirtualMethod(ICodeCompiler compiler,
                                         string methodSource,
                                         string methodName,
                                         Language language,
                                         bool debug,
                                         params string[] references);

Each method compiles the source provided, loads the result into memory, and returns a reference to an object wrapping the final result (along with any warnings generated by the compiler). If any errors are generated during the compilation, a CompilationException is thrown, which contains the compiler's errors. The "Virtual" group of methods stores the results of compilation directly to memory, using a pre-generated set of CompilerParameters.

The Language class and its subclasses provide information about an individual language to the Eval class, exposing methods that generate various language-specific segments of the code which are necessary in the CreateType and CreateMethod methods of the Eval class.

The result classes each expose the actual result through a property named by the type of the result (respectively through the Assembly, Type, and Method properties), as well as the collection of warnings generated during compilation (through the Warnings property). In addition, each class provides indirect access to some of the more commonly used abilities of its underlying result (each of which acts as described in its XML documentation).

The AssemblyResult class provides the following method:

C#
public TypeResults GetType(string typeName, bool throwOnError);

The TypeResults class provides the following methods:

C#
public MethodResults GetMethod(string name);
public MethodResults GetMethod(string name, 
                            params Type[] paramTypes);

public object Instantiate(params object[] parameters);

Finally, the MethodResults class provides the following method:

C#
public object Invoke(params object[] parameters);

This method automatically creates an instance of the declaring class on which to invoke the wrapped method, allowing non-static methods to be invoked as if they were static.

Points of interest

Microsoft included nearly everything that I needed to write this library in their standard .NET Framework, except the final wrapper. If you find that you need something beyond the capabilities of my library, the System.CodeDom.Compiler and System.Reflection namespaces should provide most of what you need.

To do

More than a few things, most likely... Currently only one update comes to mind. If there is any demand for it, I will make the results of my runtime compilation interface unloadable from memory by using AppDomains (inspired by and probably borrowing from the DevX article Dynamically Executing Code in .NET).

This is my first article posted, so I'd appreciate any feedback. E-mail feedback is accepted, but I'd prefer feedback through the message board provided by the CodeProject.

History

  • 12th January, 2006
    • Updated code (again) to fix a bug with referenced libraries and an issue with in-memory compilation. Both features should now work as intended.
  • 6th December, 2003
    • Refactored code (added Language class and subclasses, updated Eval class) in order to make the Eval class work fully with languages other than C#, added VB.NET example, and updated the article to reflect these changes.
  • 14th January, 2003
    • Updated code to fix minor bugs with referenced libraries.
  • 11th January, 2003
    • Zip files finally corrected (extensive paths removed).
  • 2nd January, 2003
    • Example project, screenshot, and To do added, a few bugs fixed.
  • 31st December, 2002:
    • Article first posted.
    • ~5:31 PM EST: <pre> blocks fixed (Sorry!).

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here


Written By
Web Developer
United States United States
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions

 
QuestionSome Query Pin
yashvi_rsy1-Oct-14 18:55
yashvi_rsy1-Oct-14 18:55 
GeneralMy vote of 5 Pin
BillWoodruff2-Jun-13 2:05
professionalBillWoodruff2-Jun-13 2:05 
GeneralMy vote of 5 Pin
Sau00229-Apr-13 4:12
Sau00229-Apr-13 4:12 
GeneralGood job! Pin
Shane Story3-Sep-10 2:37
Shane Story3-Sep-10 2:37 
GeneralMy vote of 5 Pin
Prasenjit Purohit6-Jul-10 19:28
Prasenjit Purohit6-Jul-10 19:28 
Generalnice article Pin
niuzj30-Nov-09 20:30
niuzj30-Nov-09 20:30 
GeneralByRef Arguments Pin
Member 28930215-Sep-08 3:33
Member 28930215-Sep-08 3:33 
GeneralPassing an array to the Method Pin
MisterT9917-Oct-07 6:12
MisterT9917-Oct-07 6:12 
Generalcompile error on MC++ VS003 Pin
clee005-Jul-06 12:25
clee005-Jul-06 12:25 
NewsDynamic Code Pin
Primadi6-Jun-06 0:03
Primadi6-Jun-06 0:03 
QuestionReference COM-Object? Pin
Daniel Hilgarth5-May-06 0:08
Daniel Hilgarth5-May-06 0:08 
QuestionInclude current namespace? Pin
Mellmountain24-Mar-06 0:21
Mellmountain24-Mar-06 0:21 
AnswerRe: Include current namespace? Pin
Mellmountain29-Mar-06 2:07
Mellmountain29-Mar-06 2:07 
QuestionCompile multiple language source at same time? Pin
FocusedWolf4-Mar-06 6:07
FocusedWolf4-Mar-06 6:07 
Hi, i'm not sure really how to approach this situation.

I want the option to do stuff where one script can call another, and it doesn't matter which .net language they are writtin in cause i'm calling them with reflection anyway.

So maybe theirs so compiler option flag? hmm.

Also how can multiple source files be handled?...zzz confussing to do anything but compile an assembly and call it.
QuestionUsing runtime compilation in a multi-threaded application Pin
x-cubed22-Feb-06 9:32
x-cubed22-Feb-06 9:32 
AnswerRe: Using runtime compilation in a multi-threaded application Pin
Eric Astor22-Feb-06 9:47
Eric Astor22-Feb-06 9:47 
AnswerRe: Using runtime compilation in a multi-threaded application Pin
x-cubed22-Feb-06 11:11
x-cubed22-Feb-06 11:11 
GeneralYaaay... Pin
Ross Presser13-Jan-06 3:36
Ross Presser13-Jan-06 3:36 
GeneralReferencing external assemblies Pin
Eric Astor12-Jan-06 4:05
Eric Astor12-Jan-06 4:05 
GeneralRe: Referencing external assemblies Pin
Eric Astor12-Jan-06 4:22
Eric Astor12-Jan-06 4:22 
Generalinclude dll Pin
[ac]Sebastian12-Apr-05 5:13
[ac]Sebastian12-Apr-05 5:13 
GeneralFeature suggestion: namespaces Pin
steve willer6-Aug-04 4:37
steve willer6-Aug-04 4:37 
GeneralSpecify output directory Pin
adlionelC2-Jun-04 6:55
adlionelC2-Jun-04 6:55 
GeneralRe: Specify output directory Pin
adlionelC2-Jun-04 22:12
adlionelC2-Jun-04 22:12 
GeneralRe: Specify output directory Pin
Eric Astor3-Jun-04 0:53
Eric Astor3-Jun-04 0:53 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.