过去一年的总结
在过去一年的时间里,跟着现在这家公司(简称B公司)的领导和同事学到很多东西,我觉得在这一年中主要是从技术与沟通方面得到了提高:
1.技术方面
1.1之前一直做客户端的开发,所以对那些脚本语言用的比较少,在这里加深了JQuery,JS的使用。
1.2对Ajax的异步使用多了一些了解。
1.3学会了如何使用SQL监控,学会了一些优化SQL语句的方法,当然这都是在实际项目中积累的经验,比较宝贵。
1.4学会了使用客户端与网页端的互相调用、嵌套来开发应用系统。
1.5了解了跨域请求的处理方法。
2.沟通方面
2.1搞IT的其实是一个服务角色,所以与用户沟通的时候要找到自己的定位。
2.2不管用户的要求是否合理,沟通的时候都不要充满火药味儿。
2.3互相尊重很重要,尊重对方也会使对方尊重自己。
为什么离开
来到B公司之前,在一家上市集团公司(简称A公司)做公司内部的应用系统开发,整整服务了3年,到目前为止,离开A公司已经一年十个月了,在这 期间跟之前的领导一直还是有联系,当然,这没别的意思,主要是毕竟在那里工作了3年,也是我参加工作以来服务时间最长的一家公司,多少还是有一些情谊存在 的,在我进入到B公司之后,之前A公司的领导找到了我,目的是叫我利用业余时间帮忙开发一套应用系统,我听了二话没说就答应了,为什么这么爽快呢?有如下 几个原因:
1.当时我离开的比较匆忙,可能交接的不是那么仔细,这一点我一直觉得过意不去。
2.在我离开A公司的前一天,也就是周四(因为领导是HK那边的,每周四下午都会回去HK),领导去到我办公的地方对我说:“xxx,那我走 了...”,后面的我就没听清楚了,因为当时有 点受宠若惊的感觉,他一个高级领导,完全可以不用把我这个小喽啰当回事儿的,就算是打招呼也应 该是我去给他打招呼才对,这个事情让我一直心存感激,我觉得在当时来说他 给我面子了。
3.在给我办离职事宜上,很多细节方面领导都为我考虑到了,比如说我离开的突然,按照人事部要求的话时间达不到,提前离开的话需要扣工资,但是他给我把签批日期提前了,虽然是一些小事情,但是让人心里暖暖的。
上面貌似扯了很多废话,言归正传,我的原意是在我能力范围内免费帮忙开发,因为我本身是搞这一行的,技术不是问题,就是多花点时间而已,但是项目完 成之后领导还是给了我意想不到的报酬,在这里要感谢一下他们的照顾,当然这个项目也是几经波折才完成,毕竟我是业余的,不在同一个地方,沟通很不方便(说 实话,通过这一次我也学到了很多)。
在这期间,领导跟我谈话几次,目的是希望我能回去工作;因为之前有很多项目都是我跟进的,领导可能是考虑到我比较熟悉一点吧,我考虑了很久,最终还是答应了,所以才向B公司这边提出离开的要求;说实话,在这之前我从没想过我还会回去,真的没想到,真的是世事难料...
回去之后的工作目标
1.尽量统一开发语言,之前的项目存在多种开发语言,比如VB6,C++,VFP,ASP,Delphi,C#,至今还有DOS操作系统的程序,何 必呢?难道要我精通各种开发语言吗?或者说 每一门开发语言找一个对应的开发者吗?该砍的砍、该升级的升级,不然不管是谁都玩 不下去。
2.后续的项目以Web结构为主,便于更新;客户端为辅,涉及到跟硬件通信的功能采用客户端与Web结合来完成。
3.将众多的小程序集成到一个框架中,不要搞的那么分散(目前存在很多的小程序、小补丁),不要再不停的“打补丁”。
4.搭建一个好的源代码管理平台,不管是TFS还是SVN都可以。
最近一个Web应用程序
我这个人就是这样,看见别人好的东西就想学习过来,刚来到B公司的时候,看见公司用的很多功能都比较高大上,我的学习方法是这样的:先了解这些 高大上的功能是如何实现的,不懂就请教前辈,这里又得感谢一下我的师傅了,感谢他不吝赐教、悉心指导;了解之后就自己创建一个项目试着搞一个Demo出 来,在这个过程中遇到不懂的就再次请教别人,直到自己搞懂为止,那么到了这个时候这东西已经“属于”我自己的了,我已经装进脑袋里面去了,所以从进入B公 司开始我就没有停止过学习。不管学到什么知识,一定要运用起来才行,只有在实际项目中才能学到更多、才能找出自己的不足,所以下面的项目就是我刚进入B公 司就开始着手开发的,借鉴了一些前辈们的东西,但不是直接COPY,只有图片和CSS样式是用的原来的,因为本人美工技术实在太差,想PS出来一个 LOGO图片就搞了很多次,PS出来的结果连自己看着就不爽,所以就不强求了,引用了之前的一些图片和样式,这里值得说明的是:现有公司的项目和数据库结构我们开发人员都有,这也是领导批准的,因为我们有时候会在家里办公,所以会将数据库结构和程序复制在本地以便创建真实的开发环境和调试环境。
除了图片以外其它的我都按照我自己的设计开 发的,包括数据库设计和项目架构设计,后续我会抽时间将这个项目继续完善,主要是针对制造行业,现在只有几个基础的功能,也可以说是制造行业里面基本上都 会用到的几个功能,比如条码管控与打印之类的,根据实际的需求可以灵活扩展,以下是部分截图:
描述:
条码规则就是用来控制条码的格式,对于现在的工厂来说,条码化管控是最常见的,那么不同的产品有不同的条码格式要求,每一个客户也都有自己的特殊要求,所以这个条码规则也是非常重要的,在这之前我的做法如下:
在数据库中根据当前年月日+流水号来自动生成,这样本来也能解决问题,但是如果需要扩展的时候就歇菜了,只要客户要求条码格式稍微改变一下格 式,那么我们需要改变的东西是相当的多,甚至可能因此带来很多严重的问题,因为格式已经固定不能灵活变更,所以几乎没有扩展的可能性。
#p#
鉴于此,现在的做法如下:
定义一个条码规则,包含条码的计算进制、需要屏蔽的字母或字符、条码前缀、条码后缀、条码总长度、流水号长度等等参数,在生成条码的时候选择此 规则即可,如果有新的格式要求,那只需要针对新的格式要求新增一个条码规则即可,对原有的数据结构完全没有影响,灵活多变、扩展容易!
描述:
生成条码属于一个比较关键的功能,有时候数据量可能比较大,例如某一个工单的数量是20000,那也就是说用户生成条码的时候一次性就要在数据 库产生20000笔记录,根据我个人的经验来说,这种事物不能直接交由用户来操作,防止数据库被拖死;我的做法是不管用户提交多少数量,在服务器中将用户 提交的请求保存起来,服务器定时去处理这些请求,每次按照时间排序处理最先提交的一笔请求,如果用户的请求中出现不合法的数据,则服务器直接将该请求驳回 并通知到用户,反之则完成相关的操作,这样一来可以避免用户直接操作大量的数据而导致系统变得缓慢或者数据库出现死锁之类的问题。
描述:
工艺路线方面这里值得说一下,采用了gooflow设计,当然gooflow只有UI部分,后台都是另外设计的。
支持多节点工艺路线,这里需要设置条件,例如:当前我们前面有两条路,路线A通向北京,路线B通向上海,我们现在站在路口选择具体走哪一条路? 这个时候就需要条件了,我们现在的目的是去北京看故宫,那很显然已经匹配到了条件,说明我们应该走路线A;系统会根据具体的情况来寻找应该走哪一条线,这 个逻辑比较复杂。
描述:
动态报表,可以根据不同的条件动态生成控件,实现查询功能,这样可以省掉很多的报表,简单的查询功能用这个足以。
描述:
维修功能其实是一个比较复杂的业务,根据不同的维修动作而产生不同的数据结构,比如说维修动作中包含“简单维修”,“更换物料”,“报废”等选 项,像简单维修就比较容易处理,只需按照正常的工艺路线操作并记录修理过程即可,但是更换物料就不一样了,需要记录更换了什么物料?原始料号是什么?更换 上去的料号是什么?包括Vendor,DateCode等信息;一般生产管理的应用系统中,客户都会要求有详细的维修记录和报表,这是一个重点。
描述:
以下都是调用Api实现的,具体的逻辑都在Api里面实现,Api采用工厂模式设计,根据不同的需求产生不同的数据结构,这些页面没有写一句后台代码,全是脚本。
描述:
在开发过程中,会涉及到很多的错误信息,既有存储过程中的也有程序里面的,之前我的做法是直接将报错的存储过程名称写在错误信息后面,这样一看 就知道是哪一个存储过程抛出的错误,直接就可以进行错误定位,但是发现一个问题,存储过程的名称字符串长度无法统一,导致页面布局不好控制;鉴于此,我统 一用ErrorCode来代替错误信息,针对用户来说,用户并不需要知道是哪个地方报错,系统管理员或者开发人员根据ErrorCode对照表也能快速定 位问题点,如此可以解决由于错误信息长度不统一而导致的页面布局问题,ErrorCode对照表如下图所示:
以上都是介绍了部分Web中的功能,整个项目还有一部分是客户端,客户端其实比较好理解,主要是在客户端调用了Web程序,客户端主窗体代码如下,当然只是部分代码,根据实际情况可以另行增加。
- using System;
- using System.Collections.Generic;
- using System.ComponentModel;
- using System.Data;
- using System.Drawing;
- using System.Linq;
- using System.Text;
- using System.Windows.Forms;
- using System.Security.Permissions;
- namespace MES
- {
- [PermissionSet(SecurityAction.Demand, Name = "FullTrust")]
- [System.Runtime.InteropServices.ComVisibleAttribute(true)]
- public partial class FrmMain : Form
- {
- public FrmMain()
- {
- InitializeComponent();
- }
- public WebBrowser Browser { get { return this.webBrowser1; } }
- private void FrmMain_Load(object sender, EventArgs e)
- {
- string ConfigPath = Application.StartupPath + "\\config.ini";
- if (!System.IO.File.Exists(ConfigPath))
- {
- MessageBox.Show("缺少配置文件,系统无法启动,请联系管理员!", "温馨提示", MessageBoxButtons.OK, MessageBoxIcon.Warning);
- Application.Exit();
- GC.Collect();
- return;
- }
- MES.Common.Global.ApiUrl = MES.Common.IniHelper.GetConfigValue("ApiUrl");
- MES.Common.Global.serverPath = MES.Common.IniHelper.GetConfigValue("ServerPath");
- webBrowser1.ObjectForScripting = this;
- webBrowser1.Navigate(MES.Common.Global.serverPath);
- }
- public void Print_ZPL(string Content)
- {
- System.Drawing.Printing.PrintDocument p = new System.Drawing.Printing.PrintDocument();
- string DefaultPrint = p.PrinterSettings.DefaultPageSettings.PrinterSettings.PrinterName;
- MES.Common.PrintHelper.PrintLabel(DefaultPrint, Content);
- }
- private void webBrowser1_DocumentCompleted(object sender, WebBrowserDocumentCompletedEventArgs e)
- {
- webBrowser1.ScriptErrorsSuppressed = true; //禁用脚本调试,屏蔽脚本错误提示框
- }
- /// <summary>
- /// 获取称重机的重量
- /// </summary>
- /// <param name="Com">通信COM口</param>
- /// <param name="CheckCount">循环次数</param>
- /// <returns></returns>
- public string GetWeigh(string Com,int CheckCount)
- {
- return MES.Common.WeighHelper.GetWeighData(Com, CheckCount);
- }
- /// <summary>
- /// 获取本地计算机上面的COM列表
- /// </summary>
- /// <returns></returns>
- public string GetComList()
- {
- string str = "";
- List<string> list= MES.Common.WeighHelper.GetComList();
- for (int i = 0; i < list.Count; i++)
- {
- str += list[i].ToString() + ",";
- }
- return str;
- }
- }
- }
至于如何回调客户端的函数,可以参考这篇文章:理解JavaScript回调函数