摘要:
看过卢彦先生的关于web报表打印实现文章的人,一定会为里面所提供的解决方案击节叫好,本文试图给大家展现一个更灵活的打印作业流水,并具有一定的实用性。建议大家在阅读本文之前先阅读卢彦先生的两篇文章,同时本文采用了微软的wse(1.0)作为辅助工具,对此感到陌生的朋友,建议先参考一些概念性的文章,我在文章的最后列了一些参考资料和所需工具,大家可自行体会。
目录
引言
软件原理
程序实现
注意事项
总结
参考资料
引言:
wse提供了一个非常方便的功能,就是支持附件的传输,尽管我们可以采用别的方式来达到这个目的,比如直接的返回byte型的数据,但是对于大多数应用而言,直接返回一个附件,如一张图片更为实在些,请大家不要误会我文章的标题,以为创建一个webservice就可以方便地实现了打印了,我们的软件原理和最终打印的方式跟卢彦先生里提到的并无多大区别,我们只是利用了webservice的强大的穿透性,来使这个方式更为灵活,更易于应用和拓展,本文所采用的代码全部用c#编写。
软件原理:
本文采用了xml形式的数据,客户端将需要打印的数据和一些基本的参数,如图象大小,图象形式等传送给服务器端,而服务器端则根据客户端的要求生成特定的一张或多张图片返回给客户端,由客户端的打印程序统一处理,看了这个逻辑我们就可以发现:所有的业务规则完全在服务器端运做,而客户端只需要少量的代码就可以实现报表的打印。这样就避免了各种升级所带来的烦恼,当实际运用中要求增加一种或多种图表的时候,我们所需要做的只是增加或者修改服务器端的业务规则,而客户所要做的只是告诉我们要打印这种图表就可以了。
本文采用了一些简单的xml数据,仅供示范用,客户端的demo数据如下:
<?xml version="1.0" encoding="utf-8"?>
<root printtype="line" width="450" height="500" title="print demo">
<child text="1" value="100" color="black">
</child>
<child text="2" value="60" color="orange">
</child>
<child text="3" value="30" color="red">
</child>
<child text="4" value="40" color="gray">
</child>
<child text="5" value="90" color="blue">
</child>
<child text="6" value="60" color="green">
</child>
</root>
其中printtype就是客户端要求的打印类型。
程序实现:
服务器端代码
本系统也同样采用了abstract factory的设计模式,以利于服务器端方便的扩展,此处就不再赘述。
新建一个web服务项目,添加抽象基类,实现类,解析类,建成后的界面如下所示:
需要加入对microsoft.web.services命名空间的引用

其中
printbase.cs是基类
parser.cs是解析类
lineprint.cs是实现具体图象的类
基类代码:
public class printbase
{
public printbase()
{
//
// todo: 在此处添加构造函数逻辑
//
}
public virtual stream drawimage()
{
return null;
}
}
解析类代码:
public class parser
{
public parser()
{
//
// todo: 在此处添加构造函数逻辑
//
}
public static printbase createelement(dataset ds)
{
printbase pb = null;
string l_strprinttype = ds.tables["root"].rows[0]["printtype"].tostring();
switch(l_strprinttype)
{
case "line":
pb = new lineprint(ds);
break;
default:
pb = new printbase();
break;
}
return pb;
}
}
实现类的代码:
由于此处代码较长,我只贴出部分代码供参考,大家可以根据自己的实际情况进行图形的绘制。
/// <summary>
/// 重载画的规则
/// </summary>
/// <returns>图象stream</returns>
public override stream drawimage()
{
_chartsize = new sizef(float.parse(ds.tables["root"].rows[0]["width"].tostring()),
float.parse(ds.tables["root"].rows[0]["height"].tostring()));
bitmap b = new bitmap((int)_chartsize.width,(int)_chartsize.height,
pixelformat.format32bppargb);
//初始化
_graphics = graphics.fromimage(b);
//以下省略,请自行绘制
//存储返回
memorystream s = new memorystream();
b.save(s,imageformat.png);
return s;
}
我个人一直比较喜欢用dataset操作小型的xml数据,大家请按自己喜好调整,此处展示的方法将直接供web服务类调用。
web服务类的代码:
[webmethod]
public bool createimage(dataset ds)
{
bool l_bstatus = true;
try
{
printbase pb = null;
pb = parser.createelement(ds);
stream s = pb.drawimage();
soapcontext sc = httpsoapcontext.responsecontext;
sc.attachments.add(new dimeattachment("image/png",typeformatenum.mediatype,s));
}
catch
{
l_bstatus = false;
}
return l_bstatus;
}
至此,我们服务器端的代码就基本完成了,下面我们来看客户端所需要做的工作,我在此处建立了一个winform工程来做演示用,实际运用中,则采用卢彦文章里提到的usercontrol就可以了。
客户端代码: