代码改变世界

Winform/WPF实例中的相互操作

  观海看云  阅读(794)  评论(0编辑  收藏  举报

介绍一下如何在实例中相互操作,比如在程序A中调用程序B中的方法,看起来不太可能,不过我们可以利用Win32API来实现。我们需要用到2个Win32函数:

uint RegisterWindowsMessage(string lpString);
注册一个新的窗口消息,该消息确保在系统中是唯一的,返回消息值,可在下面SendMessage函数中调用,如果该消息已经存在,则返回它的消息值

IntPtr SendMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);
向指定的窗口发送消息

IntPtr FindWindow(string IpClassName,string IpWindowName);
查找与指定类名或窗口名相同的顶层窗口句柄

IntPtr WindowProc(IntPtr hwnd, uint msg, IntPtr wParam, IntPtr lParam, ref bool handled);
处理发送给窗口的消息

有了这两个程序就可以实现了,比如说我们在程序B中通过RegisterWindowsMessage函数定义一个字符串为“TestInvoke”的系统消息,再在程序A中,通过RegisterWindowMessage函数获得“TestInvoke”的消息值,再通过FindWindow获得程序B的句柄,通过SendMessage发送给窗口B,窗口B再通过WindowProc处理这个消息,调用相关函数。这个过程看似很复杂,只要我们多加思考,还是很容易理解的。

WindowApp1:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
    }
    private void button1_Click(object sender, RoutedEventArgs e)
    {
        int handler = FindWindow(null, "WindowApp2");   //查找窗口句柄
        int testInvokeMsg = RegisterWindowMessage("TestInvoke");    //获得TestInvoke消息值
        SendMessage((IntPtr)handler, (uint)testInvokeMsg, IntPtr.Zero, IntPtr.Zero);    //发送消息
    }
    [DllImport("user32.dll")]
    public static extern int RegisterWindowMessage(string msgName);
    [DllImport("user32.dll", CharSet = CharSet.Auto)]
    public static extern IntPtr SendMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);
    [DllImport("user32.dll", EntryPoint = "FindWindow")]
    public static extern int FindWindow(string lpClassName, string lpWindowName);
}

WindowApp2:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
public partial class MainWindow : Window
{
    int testInvokeMsg = 0;
    public MainWindow()
    {
        InitializeComponent();
        testInvokeMsg = RegisterWindowMessage("TestInvoke");    //获得TestInvoke消息值
        this.SourceInitialized += new EventHandler(MainWindow_SourceInitialized);
    }
    void MainWindow_SourceInitialized(object sender, EventArgs e)
    {
        HwndSource hd = PresentationSource.FromVisual((Visual)sender) as HwndSource;
        hd.AddHook(WindowProc); //添加消息处理函数
    }
    [DllImport("user32.dll")]
    public static extern int RegisterWindowMessage(string msgName);
    public IntPtr WindowProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
    {
        if (msg == testInvokeMsg)
        {
            //TestInvoke处理
            MessageBox.Show("TestInvoke");
        }
        return IntPtr.Zero;
    }
}

代码下载

编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· .NET周刊【3月第1期 2025-03-02】
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
历史上的今天:
2010-03-22 获取数据库中的数据库有多少个
2010-03-22 远程调用存储过程
2010-03-22 动态T-SQL语句常見問題與解決方案
2010-03-22 MSSQL日期推算--本周一的日期和本周结束日期本月开始日期和本月结束日期
2010-03-22 (转)我心中的十大信息化软件开发技术
2010-03-22 C#动态调用C++编写的DLL函数
2010-03-22 判断是否有全屏程序正在运行(C#)
点击右上角即可分享
微信分享提示