第三,不管是哪种类型的报表,发展到到目前为止,均出现了多种不同的版本。客户的服务器也可能会是不同版本的Server系统。微软的组件兼容性好,但同时,也有很霸道的地方。比如,Visual Studio 2012明确只支持Windows 7及以上的OS,SQL Server 2012不再支持SQL Server 2000格式的数据库。对于这里要用到的ReportViewer控件,对于SQL Server 2012,并没有出对应的新版本的控件,所以,如果要浏览Report Builder 3 for SQL Server 2012设计出来的RDLC报表,呈现此报表的控件,仍然是SQL Server 2008版本的组件,如下版本所示
<%@ Register Assembly="Microsoft.ReportViewer.WebForms, Version=10.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a"
Namespace="Microsoft.Reporting.WebForms" TagPrefix="rsweb" %>
再以Crystal Report为例子,它的runtime就有好多个版本,同一个版本又有x86和x64的区别
最新版本的是for visual studio 2010, 主版本号13。这么多runtime要能同时支持,只有用反射,根据客户端安装的版本,来创建不同版本的runtime对象。或是以简单的方法,规定要用指定版本的runtime,其它的则不支持。
例子代码如下所示,可供您参考实现
Assembly engineAssembly = Assembly.Load(GetLongAssemblyName("CrystalDecisions.CrystalReports.Engine", version)); Assembly sharedAssembly = Assembly.Load(GetLongAssemblyName("CrystalDecisions.Shared", version)); Type printingConverterType = engineAssembly.GetType("CrystalDecisions.CrystalReports.Engine.PrintingConverter"); 动态参数支持RDLC不支持parameter,RDL才支持。微软对此的解释是,RDLC在取数时,要进行参数化处理,取出来的数据,是经过参数过滤的。RDL支持参数,传递参数时,使用像这样的代码片段
ReportParameter[] parm = new ReportParameter[1]; parm[0] = new ReportParameter("deptno", txtDeptno.Text); reportViewer.ShowCredentialPrompts = false; reportViewer.ProcessingMode = Microsoft.Reporting.WebForms.ProcessingMode.Remote; reportViewer.ServerReport.ReportServerUrl = new System.Uri("http://localhost/ReportServer"); reportViewer.ServerReport.ReportPath = "/EnterpriseSolution/SalesOrder"; reportViewer.ServerReport.SetParameters(parm); reportViewer.ServerReport.Refresh();参数处理的奥妙,如下图所示,在报表对话框中设计参数,运行时会动态创建控件,供用户输入值,再由运行时,将此值传递到报表中,达到简化编码的目的,经过这样处理,同样,不需要编写任何C#代码而达到传递参数值的目的。
字段格式,是最终要传化为参数的格式,数据类型表示控件要使用的样式。对于类型转化,微软提供了这样的方法
object obj = ReflectionHelper.GetPropertyValue(control, targetProperty); object converted = Convert.ChangeType(obj, type); ReflectionHelper.SetPropertyValue(entity, arr[1], converted);用文字来描述这段代码的含义。以TextBox为例子,第一句取到TextBox的Text属性,它是个字符串,如果type要求是的数字,则第二句把它转化为字符串,第三句则应用反射将值传递给报表对象。在Visual Studio调试时,字符串有双引号,数字则没有,虽然看起来它们都是object,值也很相似,但属于不同类型,在运算时如果没有转化,会出错。