Winform技术要点及案例项目开发小结

近期帮朋友开发了一个小型WINFORM的软件,即在软件窗体中显示相关表格的数据。由于长时期未操作WINFORM了,有点手生,尤其是对于EXCEL表格的操作,在没有第三方控件辅助的基础上,更感到了吃力。 但所幸功夫不负有心人, 终于让我找到了相关的解决方案。 现总结一下此次项目开发的经验教训,以便后期借鉴。


1.winform虽然目前已不再是主流的开发技术, 但对于小型的需求,仍不失为一个不错的选择,尤其是对于小型单机办公型应用程序,其界面的友好性,及相关控件操作的简易性,很适合开发小型的桌面应用程序。


2.闪屏现象解决方案。由于在窗体中加入大量图片,以及绘图等操作,在程序启动或加载某个界面时出现屏幕闪烁的问题。为了解决这种现象,可以在窗体构造函数中加入如下代码,以解决闪屏现象。其中第一行一般为程序自带代码, 后面几行主要是解决屏幕变化出现的闪烁现象,即进行一定的缓冲操作。

             InitializeComponent();

            this.DoubleBuffered = true;//设置本窗体
            SetStyle(ControlStyles.UserPaint, true);
            SetStyle(ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景.
            SetStyle(ControlStyles.DoubleBuffer, true); // 双缓冲


3.窗体容器panel的使用, 即如果需要对窗体背景图片进行分割,可以利用容器控件进行适当的操作。


4.基本控件的使用,如control,label,textbox等等。


5.datagridview控件的使用。对于没有格式要求的数据显示,datagridview及listview可以做为基本的显示控件,且可以通过调整相关的属性,如控件头部隐藏,边框隐藏,填充窗体的方式等进行简单的格式调整,但对于合并单元格等操作,但一般需要进行绘图等复杂操作,相应的代码较为复杂。

5.1 datagridview的读取数据操作。

string str = @"d:\cpcx\cpqlc.xls";//也可以通过打开文件对话框进行选择文件。

DataTable dt = ExcelUp(File); //  将excel文件读取到DATATABLE
dataGridView1.AutoGenerateColumns = true;  //设置datagridview自动增加列,即新添加的数据自动新添加列。
dataGridView1.DataSource = dt;  //设置数据源。

public DataTable ExcelUp(string fileName)
        {
            string filePath = fileName;//读取excel文件路径;

            DataTable dt = GetDataTable("Sheet1", filePath);

            return dt;

        }

protected DataTable GetDataTable(string strSheetName, string strExcelFileName)
        {
            //源的定义,即ADO的读取数据操作。

           //string strConn = "Provider=Microsoft.Jet.OleDb.4.0;" + "data source=" + Server.MapPath("ExcelFiles/MyExcelFile.xls") + ";Extended Properties='Excel 8.0; HDR=Yes; IMEX=1'";//此连接只能操作Excel2007之前(.xls)文件

           string strConn = string.Format("Provider=Microsoft.ACE.OLEDB.12.0;" + "Data Source={0};" + "Extended Properties='Excel 8.0;HDR=NO;IMEX=1';", strExcelFileName);//此连接可以操作.xls与.xlsx文件 (支持Excel2003 和 Excel2007 的连接字符串).

//备注: "HDR=yes;"是说Excel文件的第一行是列名而不是数据,"HDR=No;"正好与前面的相反。
//      "IMEX=1 "如果列中的数据类型不一致,使用"IMEX=1"可必免数据类型冲突。



特别注意

Extended Properties='Excel 8.0;HDR=yes;IMEX=1'

A: HDR ( HeaDer Row )设置

若指定值为Yes,代表 Excel 档中的工作表第一行是栏位名称

若指定值為 No,代表 Excel 档中的工作表第一行就是資料了,沒有栏位名称

B:IMEX ( IMport EXport mode )设置

IMEX 有三种模式,各自引起的读写行为也不同,容後再述:

0 is Export mode

1 is Import mode

2 is Linked mode (full update capabilities)

我这里特别要说明的就是 IMEX 参数了,因为不同的模式代表著不同的读写行为:

当IMEX=0 时为“汇出模式”,这个模式开启的 Excel 档案只能用来做“写入”用途。

当 IMEX=1 时为“汇入模式”,这个模式开启的 Excel 档案只能用来做“读取”用途。

当 IMEX=2 时为“连結模式”,这个模式开启的 Excel 档案可同时支援“读取”与“写入”用途。






            //Sql语句
            string strExcel = string.Format("select * from [{0}$]", strSheetName);

            //定义存放的数据表
            DataSet ds = new DataSet();

            //连接数据源
            OleDbConnection conn = new OleDbConnection(strConn);

            try
            {
               conn.Open();
                //适配到数据源
                OleDbDataAdapter adapter = new OleDbDataAdapter(strExcel, strConn);
                adapter.Fill(ds, strSheetName);
            }
            catch (Exception e)
            {
                throw e;
           }
            finally
            {
                conn.Close();
            }
           return ds.Tables[strSheetName];
        }


5.2 datagridview单元格绘制操作。


5.3 listview的读取数据操作。

listViewData.Items.Clear();
listViewData.Columns.Clear();
listViewData.Columns.Add("", 0, HorizontalAlignment.Center);

listViewData.Columns.Add("序号",40, HorizontalAlignment.Right);

listViewData.Columns.Add("快递单号",120, HorizontalAlignment.Center);

listViewData.Columns.Add("修改重量",80, HorizontalAlignment.Right);

var i =0;var strShuju ="";//取得的字符串
IDataObject iData = Clipboard.GetDataObject();

if (iData.GetDataPresent(DataFormats.Text)) { strShuju= (string)iData.GetData(DataFormats.Text); }

if (strShuju.IndexOf("\r\n") == -1)return;

var iShuju = strShuju.Length; //字符串总长度
 while (strShuju != "")

{

    try {

           var myItem =new ListViewItem(); myItem.SubItems.Add((++i).ToString());var strRow = strShuju.Substring(0, strShuju.IndexOf("\r\n"));// 每行字符串
           var iRow = strShuju.Substring(0, strShuju.IndexOf("\r\n")).Length;//每行字符串长度

            var iRow1 = 0; //每行字符串长度
             if (iRow == 0)

             break;

             iRow1 = iRow + 1;

             iRow = iRow + 2;

             iShuju = iShuju - iRow;

             strShuju = strShuju.Substring(iRow, iShuju);

             strRow = strRow + "\t";while (strRow !="") {var strColumn = strRow.Substring(0, strRow.IndexOf("\t"));//每段字符串
             var iColumn = strColumn.Substring(0, strRow.IndexOf("\t")).Length;//每段字符串长度
         

               if (iRow1 == 0) break;

               iRow1 = iRow1 - iColumn -1;

               strRow = strRow.Substring(iColumn +1, iRow1);

               myItem.SubItems.Add(strColumn); }

              listViewData.Items.Add(myItem);

         }

           

        catch (ArgumentOutOfRangeException)

                { break; } }


5.4 listview单元格绘制操作。


6.窗体布局的动态调整, 一般编程时,各个窗体及控件的大小及位置都是固定好了,这样也可以减少程序运行时的负担,即加快运行速度。但对于部分程序,窗体的大小需要根据运行屏幕的尺寸进行改变,这时就需要编码进行动太调整窗体大小了,一般代码如下,当然,窗体中的其它控件也可以进行类似的调整。

            Rectangle rect = Screen.GetWorkingArea(this);//获取屏幕尺寸。
            this.Height = rect.Height;//设置窗体尺寸。
            this.Width = rect.Width;


7.窗体样式的动态调整。如果需要在程序运行时独占屏幕,且不出现状态栏,最大/小化按钮,及关闭按钮等样式,可以进行以下代码编程。

if (this.WindowState == FormWindowState.Maximized)  //判断窗体是否最大化
            {
                this.FormBorderStyle = FormBorderStyle.None;// 窗体的边框及标题栏的状态
            }
            else
            {
                this.FormBorderStyle = FormBorderStyle.None;//
                this.WindowState = FormWindowState.Maximized;//
            }


8. VS平台智能提示异常问题的解决。即从开始->Microsoft Visual Studio 2008->Visual Studio Tools->Visual Studio 2008 命令提示,进入Common7\IDE,然后输入:devenv.exe /setup /resetuserdata /resetsettings或devenv.exe/resetsettings。


9.窗体切换操作代码:Forms f=new Forms();            f.Show();            this.Hide();


10.复杂表格界面的实现。由于VS自带的控件只能实现简单的图表界面,故在实现复杂界面时,如果堆积简单的LABEL,TEXTBOX等控件,则不仅费时,也加重程序运行负担,故需要借用第三方控件, 而大部分简便的控件都是收费的,如水晶报表,DevExpress,NOPI等,但是微软自已的控件dsoframer, 也可以实现类似功能,但操作可能相对复杂,即在调整界面时需要调整控件及EXCEL表格各自的相关属性,以实现项目要求。但dsoframer的使用一般先进行注册,即可以从微软下载后进行本机注册,也可以借助集成的注册程序进行注册。

10.1注册一下DSOFramer.ocx

    操作:将DSOFramer.ocx复制到C:/windows/system32目录下,

     开始->运行->regsvr32 DSOFramer.ocx , 系统会提示DSOFramer.ocx中的DllRegisterServer成功。

10.2添加DSOFramer.ocx到你的项目中,即添加COM组件到VS,即可在工具箱中发现安装成功。一般安装后需要重启,以便顺利使用。

10.3 dsoframer的主要属性, 即ActivationPolicy属性可以设置打开的文件是否处于激活状态;其它表头,边框调整等。

10.4 dsoframer的打开文件事件。即axFramerControl1.Open("文件名");

10.5由于dsoframer打开文件的方式是独占方式,故为了控制文件及时关闭,需要在关闭控件所在窗体时显式关闭该控件。即this.axFramerControl1.Close();


另外,由于该控件是直接打开的EXCEL文件,同时无法设置EXCEL的单元格背景色,故与效果图有些许差别。但相对于单元格错位来说,也基本达标吧。



11.有些项目在屏幕尺寸上有特殊要求,因此,如果开发平台的屏幕尺寸太小,可以在程序中定义一个尺寸基准变量,后续所有的尺寸以此为基础,而在正式提交前,可以更改此变量,从而更改所有的尺寸信息,当然,这也会增加程序启动的负担。



12. C#中是没有所谓的全局变量的.要构造出类的属性作为全局变量。

即可以在Globale类中写:public static string user = ""; //定义变量 user ="abcd";//赋值构造public string User { get { return user ; } set { user =value; }//定义属性。

这样就可以其他窗体访问此User MessageBox.Show (GlobalParams .User); //直接访问. 显示.结果:"abcd"

GlobalParams frm=new GlobalParams (); frm.User="efg"; //修改该静态变量的值 MessageBox.Show (GlobalParams .User); //直接访问. 显示.结果:"efg"


13.另外,在进行操作EXCEL时,注意相关的变量名不要与系统关键字等重复。如某一列的列名不能是password.


14.获得指定文件的目录:Path.GetDirectoryName(str);

15.创建文件。File.Create (String, Int32, FileOptions) 。

如果是创建EXCEL文件。则操作如下。

            Microsoft.Office.Interop.Excel.Application excelApp;//声明创建EXCEL的操作程序的变量,即API。
            excelApp = new Microsoft.Office.Interop.Excel.Application();
            Microsoft.Office.Interop.Excel.Workbook excelBook = excelApp.Workbooks.Add(1);//实际创建EXCEL。
            Microsoft.Office.Interop.Excel.Worksheet excelSheet = (Microsoft.Office.Interop.Excel.Worksheet)excelBook.Worksheets[1];//创建EXCEL中的工作表。
            excelApp.Visible = false;// EXCEL是否可见。
            string path = Path.GetDirectoryName(str);//获取某个文件的目录。
            excelBook.SaveAs(path+@"\"+agr);//将EXCEL保存为。
            excelSheet.Cells[1, 1] = "username";//给EXCEL标题栏赋值。
            excelSheet.Cells[1, 2] = "pwdword";
            excelSheet.Cells[1, 3] = "status";
            excelSheet.Cells[1, 4] = "classno";
            excelSheet.Cells[1, 5] = "cardno";
            excelSheet.Cells[1, 6] = "loginno";
            excelApp.DisplayAlerts = false;// 是否显示保存对话框。
            excelApp.SaveWorkspace();//将EXCEL内容存储。
            excelApp.Quit();//EXCEL程序退出。


16.SQL中的ExecuteNonQuery()只返回受影响的行数,因此,若进行查询操作,则一直返回0;此时需要进行ExecuteScalar()操作,以判断是否获取到值了。同时,由于SQL是一条记录一条记录地进行查询,因此,ExecuteScalar()返回的是所查询记录的第一个字段。 若需要获取其它字段,需要将记录复制到其他。


17. dataset中数据的访问格式:ds.Tables[0].Rows[0]["字段"]。


18.遍历某个文件夹下的文件方法。

string str=Path.GetDirectoryName(oper.Str);
            DirectoryInfo TheFolder = new DirectoryInfo(str);
           

            foreach (FileInfo NextFile in TheFolder.GetFiles())


19.在以EXCEL存储信息时,需要考虑文件的容量,从而需要考虑在容量不够时新建新的文件,从而引发了数据操作的复杂性。但也无非是多个条件判断而已。


20.C#中怎样取路径中的文件名?

FileInfo fi = new FileInfo("C:\\text.txt");
string strName = fi.Name; // text.txt


21.于C#中操作EXCEL后,关闭对象但EXCEL进程仍然存在的解决办法。设置excelApp.DisplayAlerts = false,就是不提问任何提示,这样再关闭EXCEL对象时,就不会有保存提示,也就不会卡住了.

经试验,此方法不完全正确,需要改进。

在利用SQL进行EXCEL查询,修改,插入,删除等操作时,无需建立EXCEL操作对象,只需建立连接,并及时关闭即可,因此,在此过程中没有多余的EXCEL进程存在的。

但若涉及到EXCEL对象,则需要及时关闭。即(EXCEL对象的名称)appname.quit();即可有效地关闭EXCEL对象及其进程。

若未有效关闭EXCEL进程,则无论系统中含有几个EXCEL进程,则若在程序外打开EXCEL,往往打开两个一样的界面,即原来有一个隐藏的EXCEL显示出来了。


但比较困难的是,若有此操作代码未有效关闭EXCEL进程,刚在后续关闭时较困难。可以手动在系统的进程管理中关闭,也可以在编写更复杂一些的代码进行关闭。
















你可能感兴趣的:(Winform技术类)