ReportView如何使用构造SQL语句带参数的存储过程创建报表以及为rdlc传递参数

环境:Windows2003 R2 + .NET2.0 + IIS6 + MS SQL2000 + VS2005(en)

最近要做报表,想使用MS的ReportView控件来实现,但后来的使用过程中才发现ReportView控件问题多多不够灵活,特别是在使用存储过程上,我看MSDN与网上的文章也都不够详细,所以我把我这几天研究的结果列表给大家看看。

先来看一个表结构

Create Table Test
(
 PID varchar(100),
 PName varchar(100),
 PQty int,
 PPrice money
)

创建一个存储过程

CREATE PROCEDURE procTest
 @condition varchar(1000)
AS
 DECLARE @sql varchar(8000)
 SET @sql = 'SELECT PID, PName, PQty FROM Test '
 IF @condition <> ''
  SET @sql = @sql + @condition
 EXEC(@sql)

这里需要说明的是,我看MSDN和网上的例子使用的都是直接SELECT语句带参数的存储过程,但很多时候都需要通过构造SQL语句进行查询,所以我这里主要讲的是如何使用EXEC这样构造SQL语句的存储过程。

建一 ASP.NET WebApplication,在App_Code目录(没有的话可以右键选择 Add ASP.NET Folder新建此目录)里增加一个新项,并选择DataSet,填写DataSet名,点击确定。

在打开的窗口中右键选择Add再选择TableAdapter,弹出TableAdapter连接向导对话框,连接数据库(这个过程略),选择 Use existing stored procedrues(已存在的存储过程),点击下一步,选择procTest,直接点完成,这样一个TableAdapter算是建好了。

如果那个存储过程直接是SELECT PID, PName, PQty FROM Test 那建好的TableAdapter就会直接把PID, PName, PQty 当作列显示出来,但用的是EXEC所以要手工将PID, PName, PQty 在右键->新建Column里把这几个加进去,加完后,存盘,然后新建一个Report文件,新建完后,在Report窗口左侧会把刚刚建的TableAdapter显示出来,把PID, PName, PQty 列拖到Report窗口中,布局自己搞定,存盘。

打开Default.aspx文件(自己新建一个都没所谓),在Toolbox里的Data中将ReportView控件拖到窗口上,选择刚刚新建的Report文件,然后再拖一个Textbox和一个Button,双击Button打开Code窗口填写代码

protected void Button1_Click(object sender, EventArgs e)
{
 string condition = "";
 if ( Textbox1.Text.Trim() != "" )
 {
  string condition = "PID = '" + Textbox1.Text.Trim() + "'";
 }

 //清除数据源原参数
ObjectDataSource1.SelectParameters.Clear();
//增加数据源参数
ObjectDataSource1.SelectParameters.Add("condition", condition);
 //让ReportView重新加载数据源
ReportViewer1.LocalReport.Refresh();
}

好,这样一来,报表便写成了,按下Ctrl+F5健,便可看到效果了

报表中使用最多的就是列合计了,那如何增加列合计呢?

首先使用=Sum(Fields!PQry.Value)是肯定不行的,如果这样使用,你会发现在报表中出现一个#Error的样字,刚开始我也是百思不得其解,按照MSDN的解释使用Sum是可以进行合计的啊,=Fields!PQry.Value可以显示,为什么=Sum(Fields!PQry.Value)却显示#Error呢,后经多次调试才发现问题出在TableAdapter的列中,回顾刚刚的过程,那些列都是自己手工加进去的而不是根据SQL自动创建的,所以Sum认不出Fields!PQry.Value,只能是出现一个#Error,看来只能另外想办法了。

我刚开始认为那个Report.rdlc文件上的Textbox都是可以获取的,获取后人为的把合计值加进去,可不管我用什么方法,FindByControl里总是null,看来只能想其他办法,我想应该用参数方式将合计值传过去,重新打开Report.rdlc文件,在窗口的灰色空白处右键选择Report Paraments后可以增加参数,随后随便新建一个参数paraQty,选择允许空值与允许null值,这样参数建好了,点击确定关闭窗口并拖一个Textbox到窗口上,Textbox内容填写=Parameters!paraQty.Value,就看怎么传值了。可MSDN上也没讲什么,网上问的人也不少,但都没有解决办法,郁闷的要死,又不想重新使用其他控件,看来只能细心研究了,经过多次努力,终于找到解决办法,就是使用SetParameters方法将参数传递过去,代码如下(代码加在ReportViewer1.LocalReport.Refresh();这句上面)

Microsoft.Reporting.WebForms.ReportParameter p = new Microsoft.Reporting.WebForms.ReportParameter("paraQty", "123");
ReportViewer1.LocalReport.SetParameters(new Microsoft.Reporting.WebForms.ReportParameter[] { p });

重新按下Ctrl+F5,在报表中就会看到123的字样了,好,现在只要将123替换成合计值就完成了,当然得到合计值的办法很多,比如直接从ObjectDataSource1(为ReportView控件选择rdlc文件时自动创建的数据源控件)中获取即可,值得一提的是不能使用Compute("Sum(PQty)", "true")获得,因为PQty列的数据类型是String的,所以使用Sum时会报错,只能通过累加的方式取得,完整代码如下

protected void Button1_Click(object sender, EventArgs e)
{
 string condition = "";
 if ( Textbox1.Text.Trim() != "" )
 {
  string condition = "PID = '" + Textbox1.Text.Trim() + "'";
 }

 //清除数据源原参数
ObjectDataSource1.SelectParameters.Clear();
//增加数据源参数
ObjectDataSource1.SelectParameters.Add("condition", condition);
//获取数据源
DataTable dt = ((DataView)ObjectDataSource1.Select()).Table;
//合计相关列
int qty_sum = 0;
for (int i = 0; i < dt.Rows.Count; i++)
{
qty_sum += int.Parse(dt.Rows[i]["PQty"].ToString());
}
//设置报表参数
Microsoft.Reporting.WebForms.ReportParameter p = new Microsoft.Reporting.WebForms.ReportParameter("paraQty", qty_sum.ToString());
ReportViewer1.LocalReport.SetParameters(new Microsoft.Reporting.WebForms.ReportParameter[] { p });
//让ReportView重新加载数据源
ReportViewer1.LocalReport.Refresh();
}
至此便大功告成了,但这并不是最佳解决方案,只能暂时解决问题而已。

你可能感兴趣的:(ReportView如何使用构造SQL语句带参数的存储过程创建报表以及为rdlc传递参数)