C#获取进程的主窗口句柄

通过调用Win32 API实现。

public   class  User32API
ExpandedBlockStart.gifContractedBlock.gif
{
    
private static Hashtable processWnd = null;

    
public delegate bool WNDENUMPROC(IntPtr hwnd, uint lParam);

    
static User32API()
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        
if (processWnd == null)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            processWnd 
= new Hashtable();
        }

    }


    [DllImport(
"user32.dll", EntryPoint = "EnumWindows", SetLastError = true)]
    
public static extern bool EnumWindows(WNDENUMPROC lpEnumFunc, uint lParam);

    [DllImport(
"user32.dll", EntryPoint = "GetParent", SetLastError = true)]
    
public static extern IntPtr GetParent(IntPtr hWnd);

    [DllImport(
"user32.dll", EntryPoint = "GetWindowThreadProcessId")]
    
public static extern uint GetWindowThreadProcessId(IntPtr hWnd, ref uint lpdwProcessId);

    [DllImport(
"user32.dll", EntryPoint = "IsWindow")]
    
public static extern bool IsWindow(IntPtr hWnd);

    [DllImport(
"kernel32.dll", EntryPoint = "SetLastError")]
    
public static extern void SetLastError(uint dwErrCode);

    
public static IntPtr GetCurrentWindowHandle()
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        IntPtr ptrWnd 
= IntPtr.Zero;
        
uint uiPid = (uint)Process.GetCurrentProcess().Id;  // 当前进程 ID
        object objWnd = processWnd[uiPid];

        
if (objWnd != null)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            ptrWnd 
= (IntPtr)objWnd;
            
if (ptrWnd != IntPtr.Zero && IsWindow(ptrWnd))  // 从缓存中获取句柄
ExpandedSubBlockStart.gifContractedSubBlock.gif
            {
                
return ptrWnd;
            }

            
else
ExpandedSubBlockStart.gifContractedSubBlock.gif            
{
                ptrWnd 
= IntPtr.Zero;
            }

        }


        
bool bResult = EnumWindows(new WNDENUMPROC(EnumWindowsProc), uiPid);
        
// 枚举窗口返回 false 并且没有错误号时表明获取成功
        if (!bResult && Marshal.GetLastWin32Error() == 0)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            objWnd 
= processWnd[uiPid];
            
if (objWnd != null)
ExpandedSubBlockStart.gifContractedSubBlock.gif            
{
                ptrWnd 
= (IntPtr)objWnd;
            }

        }


        
return ptrWnd;
    }


    
private static bool EnumWindowsProc(IntPtr hwnd, uint lParam)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        
uint uiPid = 0;

        
if (GetParent(hwnd) == IntPtr.Zero)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            GetWindowThreadProcessId(hwnd, 
ref uiPid);
            
if (uiPid == lParam)    // 找到进程对应的主窗口句柄
ExpandedSubBlockStart.gifContractedSubBlock.gif
            {
                processWnd[uiPid] 
= hwnd;   // 把句柄缓存起来
                SetLastError(0);    // 设置无错误
                return false;   // 返回 false 以终止枚举窗口
            }

        }


        
return true;
    }

}

 

调用User32API.GetCurrentWindowHandle()即可返回当前进程的主窗口句柄,如果获取失败则返回IntPtr.Zero。

--EOF--

 

2008年10月7日补充微软实现的获取进程主窗口句柄代码

ContractedBlock.gif ExpandedBlockStart.gif Code
public class MyProcess
ExpandedBlockStart.gifContractedBlock.gif
{
    
private bool haveMainWindow = false;
    
private IntPtr mainWindowHandle = IntPtr.Zero;
    
private int processId = 0;

    
private delegate bool EnumThreadWindowsCallback(IntPtr hWnd, IntPtr lParam);

    
public IntPtr GetMainWindowHandle(int processId)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        
if (!this.haveMainWindow)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            
this.mainWindowHandle = IntPtr.Zero;
            
this.processId = processId;
            EnumThreadWindowsCallback callback 
= new EnumThreadWindowsCallback(this.EnumWindowsCallback);
            EnumWindows(callback, IntPtr.Zero);
            GC.KeepAlive(callback);

            
this.haveMainWindow = true;
        }

        
return this.mainWindowHandle;
    }


    
private bool EnumWindowsCallback(IntPtr handle, IntPtr extraParameter)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        
int num;
        GetWindowThreadProcessId(
new HandleRef(this, handle), out num);
        
if ((num == this.processId) && this.IsMainWindow(handle))
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            
this.mainWindowHandle = handle;
            
return false;
        }

        
return true;
    }


    
private bool IsMainWindow(IntPtr handle)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        
return (!(GetWindow(new HandleRef(this, handle), 4!= IntPtr.Zero) && IsWindowVisible(new HandleRef(this, handle)));
    }


    [DllImport(
"user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    
public static extern bool EnumWindows(EnumThreadWindowsCallback callback, IntPtr extraData);

    [DllImport(
"user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    
public static extern int GetWindowThreadProcessId(HandleRef handle, out int processId);

    [DllImport(
"user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
    
public static extern IntPtr GetWindow(HandleRef hWnd, int uCmd);

    [DllImport(
"user32.dll", CharSet = CharSet.Auto)]
    
public static extern bool IsWindowVisible(HandleRef hWnd);
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值