visual-studio-2008 - 如何让 F12 进入带有 VS 2008 的 Vista x64 上的调试器?

标签 visual-studio-2008 64-bit windows-vista

F12 是跟踪 UI 阻塞操作的奇迹,但我不知道如何使它与 VS 2008 和托管代码一起工作。

帮助!或不...

编辑:事实证明它在Vista x64上的VS 2005中都不起作用,所以我想根据你的观点,要么扩大要么缩小范围:(

微信

最佳答案

这是一种替代解决方案,即使应用程序“咬”,也可以随时保持按 F12 的便利性,而无需根据我的第一个答案切换到 Visual Studio 并调用“全部中断”命令。

不幸的是,这个解决方案需要向我们想要使用 F12 闯入的应用程序添加一些额外的代码。这个特殊的调试代码可以使用例如条件编译。调试符号,以便 F12 功能在发布版本中不可用。

该解决方案通过创建后台线程来工作;然后线程注册一个全局键盘钩子(Hook)并启动一个无格式消息循环。下面的示例应用程序在主 UI 线程上创建了一个带有单个按钮“Sleep”的表单。单击按钮时,主 UI 线程将通过休眠 10 秒(Thread.Sleep)来锁定,以模拟应用程序“咬”。

要测试 F12 功能,首先单击“ sleep ”按钮,然后按 F12 - 即使主 UI 线程被阻塞,程序也会立即进入调试器。以下屏幕截图显示了 F12 之后我的机器 (Vista x64) 上当前正在运行的线程:

Active threads on F12 http://img242.imageshack.us/img242/5189/f12threadsre9.png

如您所见,主程序仍然在“GoToSleep”方法中,而在我们创建的后台线程中调用了全局钩子(Hook)。

全局 Hook 代码基于 Stephen Toub's article .

现在实现:

using System;
using System.Diagnostics;
using System.Drawing;
using System.Runtime.InteropServices;
using System.Threading;
using System.Windows.Forms;

namespace KeyboardHook
{

    public sealed class SimpleKeyboardHook : IDisposable
    {
        public SimpleKeyboardHook(Action<Keys> handler)
        {
            if (null == handler) { throw new ArgumentNullException("handler"); }
            this._handler = handler;
            var t = new Thread(this.ListenerThread) { IsBackground = true, Name = "KeyboardListener" };
            t.Start();
        }

        public void Dispose()
        {
            if (!this._disposed)
            {
                UnhookWindowsHookEx(this._id);
                this._disposed = true;
                GC.SuppressFinalize(this);
            }
        }

        public static void BreakOnF12(Keys keys)
        {
            if (keys == Keys.F12)
            {
                Debugger.Break();
            }
        }

        private void ListenerThread()
        {
            using (var currentProcess = Process.GetCurrentProcess())
            using (var mainModule = currentProcess.MainModule)
            {
                if (null == mainModule) { throw new InvalidOperationException("Unable to determine main module for the current process"); }

                this._id = SetWindowsHookEx(
                    WH_KEYBOARD_LL,
                    this.HookCallback,
                    GetModuleHandle(mainModule.ModuleName), 0);
            }

            Application.Run();
        }

        private IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
        {
            if (nCode >= 0 && wParam == (IntPtr)WM_KEYDOWN)
            {
                var vkCode = Marshal.ReadInt32(lParam);
                this._handler((Keys)vkCode);
            }
            return CallNextHookEx(this._id, nCode, wParam, lParam);
        }

        private delegate IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, IntPtr lParam);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr SetWindowsHookEx(int idHook,
            LowLevelKeyboardProc lpfn, IntPtr hMod, uint dwThreadId);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        [return: MarshalAs(UnmanagedType.Bool)]
        private static extern bool UnhookWindowsHookEx(IntPtr hhk);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode,
            IntPtr wParam, IntPtr lParam);

        [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr GetModuleHandle(string lpModuleName);

        private const int WH_KEYBOARD_LL = 13;
        private const int WM_KEYDOWN = 0x0100;
        private IntPtr _id;
        private readonly Action<Keys> _handler;
        private volatile bool _disposed;
    }


    static class Program
    {
        private static void GoToSleep(object sender, EventArgs args)
        {
            Thread.Sleep(10000);
        }

        [STAThread]
        static void Main()
        {
            using (new SimpleKeyboardHook(SimpleKeyboardHook.BreakOnF12))
            {
                Application.EnableVisualStyles();
                Application.SetCompatibleTextRenderingDefault(false);

                var form = new Form { Text = "Sleepy form", Size = new Size(160,80), Padding = new Padding(6) };
                var btn = new Button { Dock = DockStyle.Fill, Text = "Sleep", Location = new Point(10, 10) };
                btn.Click += GoToSleep;
                form.Controls.Add(btn);
                Application.Run(form);
            }
        }
    }
}

关于visual-studio-2008 - 如何让 F12 进入带有 VS 2008 的 Vista x64 上的调试器?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/408744/

相关文章:

visual-studio - Visual Studio 2008 数据库版 - 它是什么?

.net - x64 应用程序可以使用 x86 程序集吗?反之亦然?

visual-studio - Visual Studio 引用 x64 GAC

带有 RS-232 串行端口的 Javascript 接口(interface)

visual-studio - 将库添加到 Visual Studio 问题

visual-studio-2008 - 在 Visual Studio 2008 中使用自定义安装程序时出现错误 1001

visual-studio-2010 - Visual Studio 扩展 : Wait for all projects to complete loading with IVsSolutionEvents OnAfterOpenSolution

.net - 在同一解决方案/项目中使用 Visual Studio 针对 32 位和 64 位

c++ - 如何为我的应用程序禁用 UAC

c++ - 我可以使用 ChangeWindowMessageFilter 从保护模式 IE 接收 DocumentComplete 事件吗?