WinForm学习笔记(2)

 


1,MDI窗体
设有两个窗体frmMain,frmChild,则:
frmMain:  设IsMdiContainer属性为true
 打开子窗口:
  在相关事件中写如下代码:
  frmChild child=new frmChild();
  child.MdiParent=this;//this表示本窗体为其父窗体
  child.Show();
  在打开子窗体时,如果只允许有一个子窗体,可以加入如下判断:
  if (this.ActiveMdiChild!=null)
  {
     this.ActiveMdiChild.Close(); //关闭已经打开的子窗体
     //....
  }
更改MDI主窗体背景
 先声明一个窗体对象
 private System.Windows.Forms.MdiClient m_MdiClient;
 在Form_Load等事件中,添加如下代码:
 int iCnt=this.Controls.Count;
 for(int i=0;i<iCnt;i++)
 {
 if(this.Controls[i].GetType().ToString()=="System.Windows.Forms.MdiClient")
 {
  this.m_MdiClient=(System.Windows.Forms.MdiClient)this.Controls[i];
  break;
 }
  }
 this.m_MdiClient.BackColor=System.Drawing.Color.Silver;
 具体可参见:http://cnblogs.com/Daview/archive/2004/05/06/8381.aspx

2,创建系统托盘菜单
  2.1,创建一个contextMenu(cmnMain)菜单
  2.2,添加一个NotifyIcon组件,设置ContextMenu属性为cmnMain
  2.3,相应窗体改变事件(最小化等)
   private void frmMain_SizeChanged(object sender,EventArgs e)
   {
      if (this.WindowState==FormWindowState.Minimized)
      {
         this.Hide();
         noiMain.Visible=true;
      }
   }

  2.4,相应用户单击系统托盘上contextmenu菜单事件
  private void mniOpen(object sender,EventArgs e)
  {
      noiMain.Visible=false;
      this.Show();
      this.Focus();
  }

  2.5,响应用户双击系统托盘图标事件
  private void noiMain_DoubleClick(object s,EventArgs e)
  {
      minOpen.PerformClick(); //相当与mniOpen按钮的单击事件
  }
  **注意添加相应的事件句柄**

3,创建不规则窗体
 3.1,在窗体上创建不规则图象,可以用gdi+绘制,或在图象控件上使用图象填充
 3.2,设置窗体的backcolor为colorA,然后设置TransparencyKey为colorA
 3.3,设置FormBorderStyle为none;

4,创建顶部窗体
  this.TopMost=true;//把窗体的TopMost设置为true


5,调用外部程序

  using System.Diagnostics

  Process proc=new Process();
  proc.StartInfo.FileName=@"notepad.exe";  //注意路径
  proc.StartInfo.Arguments="";
  proc.Start();

//获得当前目录Directory.GetCurrentDirectory() (using System.IO)

6,Toolbar的使用
  Toolbar控件通常需要imagelist控件结合使用(需要用到其中图标)
  响应Toolbar单击事件处理程序代码:
  switch(ToolbarName.Buttons.IndexOf(e.Button))
  {
      case 0:   //第一个按钮
          //code ...
          break;
      case 1:   //第二个按钮
          //code ...
          break;
      //other case code
      default:  //默认处理,但以上所有项都不符合时
          //code ...
          break;
  }

7,弹出对话框获得相关返回值
  在窗体的closing事件中运行如下代码,可以在用户关闭窗体时询问
  DialogResult result=MessageBox.Show(this,"真的要关闭该窗口吗?","关闭提示",MessageBoxButtons.OKCancel,MessageBoxIcon.Question);
  if (result==DialogResult.OK)
  {
  //关闭窗口
  e.Cancel=false;
  }
  else
  {
  //取消关闭
  e.Cancel=true;
  }

8,打印控件
  最少需要两个控件
  PrintDocument
  PrintPreviewDialog:预览对话框,需要printdocument配合使用,即设置document属性为
                     对应的printDocument
  printdocument的printpage事件(打印或预览事件处理程序)代码,必须.

 float fltHeight=0;
 float fltLinePerPage=0;
 long lngTopMargin=e.MarginBounds.Top;
 int intCount=0;
 string strLine;

 //计算每页可容纳的行数,以决定何时换页
 fltLinePerPage=e.MarginBounds.Height/txtPrintText.Font.GetHeight(e.Graphics);

  
 while(((strLine=StreamToPrint.ReadLine()) != null) && (intCount<fltLinePerPage))
 {
    intCount+=1;
    fltHeight=lngTopMargin+(intCount*txtPrintText.Font.GetHeight(e.Graphics));
    e.Graphics.DrawString(strLine,txtPrintText.Font,Brushes.Green,e.MarginBounds.Left,fltHeight,new StringFormat());
 }

 //决定是否要换页
 if (strLine!=null)
 {
    e.HasMorePages=true;
 }
 else
 {
    e.HasMorePages=false;
 }
  以上代码的StreamToPrint需要声明为窗体级变量:
  private System.IO.StringReader StreamToPrint;

  打开预览对话框代码(不要写在printpage事件中)
  StreamToPrint=new System.IO.StringReader(txtPrintText.Text);
  PrintPreviewDialogName.ShowDialog();

9,string对象本质与StringBuilder类,字符串使用
string对象是不可改变的类型,当我们对一个string对象修改后将会产生一个新的string对
象,因此在需要经常更改的字符对象时,建议使用StringBuilder类:
[范例代码]构造一个查询字符串
StringBuilder sb=new StringBuilder("");
sb.Append("Select * from Employees where ");
sb.Append("id={0} and ");
sb.Append("title='{1}'");
String cmd=sb.ToString();

sb=null; //在不再需要时清空它

cmd=String.Format(cmd,txtId.Text,txtTile.Text); //用实际的值填充格式项

判断字符串是否为空:
检查一个字符串是否为空或不是一个基本的编程需要,一个有效的方法是使用string类的Length属性来取代使用null或与""比较。

比较字符串:使用String.Equals方法来比较两个字符串
string str1="yourtext";
if (str1.Equals("TestSting") )
{
  // do something
}

10,判断某个字符串是否在另一个字符串(数组)中
 需要用到的几个方法
 string.Split(char);//按照char进行拆分,返回字符串数组
 Array.IndexOf(Array,string):返回指定string在array中的第一个匹配项的下标
 Array.LastIndexOf(Array,string):返回指定string在array中的最后一个匹配项的下标
 如果没有匹配项,则返回-1
 [示例代码]:
 string strNum="001,003,005,008";
 string[] strArray=strNum.Split(',');//按逗号拆分,拆分字符为char或char数组
 
 Console.WriteLine(Array.IndexOf(strArray,"004").ToString());

11,DataGrid与表和列的映射
从数据库读取数据绑定到DataGrid后,DataGrid的列标头通常跟数据库的字段名相同,如果
不希望这样,那么可以使用表和列的映射技术:
using System.Data.Common;

string strSql="select * from Department";
OleDbDataAdapter adapter=new OleDbDataAdapter(strSql,conn);

DataTableMapping dtmDep=adapter.TableMappings.Add("Department","部门表");

dtmDep.ColumnMappings.Add("Dep_Id","部门编号");
dtmDep.ColumnMappings.Add("Dep_Name","部门名称");

DataSet ds=new DataSet();

adapter.Fill(ds,"Department"); //此处不能用"部门表"

响应单击事件(datagrid的CurrentCellChanged事件)
DataGridName.CurrentCell.ColumnNumber;//所单击列的下标,从0开始,下同
DataGridName.CurrentCell.RowNumber;//所单击行的下标
DataGridName[DataGridName.CurrentCell];//所单击行和列的值

DataGridName[DataGridName.CurrentRowIndex,n].ToString();//获得单击行第n+1列的值

12,动态添加菜单并为其添加响应事件
 添加顶级菜单:
 MainMenuName.MenuItems.Add("顶级菜单一");//每添加一个将自动排在后面

 添加次级菜单:
 MenuItem mniItemN=new MenuItem("MenuItemText")
 MenuItem mniItemN=new MenuItem("MenuItemText",new EventHandler(EventDealName))
 MainMenuName.MenuItems[n].MenuItems.Add(mniItemN);//n为要添加到的顶级菜单下标,从0开始

 创建好菜单后添加事件:
 mniItemN.Click+=new EventHandler(EventDealName);

 也可以在添加菜单的同时添加事件:
 MenuItem mniItemN=new MenuItem("MenuItemText",new EventHandler(EventDealName));
 MainMenuName.MenuItems[n].MenuItems.Add(mniItemN);

13,正则表达式简单应用(匹配,替换,拆分)
 using System.Text.RegularExpressions;

 //匹配的例子
 string strRegexText="你的号码是:020-32234102";
 string filter=@"\d{3}-\d*";

 Regex regex=new Regex(filter);
 Match match=regex.Match(strRegexText);

 if (match.Success) //判断是否有匹配项
 {
     Console.WriteLine("匹配项的长度:"+match.Length.ToString());
     Console.WriteLine("匹配项的字符串:"+match.ToString());
     Console.WriteLine("匹配项在原字符串中的第一个字符下标:"+match.Index.ToString());
 }
 
 //替换的例子
 string replacedText=regex.Replace(strRegexText,"020-88888888");
 Console.WriteLine(replacedText);//输出"你的号码是:020-88888888"

 //拆分的例子
 string strSplitText="甲020-32654已020-35648丙020-365984";
 foreach(string s in regex.Split(strSplitText))
 {
    Console.WriteLine(s); //依次输出"甲乙丙"
 }

13,多线程简单编程
 using System.Threading;

 Thread ThreadTest=new Thread(new ThreadStart(ThreadCompute));
 ThreadTest.Start();//使用另一个线程运行方法ThreadCompute
 
 ThreadCompute方法原型:
 private void ThreadCompute()
 {}

14,操作注册表
using System.Diagnostics;
using Microsoft.Win32;
  //操作注册表
  RegistryKey RegKey=Registry.LocalMachine.OpenSubKey("Software",true);

  //添加一个子键并给他添加键值对
  RegistryKey NewKey=RegKey.CreateSubKey("regNewKey");
  NewKey.SetValue("KeyName1","KeyValue1");
  NewKey.SetValue("KeyName2","KeyValue2");

  //获取新添加的值
  MessageBox.Show(NewKey.GetValue("KeyName1").ToString());

  //删除一个键值(对)
  NewKey.DeleteValue("KeyName1");

  //删除整个子键
  RegKey.DeleteSubKey("regNewKey");
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: .NET WinForm是一种用于开发Windows桌面应用程序的技术框架。我在使用.NET WinForm开发过程中积累了一些经验和心得,现将其总结如下。 首先,.NET WinForm具有丰富的控件库,并且提供了灵活的布局方式。我们可以根据需要选择合适的控件来设计用户界面,并通过属性和事件的设置来实现交互功能。此外,WinForm还支持多种布局方式,例如流式布局、表格布局等,能够满足不同界面设计的需求。 其次,开发过程中要注意控件的命名规范和代码的结构化。一个良好的命名规范可以使代码更易读、易维护,并且方便与团队合作。另外,将代码按功能模块进行组织,可以提高代码的可读性和可维护性。 另外,对于数据处理方面,我们可以使用.NET提供的ADO.NET来进行数据库的操作。ADO.NET提供了一套完整的对象模型,可以方便地连接、查询和更新数据库。此外,为了提高性能,我们可以使用数据绑定技术将数据与控件进行关联,从而实现数据的显示和编辑。 此外,为了提高用户体验,我们可以使用多线程来处理耗时的操作。在WinForm中,主线程负责界面的渲染和响应用户操作,而通过新建一个后台线程来处理耗时任务,可以避免界面的阻塞,提高用户体验。 最后,为了保证应用程序的稳定性和可靠性,我们需要进行适当的异常处理。在编写代码的过程中,我们应该考虑到各种可能出现的异常情况,并编写相应的异常处理代码,以防止程序崩溃或出现不可预料的错误。 综上所述,使用.NET WinForm进行开发可以帮助我们快速构建功能丰富的Windows桌面应用程序。在开发过程中,我们应该注意控件的选择和布局的合理安排,遵循良好的命名规范和代码结构化,合理处理数据和耗时操作,添加适当的异常处理,以保证应用程序的稳定性和可靠性。 ### 回答2: .NET WinForm是一种使用C#或其他.NET支持的语言开发的桌面应用程序的框架。在.NET WinForm开发过程中,我积累了一些经验和技巧,我将在下面的笔记中进行总结。 首先,我要着重强调良好的用户界面设计。一个简洁、直观和易用的界面对于用户来说非常重要,因此在设计界面时,我会参考标准的用户体验原则和设计规范,确保界面元素的布局合理,按钮和菜单项的命名清晰明了。 其次,我经常使用控件的继承和自定义控件来提高开发效率和代码复用性。通过继承现有控件,并添加自定义功能或修改样式,可以快速创建符合特定需求的控件。这样可以减少代码的重复编写,并且便于后续的维护和扩展。 此外,我会尽量避免在界面上编写过多的业务逻辑。为了保持界面的简洁和响应速度,我将业务逻辑封装在独立的类中,并使用事件机制进行交互。这样可以实现界面和数据的解耦,提高代码的可维护性和可测试性。 对于数据的处理,我会使用.NET提供的数据绑定机制。通过将数据源与控件进行绑定,可以简化数据的展示和操作,减少手动编写代码的工作量。同时,数据绑定还可以实现数据的实时更新和显示,提升用户体验。 最后,我会使用调试工具和日志记录来帮助定位和解决问题。.NET WinForm提供了丰富的调试功能,如断点调试、条件断点和调试输出等。通过适当的调试设置和日志记录,可以及时发现问题,并进行相关的修复和优化。 总之,这些是我在.NET WinForm开发过程中的一些心得体会。良好的界面设计、控件的继承和自定义、业务逻辑和界面的解耦、数据绑定的使用以及调试工具和日志记录的运用,都是我在开发过程中重要的思考和实践。我相信通过不断学习和尝试,我将能够提高开发效率和代码质量,为用户创造出更好的应用程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值