Как повредить WinDbg в анонимном методе?

В заголовке отчасти говорится все это. Обычная команда SOS! bpmd не делает большое хорошее без имени.

Некоторые идеи я имел:

  • выведите каждый метод, затем используйте! bpmd-md, когда Вы находите соответствующий MethodDesc
    • не практичный в использовании реального мира, от того, что я могу сказать. Даже если я записал макрос для ограничения дампа анонимными типами/методами, нет никакого очевидного способа сказать им независимо.
  • используйте Отражатель для дампа имени MSIL
    • не помогает при контакте с динамическими блоками и/или Отражением. Испустить. Неспособность Visual Studio считать локальный Вар в таких сценариях является целой причиной, которую я повернул к Windbg во-первых...
  • установите точку останова в VS, ожидайте его для удара, затем изменитесь на Windbg с помощью неразрушающего приема
    • попытка отсоединиться из VS заставляет это зависать (наряду с приложением). Я думаю, что это - то, вследствие того, что управляемый отладчик является "мягким" отладчиком через инжекцию потока вместо стандартного "твердого" отладчика. Или возможно это - просто ошибка VS, характерная для Silverlight (едва было бы первым, я встретился).
  • установите точку останова на некотором другом местоположении, которое, как известно, звонило в анонимный метод, затем одноэтапный Ваш путь в
    • мой план резервного копирования, хотя я не обратился бы к нему, если это Вопросы и ответы показывает лучший путь

8
задан Richard Berg 12 March 2010 в 06:33
поделиться

2 ответа

Анонимный метод на самом деле не анонимен. Он просто прячется за именем, сгенерированным компилятором.

Рассмотрим этот небольшой пример:

Func<int, int> a = (x) => x + 1;

Console.WriteLine(a.Invoke(1));

Чтобы найти возвращаемое значение, нам нужно найти имя реализации метода. Для этого нам нужно найти MethodDesc окружающего метода. В этом примере это Main () , поэтому:

0:000> !name2ee * TestBench.Program.Main
Module: 6db11000 (mscorlib.dll)
--------------------------------------
Module: 00162c5c (TestBench.exe)
Token: 0x06000001
MethodDesc: 00163010
Name: TestBench.Program.Main()
JITTED Code Address: 001e0070

Через MethodDesc мы можем выгрузить IL для Main ()

0:000> !dumpil 00163010
ilAddr = 003f2068
IL_0000: nop 
IL_0001: ldstr "press enter"
IL_0006: call System.Console::WriteLine     
IL_000b: nop 
IL_000c: call System.Console::ReadLine 
IL_0011: pop 
IL_0012: ldsfld TestBench.Program::CS$<>9__CachedAnonymousMethodDelegate1
IL_0017: brtrue.s IL_002c
IL_0019: ldnull 
IL_001a: ldftn TestBench.Program::<Main>b__0
IL_0020: newobj class [System.Core]System.Func`2<int32,int32>::.ctor 
IL_0025: stsfld TestBench.Program::CS$<>9__CachedAnonymousMethodDelegate1
IL_002a: br.s IL_002c
IL_002c: ldsfld TestBench.Program::CS$<>9__CachedAnonymousMethodDelegate1
IL_0031: stloc.0 
IL_0032: ldloc.0 
IL_0033: ldc.i4.1 
IL_0034: callvirt class [System.Core]System.Func`2<int32,int32>::Invoke 
IL_0039: call System.Console::WriteLine 
IL_003e: nop 
IL_003f: ret 

Обратите внимание на забавно выглядящие имена. Это имена типа делегата создания и фактического метода. Метод называется

b__0 . Давайте посмотрим на метод:

0:000> !name2ee * TestBench.Program.<Main>b__0
Module: 6db11000 (mscorlib.dll)
--------------------------------------
Module: 00152c5c (TestBench.exe)
Token: 0x06000003
MethodDesc: 00153024
Name: TestBench.Program.<Main>b__0(Int32)
Not JITTED yet. Use !bpmd -md 00153024 to break on run. 

Вот и все. MethodDesc - 00153024, и, как сказано в комментарии, вы можете использовать! Bpmd для установки точки останова с помощью MethodDesc.

9
ответ дан 5 December 2019 в 18:58
поделиться

Если найти имя «<> ....» сложно для вашего сценария, как насчет того, чтобы сделать его обычным методом? Обычно это очень просто; единственная сложность - это фиксированные переменные, но это не так уж плохо - например, они делают то же самое:

    static void Main()
    {
        List<int> list = new List<int> { 1, 2, 3, 4, 5 };
        int div = 2;
        foreach (var item in list.Where(x => x % div == 0))
        {
            Console.WriteLine(item);
        }

        ListSearcher ls = new ListSearcher();
        ls.div = 2;
        foreach (var item in list.Where(ls.Test))
        {
            Console.WriteLine(item);
        }
    }
    class ListSearcher
    {
        public int div;
        public bool Test(int x)
        {
            return x % div == 0;
        }
    }
0
ответ дан 5 December 2019 в 18:58
поделиться
Другие вопросы по тегам:

Похожие вопросы: