winfrom开发框架源码

来看消息循环这个心脏部件。Application.Run方法里的那坨代码其实是个永动机:

while (GetMessage(ref msg, IntPtr.Zero, 0, 0))
{
    TranslateMessage(ref msg);
    DispatchMessage(ref msg);
}

这死循环看着吓人,但正是它让窗口能持续响应事件。有意思的是微软在这里埋了个彩蛋——Application.DoEvents()其实就是手动触发消息处理,用不好容易让界面抽风,新手慎碰。

控件绘制这块,Control类的CreateGraphics方法暗藏玄机:

public Graphics CreateGraphics()
{
    IntPtr hdc = UnsafeNativeMethods.GetDC(new HandleRef(this, Handle));
    return Graphics.FromHdcInternal(hdc);
}

这暴露了GDI+的老底,每次调用都会创建新Graphics对象。所以千万别在OnPaint外面乱用这玩意,内存泄漏分分钟教你做人。记得用using包裹或者重写Paint事件才是正解。

数据绑定方面,BindingSource的机制有点意思。看这段同步代码:

private void CurrencyManager_CurrentChanged(object sender, EventArgs e)
{
    if (_inCurrentChanged) return;
    _inCurrentChanged = true;
    // 同步UI和数据的黑魔法
    UpdateControls();
    _inCurrentChanged = false;
}

这个_inCurrentChanged标志位玩得溜,防住了无限递归。实战中要是自己写双向绑定,记得抄这个防呆设计,不然改个数值能让界面和后台数据打起来。

再看控件树的处理,Control.ControlCollection的Add实现里藏着彩蛋:

public virtual void Add(Control value)
{
    if (value == null) return;
    if (value.Parent != null) value.Parent.Controls.Remove(value);
    // 这里开始排列组合Z序
    InnerList.Add(value);
    value.AssignParent(this._owner);
}

这解释了为什么同一个控件不能有多个爹。有意思的是InnerList用ArrayList而非泛型集合,估计是.NET 1.0时代的老代码没改,现在看着确实有点考古的味道。

最后给个实战建议:处理复杂界面时,别被自动生成的Designer.cs文件带沟里。试着重写控件的布局逻辑:

protected override void OnLayout(LayoutEventArgs levent)
{
    base.OnLayout(levent);
    // 在这里玩自定义布局
    textBox1.Location = new Point((this.Width - textBox1.Width) / 2, 10);
}

比在属性面板调坐标灵活多了,特别是需要动态适配大小时,这招比锚定布局更暴力直接。

Logo

开源鸿蒙跨平台开发社区汇聚开发者与厂商,共建“一次开发,多端部署”的开源生态,致力于降低跨端开发门槛,推动万物智联创新。

更多推荐