visual studio(2015/2017)调试多线程程序

教程来自巨头微软官方机翻

动手完成全部内容大约需要半小时

完成整个教程的效果图:(本人使用vs2015 & C++)

正文:

开始调试多线程应用程序 (C#,Visual Basic、 c + +)

Visual Studio 提供多种工具和用户界面元素,用于调试多线程应用程序。 本教程演示如何使用线程标记、“并行堆栈”窗口、“并行监视”窗口、条件断点、筛选器断点。 完成本教程只需数分钟,然后你就会熟悉用于调试多线程应用程序的功能。

下面两个主题额外介绍了如何使用其他多线程调试工具:

首先需要一个多线程应用程序项目。 示例如下。

创建一个多线程应用项目

  1. 在“文件”菜单上,选择“新建” > “项目”。

    此时将出现“新建项目”对话框。

  2. 选择语言:Visual C#Visual C++Visual Basic

  3. 在“Windows 桌面”下,选择“控制台应用”。

  4. 在“名称”字段中,输入 MyThreadWalkthroughApp。

  5. 选择“确定”。

    新的控制台项目随即显示。 创建该项目后,将显示源文件。 根据所选语言,源文件名称可能是 Program.cs、MyThreadWalkthroughApp.cpp 或 Module1.vb。

  6. 删除出现在源文件中的代码,将其替换为下面列出的相应示例代码。

    C#

using System;
using System.Threading;

public class ServerClass
{

    static int count = 0;
    // The method that will be called when the thread is started.
    public void InstanceMethod()
    {
        Console.WriteLine(
            "ServerClass.InstanceMethod is running on another thread.");

        int data = count++;
        // Pause for a moment to provide a delay to make
        // threads more apparent.
        Thread.Sleep(3000);
        Console.WriteLine(
            "The instance method called by the worker thread has ended.");
    }
}

public class Simple
{
    public static void Main()
    {
        for (int i = 0; i < 10; i++)
        {
            CreateThreads();
        }
    }
    public static void CreateThreads()
    {
        ServerClass serverObject = new ServerClass();

        Thread InstanceCaller = new Thread(new ThreadStart(serverObject.InstanceMethod));
        // Start the thread.
        InstanceCaller.Start();

        Console.WriteLine("The Main() thread calls this after "
            + "starting the new InstanceCaller thread.");

    }
}

C++

#include "pch.h"
#include <thread>
#include <iostream>
#include <vector>

int count = 0;

void doSomeWork() {

    std::cout << "The doSomeWork function is running on another thread." << std::endl;
    int data = count++;
    // Pause for a moment to provide a delay to make
    // threads more apparent.
    std::this_thread::sleep_for(std::chrono::seconds(3));
    std::cout << "The function called by the worker thread has ended." << std::endl;
}

int main() {
    std::vector<std::thread> threads;

    for (int i = 0; i < 10; ++i) {

        threads.push_back(std::thread(doSomeWork));
        std::cout << "The Main() thread calls this after starting the new thread" << std::endl;
}

for (auto& thread : threads) {
    thread.join();
}

return 0;
}

VB

  1. Imports System.Threading
    
    Public Class ServerClass
        ' The method that will be called when the thread is started.
        Public count = 0
        Public Sub InstanceMethod()
            Console.WriteLine(
                    "ServerClass.InstanceMethod is running on another thread.")
    
            Dim data = count + 1
            ' Pause for a moment to provide a delay to make
            ' threads more apparent.
            Thread.Sleep(3000)
            Console.WriteLine(
                    "The instance method called by the worker thread has ended.")
        End Sub
    
    End Class
    
    Public Class Simple
    
        Public Shared Sub Main()
    
            Dim ts As New ThreadStarter
            For index = 1 To 10
                ts.CreateThreads()
            Next
    
        End Sub
    
    End Class
    Public Class ThreadStarter
        Public Sub CreateThreads()
            Dim serverObject As New ServerClass()
    
            ' Create the thread object, passing in the
            ' serverObject.InstanceMethod method using a
            ' ThreadStart delegate.
            Dim InstanceCaller As New Thread(AddressOf serverObject.InstanceMethod)
    
            ' Start the thread.
            InstanceCaller.Start()
    
            Console.WriteLine("The Main() thread calls this after " _
                        + "starting the new InstanceCaller thread.")
    
        End Sub
    End Class
    
  2. 在“文件”菜单上,单击“全部保存”。

  3. (仅限 Visual Basic)在解决方案资源管理器 (右窗格),右键单击项目节点,选择属性。 下应用程序选项卡上,更改启动对象简单

调试多线程应用程序

  1. 在源代码编辑器中,查找以下代码段之一:

    C#

Thread.Sleep(3000);  
Console.WriteLine();  

C++

std::this_thread::sleep_for(std::chrono::seconds(3));
std::cout << "The function called by the worker thread has ended." << std::endl; 

VB

  1. Thread.Sleep(3000)
    Console.WriteLine()
    
  2. 左键单击中的左滚动条槽Thread.Sleepstd::this_thread::sleep_for语句将新断点。

    滚动条槽中的红色圆圈指示在此位置设置断点。

  3. 在“调试”菜单上,单击“开始调试(F5)”。

    Visual Studio 将生成该解决方案,应用在附加了调试器的情况下开始运行,然后在断点处停止。

  4. 在源代码编辑器中,找到包含该断点的行。

发现线程标记  

  1. 在调试工具栏中,单击“在源中显示线程”按钮在源中显示线程

  2. 按一下 F11 使调试器前进一个代码行。

  3. 查看窗口左侧的滚动条槽。 在此行中,会看到线程标记 图标线程标记 ,类似于一条双绞线。 线程标记指示线程在此位置停止。

    线程标记可以被断点部分隐藏。

  4. 将指针悬停在线程标记上。 此时会出现一个数据提示,告知你每个已停止线程的名称和线程 ID 号。 在这种情况下,名称可能是 <noname>

  5. 选择线程标记,以查看快捷菜单上的可用选项。

查看线程位置

在“并行堆栈”窗口中,可以在“线程”视图和“任务”视图(适用于基于任务的编程)之间进行切换,并且可以查看每个线程的调用堆栈信息。 在此应用中,我们可以使用“线程”视图。

  1. 通过选择“调试” > “窗口” > “并行堆栈”,打开“并行堆栈”窗口。 此时会看到如下所示的内容。 确切信息取决于每个线程的当前位置、硬件以及编程语言。

    并行堆栈窗口

    在此示例中,从左到右会看到托管代码的以下信息:

    • 主线程(左侧)已停止在 Thread.Start 上,由线程标记图标指示停止点线程标记
    • 两个线程已进入 ServerClass.InstanceMethod,其中一个线程是当前线程(黄色箭头),另一个线程已停止在 Thread.Sleep 中。
    • 新线程(右侧)也已启动,但是停止在 ThreadHelper.ThreadStart 上。
  2. 右键单击“并行堆栈”窗口中的条目,查看快捷菜单上的可用选项。

    可以通过这些右键单击菜单执行各种操作,但在本教程中,我们将在“并行监视”窗口中展示更多这些细节(后续部分)。

    备注

    若要查看包含每个线程的信息的列表视图,请改用“线程”窗口。 请参阅演练:调试多线程应用程序

对变量设置监视

  1. 通过选择“调试” > “窗口” > “并行监视” > “并行监视 1”,打开“并行监视”窗口。

  2. 选择 <Add Watch> 文本所在的单元格(或第 4 列中的空标头单元格),输入 data

    在窗口中显示每个线程的数据变量的值。

  3. 选择 <Add Watch> 文本所在的单元格(或第 5 列中的空标头单元格),输入 count

    窗口中会显示每个线程的 count 变量的值。 如果看不到这么多的信息,请尝试按 F11 几次以继续在调试器中执行线程。

    并行监视窗口

  4. 右键单击其中一个窗口以查看可用选项中的行。

标记线程和取消标记线程

可以通过标记线程来追踪重要的线程,并忽略其它线程。

  1. 在中并行监视窗口中,按住Shift键并选择多个行。

  2. 右键单击并选择标志

    所选的所有线程都将都标记。 现在,您可以筛选为仅显示已标记的线程。

  3. 在中并行监视窗口中,选择仅显示标记的线程按钮显示已标记线程

    标记的线程显示在列表中。

    提示

    在标记一些线程后,可以右键单击代码编辑器中的代码行,然后选择“将标记的线程运行到光标处”。 请确保选择所有已标记的线程将达到的代码。 Visual Studio 将在选择的代码行处暂停线程,这样就可以通过冻结和解冻线程更容易地控制执行顺序。

  4. 选择仅显示标记的线程按钮将再次切换回显示所有线程模式。

  5. 若要取消标记线程,请在“并行监视”窗口右键单击一个或多个已标记线程,然后选择“取消标记”。

冻结和解冻线程执行

提示

可以通过冻结和解冻(暂停和恢复)线程来控制线程执行工作的顺序。 这有助于解决并发问题,例如死锁和争用条件。

  1. 在“并行监视”窗口中,在选中所有行的情况下,右键单击并选择“冻结”。

    在第二个列中,每个行出现一个暂停图标。 暂停图标指示该线程已冻结。

  2. 仅选择一行,取消选中其他行。

  3. 右键单击某一行,然后选择解冻

    暂停图标在此行上消失,表明线程已不再被冻结。

  4. 切换到代码编辑器,按 F11。 仅运行未冻结的线程。

    应用还实例化某些新线程。 任何新线程均处于未标记状态,不会被冻结。

请按照具有条件断点的单一线程

可以在调试器中对单线程的执行情况进行跟踪。 一种方法是冻结不感兴趣的线程。 在某些情况下,可能需要在不冻结其它线程的情况下跟踪单个线程,例如重现特定 Bug。 若要在不冻结其他线程的情况下跟踪某个线程,必须避免在不感兴趣的线程上中断。 您可以执行此操作通过设置条件断点

可以根据不同的条件(例如,线程名称或线程 ID)来设置断点。 如果你知道数据对于每个线程都是唯一的,则可根据该数据设置条件断点。 这是常见的调试场景,即相比于特定的线程,你对某些特定的数据值更感兴趣。

  1. 右键单击以前创建的断点并选择条件

  2. 在“断点设置”窗口中,输入 data == 5 作为条件表达式。

    条件断点

    提示

    如果对特定的线程更感兴趣,则请使用线程名称或线程 ID 作为条件。 若要在“断点设置”窗口中执行此操作,请选择“筛选器”而不是“条件表达式”,并按照筛选器提示操作。 可能需要在应用代码中指定线程名称,因为线程 ID 在重启调试器时会更改。

  3. 关闭断点设置窗口。

  4. 选择重启重启应用按钮以重启调试会话。

    将在数据变量的值为 5 的线程中,中断代码执行。 请在“并行监视”窗口中,寻找表示当前调试器上下文的黄色箭头。

  5. 现在,可以单步执行代码 (F10) 和单步执行代码 (F11) 并按照单个线程执行。

    只要断点条件是唯一的线程,且调试器不会命中 (可能需要禁用它们) 的其他线程上的任何其他断点,可以单步执行代码并单步执行代码而无需切换到其他线程。

    备注

    当您推进调试器进度时,将运行所有线程。 但是,调试器不会中断到其他线程上的代码中,除非其中一个其他线程遇到断点。

请参阅

调试多线程应用
如何:在调试时切换到另一个线程
如何: 使用并行堆栈窗口
如何:使用“并行监视”窗口

 

  • 21
    点赞
  • 103
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Visual Studio中,调试多线程可以使用多种工具和技巧。其中一些常用的工具包括线程窗口、线程标记、并行堆栈窗口和并行监视窗口。通过这些工具,你可以查看并行堆栈信息,进行线程标记、切换和冻结等操作。\[1\] 当你在应用程序调试模式中停在断点时,应用程序会被暂停,但其他线程仍然存在。你可以使用Visual Studio调试菜单中的"Show Threads in Source"选项来查看这些线程。这个选项可以在源代码中显示线程,方便你进行调试。\[2\] 除了线程窗口和线程标记,Visual Studio还提供了其他调试技巧,如多类型断点和数据监视。这些技巧可以帮助你更好地调试多线程应用程序。\[3\] 总之,Visual Studio提供了丰富的工具和技巧来调试多线程应用程序。通过使用线程窗口、线程标记、并行堆栈窗口和并行监视窗口等工具,以及多类型断点和数据监视等技巧,你可以更好地理解和调试多线程应用程序。 #### 引用[.reference_title] - *1* *3* [Visual Studio 高级调试-代码调试](https://blog.csdn.net/qq_40404477/article/details/129051971)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Visual Studio调试多线程应用程序](https://blog.csdn.net/daocaorencrl/article/details/7865384)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值