企业管理软件包含一些公共的组件,这些基础的组件在每一个新项目立项阶段就必须考虑。核心的稳定不变功能,方便系统开发与维护,也为系统二次开发提供了诸多便利。好比通用权限管理系统,通用附件管理,通用查询等组件,如果在项目开发前就准备好了这些组件,为项目如期交付提供了保证。html
在系统维护过程当中,不停的增长新的字段,通常不会当即重写系统现有的查询,但须要一种方式能够当即看到系统的更改,或者是系统现有的查询不能知足客户的需求,查询设计器就是为了解决系统查询功能的不足开发的。数据库
我考虑到了如下几种查询的方式,简单介绍它们的实现方式供参考。编程
假设客户的系统维护人员懂SQL语句,通过多年客户积累的系统,软件公司也有许多经常使用的查询语句,将这些查询语句放到一个查询功能中执行一下,便可获取数据。服务器
参考下面的SqlDbx程序的例子,在系统中能够对这个界面进行简化,只保留输入SQL语句和显示查询结果的地方。框架
为了方便最终用户分析结果数据,系统须要提供对查询结果数据的导出(Microsoft Excel),过滤,排序,分组等功能。工具
ERP系统将每一个查询保存起来,下次用户只须要敲查询编号便可看到查询结果,并对结果数据进行操做。布局
若是用户不懂SQL语句,系统考虑提供一种图形化的方式供用户设计查询。记得10年前本身在学习SQL语句的时候,是很是期待有一个智能化的工具,可让我选择要查询的数据表和字段,再设置数据表之间关联,最后就获得我需用的查询语句。系统参考了Access 的查询设计器,参考一下经典的Access的查询设计界面:学习
微软Office套件中Access的查询设计器通过多年的发展,应该是有足够的理由相信这个界面是最容易让非IT人士接受的查询设计界面。只须要用鼠标选一下要查询的表,再选择要显示的字段,系统自动产生相应的SQL语句。开发工具
图纸化查询设计这种功能会常常出如今报表设计器中,报表设计器通常都会附加一个图形化的查询设计工具,咱们能够在那里找到它的界面原型,通过简化后变成ERP系统的查询设计工具。ui
若是SQL语句或是表关联也不能知足数据的查询要求,系统能够考虑增长存储过程支持,以知足更复杂的查询需求。在ERP的财务报表中,各类财务统计报表的确至关的繁琐,非用存储过程不可。咱们须要考虑好,如何将参数传递到存储过程当中,显示存储过程的返回结果就可知足这种需求。为此,设计以下的查询语句:
EXEC spRpt_OrderAmtTotal %1, %2
存储过程的后面两个参数是占位符号,表示须要给此存储过程传递参数。因而,还须要设计一种参数映射,设定参数的类型,长度,运行此查询时,将用户输入的值替换到上面的两个占位置符中,传递到存储过程当中。
存储过程的定义已经定义它的参数类型,系统运行查询时,系统须要将上面的%1的值所表明的值转换成存储过程须要的参数类型,好比上图中的%1 所表明的Date From,系统须要用户的输入值进行强制类型转换为日期时间类型,再传递到存储过程当中获取返回结果。
用程序代码写查询能够分为二种状况,单据查询,列表查询。单据查询只须要继承原有的单据编辑窗体,设置窗体不可编辑,这样就完成了单据查询。列表查询是指须要用户输入一种或几种过滤条件,根据过滤条件获得查询结果。
单据查询的代码很是简单,只是重设几个属性便可,参考下面的代码例子。
public partial class SalesContractEnquiry : Foundation.Sales.Entry.SalesContractEntry { public CostSheetEnquiry() { InitializeComponent(); this.SupportAdd = false; this.SupportDelete = false; this.SupportEdit = false; }
程序代码中禁用了单据的新增,删除,编辑操做,这样单据窗体变也了查询窗体。
列表查询的界面参考以下,界面中上方是过滤条件输入控件,下面显示查询结果。对查询结果能够导出Excel,过滤,分组,或是以图表的方式呈现查询结果。
报表设计器分二个组件,一个是报表设计,另外一个是报表显示,前者是design,后者是render。市面上有许多报表设计工具,我比较熟悉是的水晶报表(Crystal Report)和报表服务(SQL Server Reporting Services)。刚毕业那时还接触到开源的报表设计器RDLC Designer,是微软报表服务的一个开源实现。工做中接触水晶报表的时间比较多,个人技术总监写的一个水晶报表查看器,全是反射调用作成的报表查看器,不依赖于水晶报表的版本,发现水晶报表对.NET的支持至关稳定,从Crystal Report for Visual Studio 2008 runtime到如今的Crystal Report 13.10,几乎没有改动代码就能够完美的运行技术总监的代码,水晶报表是.NET报表制做的工业标准。
能力有限,实在没有精力去维护一份报表设计器代码,报表设计选用SAP的水晶报表设计器,这个工具备不少年没有更新,目前能找到的最新版本是Crystal Report 200 SP2。
因此这一部分的内容测重于报表呈现(Render),力求设计一个完美的报表阅读器。做为核心功能,列出以下需求:
水晶报表分三种类型的参数,经过调节这三个数值来改变水晶报表的数据。定义如下枚举:
public enum ReportFieldType { [DisplayText("Selection Formula"), StringValue("0")] SelectionFormula = 0, [DisplayText("Formula Field"), StringValue("1") ] FormulaField = 1, [DisplayText("Parameter"), StringValue("2")] Parameter = 2 }
以上三种种方式的定义,与下面的水晶报表截图能够更清楚的了解它们的含义:
0 表示表记录选择条件,1 表示公式,2表示参数。 经过这三种方式,能够定义以下表结构:
运行报表时,先根据上面的参数表生成控件,获取控件的值,传递到水晶报表中,即完成了参数传递。
这样开发的好处是技术支持人员可独立制做和开发报表,不须要开发部专门为每一个界面开发参数选择界面。
如何只设计一份报表,却可让它同时以三种本地化语言显示报表。经历了如下几种方案演化。
1) 定义一个LanguageCode的公式或参数,运行时由系统传入到报表中来,表示当前要显示的语言。水晶报表中每一个文字标签都用公式表示,公式Ccy的例子参考以下:
if LanguageCode=1 then "Currency" else if LanguageCode=2 then “貨幣” else "货币"
从公式中能够看到,1表示英语,2表示繁体中文,其它的数字表示简体中文。这样根据传入的LanguageCode的值,来显示文本标签的值,实现了报表多语言显示。
2) 使用.NET Localization方案,定义三种资源文件,分别是Resource.en-us.resx,Resource.zh-cn.resx,Resource.zh-tw.resx,编译这个程序集后会生成三个带语言标识的子程序集,.NET运行时会根据语言查找相应的资源文件,调用语言配对的资源。关键的代码调用以下所示:
System.Threading.Thread.CurrentThread.CurrentUICulture = CultureInfo.CreateSpecificCulture(cultureName);
System.Threading.Thread.CurrentThread.CurrentCulture = new CultureInfo(cultureName);
3) 翻译水晶报表控件TextObject
这个方法是借助于水晶报表.NET API,找到水晶报表中须要翻译的对象,通常是TextObject,将它翻译成对应的本地化语言再显示,这种方案深受报表开发人员喜好,开发报表时只须要按照标准英文版的作法,当须要显示为其它语言时,自动转化为本地化语言。可参考下面的代码片断以加深了解。
IEnumerator reportObjectEnumerator = (IEnumerator)ReflectionHelper.InvokeMethod(reportObjects, "GetEnumerator"); while (reportObjectEnumerator.MoveNext()) { try { object reportObject = reportObjectEnumerator.Current; object objectKind = ReflectionHelper.GetPropertyValue(reportObject, "Kind"); string objectKindName = Enum.GetName(objectKind.GetType(), objectKind); if (string.CompareOrdinal(objectKindName, "TextObject") == 0 || string.CompareOrdinal(objectKindName, "FieldHeadingObject") == 0) { string text = (string)ReflectionHelper.GetPropertyValue(reportObject, "Text"); if (!string.IsNullOrEmpty(text)) { string translatedText = ComponentCommon.TranslateText(text, false); if (string.CompareOrdinal(text, translatedText) != 0) ReflectionHelper.SetPropertyValue(reportObject, "Text", translatedText); } } } catch { } }
反射调用一个对象要实现foreach的效果,须要调用GetEnumerator方法。
写一个水晶报表运行库的检测程序,它能够检测安装的水晶报表的版本。水晶报表控件所有用反射方法调用,参考下面的代码例子,这样就实现了编译时不依赖于水晶报表版本的效果,部署时更加灵活方便。
ReflectionHelper.SetPropertyValue(this._crystalReportViewer, "ReportSource", this._report); ReflectionHelper.InvokeMethod(this._crystalReportViewer, "Update"); ReflectionHelper.InvokeMethod(this._crystalReportViewer, "Zoom", new System.Type[] { typeof(int) }, new object[] { 1 }); ReflectionHelper.InvokeMethod(this._crystalReportViewer, "Zoom", new System.Type[] { typeof(int) }, new object[] { this._zoomFactor });
引入一个替代报表(Alternate report)的概念,将标准报表嵌入到程序集或放置在标准报表目录中,系统也支持一个替代报表的路径选项,系统读取报表时优先查找替代报表路径中的报表文件,找到则用替代报表显示,不然继续在标准报表路径中查找报表。由于两种报表放置在不一样的路径中,因此相同的报表文件也不会相互影响和覆盖,解决了客户定制报表与系统报表取舍的难题。替代(Alternate)的概念还用在物料清单的替代物料中,生产发料时当物料不够发料,能够去查找替代物料发料,比如咱们口渴了能够喝汽水,也能够选择喝凉茶。
窗体设计器在ERP/MIS领域应用的至关普遍,Visual Studio自己就是一个设计精良的窗体设计器。金蝶ERP的BOS系统所有依赖于它的窗体设计器,在此基础上作数据绑定和插件开发。微软的InfoPath也是一个自定义表单工做,经常使用来作OA系统的自定义表单。刚毕业那会也很是喜欢研究form designer re-host技术,惋惜一直没有找到技术突破点,也不知道这样的设计是否合理。曾经接触过《像搭积木同样作软件》这本书,全书讲解的就是以窗体设计器为基础,作表达式求值,作事件绑定和属性绑定,不须要编码而开发企业管理软件。
然而这种美好的技术终究是一种幻觉,Visual Studio 仍旧是最流行的开发工具,窗体开发仍旧是企业管理软件开发的重点。窗体设计器所产生的代码,只有一小部分间接的用途。我没有深刻接触这个主题,但就我所知道的知识点列举以下。
1 窗体设计器能够生成CS/VB/Xml 三种代码格式。NET动态编译技术已经很成熟,直接调用.NET API就能够将CS/VB代码编译为程序集,在这里我选择第三种格式,我并不须要用窗体设计器彻底开发一个新功能,那样涉及到数据绑定,主从数据等一系列难题,我只须要设计器产生Xml格式,运行时我能够优先加载这个自定义布局,因此Xml格式足矣。
2 要设计的窗体对象是现有的系统功能。用户可能要修改界面控件的布局或是外观。实施过程当中,看到不少客户喜欢将控件标成红色或蓝色以加快阅读速度,然而当满屏幕都是花花绿绿的控件,反而会下降阅读理解的速度。另外一个就是控件的布局,一些用户不须要的选项卡,控件能够经过窗体设计器隐藏。
3 窗体设计器最重要的地方是能够修改界面控件的查找。参考下图。
窗体设计器能够修改Customer No中查找按钮的过滤条件,这一重要的功能大大减轻了开发人员的负担。
4 窗体设计器不能够用来彻底从新开发一个新功能,从界面设计到数据绑定,再到数据验证,以及数据之间的勾稽
引用,这些功能的实现不可能经过鼠标拖放控件就完成。即便经过大量的二次开发,像金蝶那样作成BOS,它的可扩展性和灵活性仍那以控制。好比单价 * 数量=金额,要作到输入单价或数量时,自动计算金额。BOS要考虑的内容项太多,我终究是完全放弃这种开发模式,只用到窗体设计器的一小部分功能:控件外观与布局修改,控件查找定制。
工做流实现的四大基础功能:通知提醒,批核,计划任务,调用自定义代码。
为了实现这个目标,基于微软的.NET WF,作了如下工做以达到上述目的。
活动是工做流中的代码执行单元,一个工做流定义自己也是一个活动。根据业务须要,定义了以下活动库:
文档批核活动,发送消息活动,发送邮件活动,调用.NET 代码活动,执行数据库查询活动,报表生成活动。
根据业务的须要,定义如下几种业务类型:
单据类:单据保存,单据更新,单据删除,单据新建。
业务类:文档送审,文档批核,业务过账,业务完成,业务取消。
任务计划:在预约时间执行工做流
MSDN 中提供了rehost工做流设计器的例子,直接把它拿来参考,添加自定义活动组件和自定义工做流类型,再将工做流设计器输出格式保存为XOML,便可完成工做流设计器的绝大部分功能。
这里比较复杂的一点是自定义条件表达式,须要一个与对象表达式求值工具。Code Project中有许多条件表达式的例子,难点在于如何将业务单据的状态与工做流挂接。
系统须要一个可视化的工具查看每一个流程当前正在运行的结点,MSDN中有例子可参考。
.NET提供的基础服务,建立一个工做流状态保存数据库和一个工做流跟踪数据库。
工做流与系统业务部分的交互,专门开一个独立的端口用于数据的读写。
企业应用中的做业调度,常见的操做以下:邮件提醒和告警,执行文件传输操做,建立复杂报表。
系统分为两种计划任务调度器,一种是基于SQL Server Agent实现,定时执行SQL语句,另外一种是基于Quartz框架库实现,用.NET代码开发任务调度程序。
SQL Server Job是一个按期执行脚本的对象,给它加一个时间选项便可完成基于SQL Server Agent的计划调度程序。
这个界面会依据控件值的不一样,生成不一样参数的SQL Server Job,参考下面的程序代码片断:
private void btnOk_Click(object sender, EventArgs e) { if (Schedule == null) Schedule = new SQLschedule(); Schedule.name = txtName.Text.Trim(); if (chkEnabled.Checked) Schedule.enabled = 1; else Schedule.enabled = 0; if (cmbScheduleType.SelectedIndex == 1) { Schedule.freq_type = 1; // One-time Schedule.active_start_date = dtOneTimeOccurDate.Value; Schedule.active_end_date = new DateTime(9999, 12, 31); Schedule.active_start_time = dtOneTimeOccurTime.Value; Schedule.active_end_time = new DateTime(2000, 1, 1, 23, 59, 59); } else { if (cmbFrequencyOccurs.SelectedIndex == 0) { Schedule.freq_type = 4; // Daily Schedule.freq_interval = (int) numFrequencyRecurs.Value; } if (cmbFrequencyOccurs.SelectedIndex == 1) { Schedule.freq_type = 8; // Weekly int freq_interval = 0; if (chkFrequencyRecursSun.Checked) freq_interval += 1; if (chkFrequencyRecursMon.Checked) freq_interval += 2; if (chkFrequencyRecursTue.Checked) freq_interval += 4;
这个工具来源于Code Project上的一篇文章,能够用SQL Agent Job Editor尝试找到它。
这是由Java项目转化过来的著名项目,用.NET代码重定了它的逻辑。看一个最简单的任务计划的源代码。
public class HelloJob : IJob { private static ILog _log = LogManager.GetLogger(typeof(HelloJob)); public HelloJob() { } public virtual void Execute(IJobExecutionContext context) { // Say Hello to the World and display the date/time _log.Info(string.Format("Hello World! - {0}", System.DateTime.Now.ToString("r"))); } }
Quartz.NET处理好了关于任务计划调度方面的各个细节,很容易上手,官方提供的例子也至关丰富。