几天遇见一个问题需要检查某个wpf程序是否已经运行,如果没有运行则启动传递参数,如果已运行则需要直接传递消息。在没有运行 情况下传递参数很简单,我们只需要Process cmd窗口启动并传递参数,在程序中处理。但是如果程序已经启动有点麻烦,凭着我曾winform的经验第一时间想到的是win32 api  SendMessage,我们的C#程序只需要DllImport就可以调用了。经过一番查找和对wpf window和DispatcherObject的Reflector,花了我大半天终于找到了System.Windows.Interop.HwndSource中有AddHock方法可以添加对win32消息机制的监听。这下就很好办了我们只需要注册MainWindow的这个时间,来监听win32消息处理我们的0x004A消息。

    控制台代码,主要应用的FindWindow 这个win32方法查找我们的窗体,SendMessage发送我们的消息,和winform没有什么差别,对于win32的使用你可以参考毒龙的程序人生关于C#中实现两个应用程序消息通讯的问题。难得查win32 Api直接copy,借来用用。

程序:

 

  1. 查看代码   
  2. using System;   
  3. using System.Collections.Generic;   
  4. using System.Linq;   
  5. using System.Text;   
  6. using System.Runtime.InteropServices;   
  7.  
  8. namespace ConsoleApplication1   
  9. {   
  10.     class Program   
  11.     {   
  12.         static void Main(string[] args)   
  13.         {   
  14.             string ch = "";   
  15.             while (ch.ToLower() != "q")   
  16.             {   
  17.  
  18.                 if (!SendMessage("Window1", @"Hello,I am from Console Program:" + ch))   
  19.                 {   
  20.                    Console.WriteLine("no window");  
  21.  
  22.                 };   
  23.                 ch = Console.ReadLine();   
  24.             }   
  25.         }   
  26.  
  27.         public static bool SendMessage(string windowName, string strMsg)   
  28.         {   
  29.             if (strMsg == nullreturn true;   
  30.             IntPtr hwnd = (IntPtr)FindWindow(null, windowName   );   
  31.             if (hwnd != IntPtr.Zero)   
  32.             {   
  33.                 CopyDataStruct cds;   
  34.                 cds.dwData = IntPtr.Zero;   
  35.                 cds.lpData = strMsg;   
  36.                 cds.cbData = System.Text.Encoding.Default.GetBytes(strMsg).Length + 1;   
  37.                 int fromWindowHandler = 0;   
  38.                 SendMessage(hwnd, 0x004A, fromWindowHandler, ref  cds);   
  39.                 return true;   
  40.             }   
  41.             return false;   
  42.         }   
  43.  
  44.         [DllImport("User32.dll", EntryPoint = "FindWindow")]   
  45.         private static extern int FindWindow(string lpClassName, string lpWindowName);   
  46.  
  47.         [DllImport("User32.dll", EntryPoint = "SendMessage")]   
  48.  
  49.         private static extern int SendMessage   
  50.  
  51.         (   
  52.  
  53.         IntPtr hWnd,   
  54.  
  55.         int Msg,   
  56.  
  57.         int wParam,   
  58.  
  59.         ref  CopyDataStruct lParam   
  60.  
  61.         );   
  62.  
  63.     }   
  64.  
  65.     [StructLayout(LayoutKind.Sequential)]   
  66.  
  67.     public struct CopyDataStruct   
  68.     {   
  69.  
  70.         public IntPtr dwData;   
  71.  
  72.         public int cbData;   
  73.  
  74.         [MarshalAs(UnmanagedType.LPStr)]   
  75.  
  76.         public string lpData;   
  77.  
  78.     }   
  79.  
  80. }  
  81. 复制代码 

wpf端程序:主要需要在MainWindow中loaded事件订阅消息监听:这里需要System.Windows.Interop.HwndSource的AddHock方法注册
程序:

  1. 查看代码   
  2.  
  3. IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)   
  4.        {   
  5.  
  6.            if (msg == 0x004A)   
  7.            {   
  8.  
  9.                CopyDataStruct cds = (CopyDataStruct)System.Runtime.InteropServices.Marshal.PtrToStructure(lParam, typeof(CopyDataStruct));   
  10.  
  11.                MessageBox.Show(cds.lpData);   
  12.                this.Visibility = Visibility.Visible;   
  13.            }   
  14.  
  15.            return hwnd;   
  16.  
  17.        }   
  18.  
  19.        private void Window_Loaded(object sender, RoutedEventArgs e)   
  20.        {             
  21.  
  22.            (PresentationSource.FromVisual(this) as System.Windows.Interop.HwndSource).AddHook(new System.Windows.Interop.HwndSourceHook(WndProc));   
  23.        }  
  24. 复制代码 

截个图:

很简单的东西结果被MS封装的不知哪里去,让我查了半天(其实应该是我的无知吧,不管怎么解决了就是心情舒畅了);