原文:
关于VS2008和VS2010中.sln文件(解决方案文件)的介绍MSDN中只有英文版,现翻译供读者参考,希望对您有帮助。水平有限文中难免有错误,欢迎大家指正,拜谢~
解决方案就是Visual Studio中用来组织项目的结构。他的作用类似于VB6.0中的Windows程序组(.vbg)文件以及VC6.0中的项目工作区(.dsw)文件。解决方案将项目的状态信息保存在.sln文件(该文件是基于文本,而且是可共享的)和.suo文件(该文件是二进制的,而且是解决方案中可选的)中。更多关于.suo的信息请查看。
如果你的VSPackage由于在.sln文件中被引用而加载,那么编译环境调用读取.sln文件。
.sln文件包含基于文本的信息,编译环境通过这些信息找到并且加载他引用到的持久数据和VSPackages项目。当用户打开解决方案的时候编译环境通过包含在.sln文件中的preSolution,Project,postSolution信息加载解决方案,包含在解决方案中的项目以及任何和解决方案有联系的持久信息。
每个项目的文件都包含额外的信息,编译环境通过读取这些信息把项目中所需的项目补充完整。数据持久层是受项目控制的;所以数据一般不包含在.sln文件中。尽管你可以有意的将项目信息写入.sln文件中。更多的信息请参考和。
解决方案文件内容
.sln的文件都是由几部分如图所示的下列代码构成的,他被保存在解决方案浏览器的底部。更多有关解决方案注释章节的信息,参见。
1: Project("{F184B08F-C81C-45F6-A57F-5ABD9991F28F}") = "Project1", "Project1.vbproj", "{8CDD8387-B905-44A8-B5D5-07BB50E05BEA}"2: EndProject3: Global4: GlobalSection(SolutionNotes) = postSolution5: Name1 = Note26: Issue1 = N7: Text1 = This is a shared note.8: Name2 = Note39: Issue2 = N10: Text2 = This is also a shared note. The second of two shared notes.11: NumNotes = 212: EndGlobalSection13: GlobalSection(SolutionConfiguration) = preSolution14: ConfigName.0 = Debug15: ConfigName.1 = Release16: EndGlobalSection17: GlobalSection(ProjectDependencies) = postSolution18: EndGlobalSection19: GlobalSection(ProjectConfiguration) = postSolution20: {8CDD8387-B905-44A8-B5D5-07BB50E05BEA}.Debug.ActiveCfg = Debug|.NET21: {8CDD8387-B905-44A8-B5D5-07BB50E05BEA}.Debug.Build.0 = Debug|.NET22: {8CDD8387-B905-44A8-B5D5-07BB50E05BEA}.Release.ActiveCfg = Release|.NET23: {8CDD8387-B905-44A8-B5D5-07BB50E05BEA}.Release.Build.0 = Release|.NET24: EndGlobalSection25: GlobalSection(ExtensibilityGlobals) = postSolution26: EndGlobalSection27: GlobalSection(ExtensibilityAddIns) = postSolution28: EndGlobalSection29: EndGlobal
编译环境执行如下顺序的工作去加载解决方案
1. 编译环境读取.sln文件的全局语句并且处理所有的标记为preSolution的语句。在上文的例子中,有这样的声明:
1: GlobalSection(SolutionConfiguration) = preSolution2: ConfigName.0 = Debug3: ConfigName.1 = Release
当编译环境读取到GlobalSection(‘name’)标记的时候,他通过注册表将名字映射到VSPackage中。关键字name应该在注册表的[HKLM/<Application ID Registry Root>/SolutionPersistence]目录下。关键字的默认值是写入VSPackage目录的Package GUID (REG_SZ)中的。
2. 环境加载VSPackage的时候,为接口调用VSpackage的QueryInterface,并且通过文件中的数据调用ReadSolutionProps方法,所以VSPackage能够存储数据。编译环境对每个有preSolution的地方重复做上面那样的操作。
3. 编译环境循环遍历项目的模块。那好,这里有一个项目(让我们来分析一下)
1: Project("{F184B08F-C81C-45F6-A57F-5ABD9991F28F}") = "Project1",2: "Project1.vbproj", "{8CDD8387-B905-44A8-B5D5-07BB50E05BEA}"3: EndProject
上面这段代码中包含独一无二的项目全局唯一标识符(GUID)以及项目类型的全局唯一标识符(GUID)。这些信息被编译环境用来寻找项目文件爱你或者那些属于解决方案的文件,还有那些任何项目需要的特定的VSPackage。项目的全局唯一标识符(GUID)被传递到用来加载和项目有关的特殊的VSPackage,然后这个项目被载入。在上面的代码中被载入的VSPackage是Visual Basic。
每个项目中都存在一个唯一的项目实例ID,只有这样这个项目才能被其他解决方案中需要他的项目使用。理想状态下,如果解决方案和项目在源代码控制之下,项目的路径应该和解决方案的路径是相关的。当解决方案被第一次加载时,项目文件不能够存在于用户的电脑中。将项目文件存放在和解决方案文件相关的服务文件中,通过这个方法将相对简单的找到项目文件或者将项目文件复制到用户的电脑中。然后编译环境复制并加载剩下的项目所需文件。
4. 根据项目的.sln文件中包含的信息编译环境加载每一个项目文件。项目本身很好的填充项目的层次而且加载任何嵌入的项目。
5. 然后编译环境执行被标记为postSolution的那些全局语句。下面就是上文代码片段中的一个例子。
1: GlobalSection(SolutionNotes) = postSolution2: Name1 = Note23: Issue1 = N4: Text1 = This is a shared note.5: Name2 = Note36: Issue2 = N7: Text2 = This is also a shared note. The second of two shared notes.8: NumNotes = 29: EndGlobalSection
正如上面的这种情况,在.sln文件中有两个共享的note.Note1没有包含在文件中,因为他标记为不共享。他的内容放在.suo文件中。语句中的每个有名字值组成的条目和每个note是一一对应的。更多有关实现解决方案Note的内容请查看。
6. 最后.sln文件执行完毕,解决方案在解决方案浏览器中显示出来,并等待用户的操作。
如果有关项目的任何VSPackage加载失败, 方法将被调用而且每个解决方案中的项目将尝试取消在加载过程中的改变。如果解析发生错误,尽可能的收集有关解决方案文件的资料而且编译环境显示一个对话框警告这个解决方案已经被破坏了。
当解决方案被保存或者被关闭的时候方法将被调用并传到层次结构检查解决方案中是否存在那些需要保存到.sln文件中的改变。在中传递到QuerySaveSolutionProps的一个空值表明这个信息将被保存。如果值不为空表明保存的信息是特殊的项目,这取决于接口
如果有信息被保存接口将被指向的方法调用。然后方法将被编译环境调用,这个方法用来检验与来自IPropertyBag的接口对应的名字值并且将相关信息写入.sln文件中。
为了检验来自IPropertyBag接口的信息是否被保存SaveSolutionProps和WriteSolutionProps对象被编译环境递归调用,直到所有的更改被写进.sln文件。用这种方法可以确保解决方案的信息可以保存并且下次可以顺利打开解决方案。
每个加载的VSPackage都被列举用来查看是否有需要保存到.sln文件的东西。只有在加载的时候才需要注册码。编译环境知道所有的加载的包,因为当解决方案保存的时候他们就被记录下来了。
只有.sln包含preSolution和postSolution中的条目。在.suo文件中没有相似的语句,因为解决方案需要正确的被加载。.suo文件包含用户的可选项,比如个人的注释,这些是不需要被共享或者被源码控制的。