基于EasyUI的Web應用程序及過去一年的總結
過去一年的總結
在過去一年的時間里,跟著現在這家公司(簡稱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回調函數