1. “csproj文件究竟是做什么用的”
csproj文件,即C#项目文件的扩展名,是"C Sharp Project"的缩写。它主要用于为开发工具提供信息,比如我们熟悉的Visual Studio和MSBuild。Visual Studio会根据csproj中的XML定义来管理项目文件及相关数据,并执行各种操作。而MSBuild则会利用csproj文件来了解编译这个项目需要哪些依赖项、默认输出路径、Pre-Build和Post-Build需要哪些操作等。
在开发环境中,csproj的存在是至关重要的,因为它为开发工具提供了必要的信息。然而,到了运行环境中,操作系统并不关心所谓的csproj文件,只关注exe、dll、配置文件或资源文件等实际运行时需要的东西。例如,IIS这样的运行环境更不会去理会csproj的存在,因为它只需要web.config文件。
关于pdb文件是否需要放在Debug目录下才有效的问题,其实是一个误解。Debug目录只是VS的模板所默认存在的编译规则所生成的目录,我们在调试时完全可以由VS指定pdb文件存在的目录,甚至不需要VS也能使用pdb文件。
“模板”是开发环境中的重要组成部分,我们在VS中选择New Item或New Project时可以看到各种模板。这些模板是人为生成给VS用的,可以自行学习如何建立一个模板。然而,在运行环境中,是不会知道开发中用了什么模板的。
例如,为什么Web Site中无法使用ASP.NET AJAX,而Web Application就可以?这与运行时期并没有关系,而是与Web.config配置和页面上的信息有关。很多时候,排错并没有什么妙法,唯有仔细。
而“模板”在开发环境中的地位其实比csproj文件还要低。因为只要通过模板创建好内容之后,就无法说明结果和自己有什么联系了。例如使用模板创建一个AjaxControlToolkit的Extender,其中会生成一个.cs,一个.designer.cs和一个js文件,谁又能证明这三个文件不是我们手动创建的呢?这就是“开发环境”,一切都是为了提高开发效率,一切都是为了最终产生一个可执行的二进制文件。
开发环境中的最后一个成员“编译器”工作完成之后,所有开发工具便默默地退居二线。
2. 如何用C#动态编译,执行代码 CSharpCodeProvider
如何用C#动态编译、执行代码
在开始之前,先熟悉几个类及部分属性、方法:CSharpCodeProvider、ICodeCompiler、CompilerParameters、CompilerResults、Assembly。
一、CSharpCodeProvider
提供对C#代码生成器和代码编译器的实例的访问。如果要动态生成VB代码,可以使用VBCodeProvider。
CreateCompiler():获取编译器的实例。
二、ICodeCompiler
定义用于调用源代码编译的接口或使用指定编译器的CodeDOM树。每种编译方法都接受指示编译器的CompilerParameters对象,并返回指示编译结果的CompilerResults对象。
CompilerAssemblyFromSource(CompilerParameters option, string source):使用指定的编译器,从包含源代码的字符串设置编译程序集。
三、CompilerParameters
表示用于调用编译器的参数。
ReferencedAssemblies:获取当前项目所引用的程序集。Add方法为程序集添加引用。
GenerateExecutable:获取或设置一个值,该值指示是否生成可执行文件。若此属性为false,则生成DLL,默认是false。
GenerateInMemory:获取或设置一个值,该值指示是否在内存中生成输出。
四、CompilerResults
表示从编译器返回的编译结果。
CompiledAssembly:获取或设置以编译的程序集,Assembly类型。
五、Assembly
就是程序集了(不知道如何描述了)。
大致了解了以上知识之后,就可以使用C#动态的编译并执行代码了,一下是一段示例程序:
using System;
using System.Reflection;
using System.Globalization;
using Microsoft.CSharp;
using System.CodeDom;
using System.CodeDom.Compiler;
using System.Text;
namespace ConsoleApplication1
{
public class Program
{
static void Main(string[] args)
{
// 1.CSharpCodePrivoder
CSharpCodeProvider objCSharpCodePrivoder = new CSharpCodeProvider();
// 2.ICodeComplier
ICodeCompiler objICodeCompiler = objCSharpCodePrivoder.CreateCompiler();
// 3.CompilerParameters
CompilerParameters objCompilerParameters = new CompilerParameters();
objCompilerParameters.ReferencedAssemblies.Add("System.dll");
objCompilerParameters.GenerateExecutable = false;
objCompilerParameters.GenerateInMemory = true;
// 4.CompilerResults
CompilerResults cr = objICodeCompiler.CompileAssemblyFromSource(objCompilerParameters, GenerateCode());
if (cr.Errors.HasErrors)
{
Console.WriteLine("编译错误:");
foreach (CompilerError err in cr.Errors)
{
Console.WriteLine(err.ErrorText);
}
}
else
{
// 通过反射,调用HelloWorld的实例
Assembly objAssembly = cr.CompiledAssembly;
object objHelloWorld = objAssembly.CreateInstance("DynamicCodeGenerate.HelloWorld");
MethodInfo objMI = objHelloWorld.GetType().GetMethod("OutPut");
Console.WriteLine(objMI.Invoke(objHelloWorld, null));
}
Console.ReadLine();
}
static string GenerateCode()
{
StringBuilder sb = new StringBuilder();
sb.Append("using System;");
sb.Append(Environment.NewLine);
sb.Append("namespace DynamicCodeGenerate");
sb.Append(Environment.NewLine);
sb.Append("{");
sb.Append(Environment.NewLine);
sb.Append(" public class HelloWorld");
sb.Append(Environment.NewLine);
sb.Append(" {");
sb.Append(Environment.NewLine);
sb.Append(" public string OutPut()");
sb.Append(Environment.NewLine);
sb.Append(" {");
sb.Append(Environment.NewLine);
sb.Append(" return \"Hello world!\";");
sb.Append(Environment.NewLine);
sb.Append(" }");
sb.Append(Environment.NewLine);
sb.Append(" }");
sb.Append(Environment.NewLine);
sb.Append("}");
string code = sb.ToString();
Console.WriteLine(code);
Console.WriteLine();
return code;
}
}
}