Obtener el nombre de la función que ejecuta un hilo.

Iniciado por El Benjo, 18 Febrero 2016, 07:36 AM

0 Miembros y 2 Visitantes están viendo este tema.

El Benjo

Hola. Buen día.

No encuentro la manera de obtener el nombre del método que está ejecutando un hilo en particular.

Ya busqué dentro de la clase ProcessThread pero creo que esa clase no me da esa información.

Muchas gracias por su ayuda.
www.es.neftis-ai.com

Sí hay un mejor lenguaje de programación y es ese con el que puedes desarrollar tus objetivos.

fary

Quieres saber el nombre que tu le diste a la funcion al programarla?
Para que quieres saber eso?

Saludos.
Un byte a la izquierda.

Eleкtro

#2
Creo que tu pregunta se puede interpretar de varias maneras, y de si intentas obtenerlo el nombre del método desde el thread adicional o el thread de la UI, por favor, muestra el código relevante.

De todas formas, creo que este ejemplo basado en Reflection solventará tus dudas:

Vb.Net:
Código (vbnet) [Seleccionar]
Imports System.Reflection
Imports System.Threading.Tasks

Public NotInheritable Class Main : Inherits Form

   Private Sub Test() Handles MyBase.Shown

       Dim t As New Thread(AddressOf Method1)
       t.Start()
       Debug.WriteLine("Main: " & MethodBase.GetCurrentMethod().Name) ' Test

       Task.Factory.StartNew(
           Sub()
               Debug.WriteLine("Lambda: " & MethodBase.GetCurrentMethod().Name) ' _Lambda$__n
           End Sub)

   End Sub

   Private Sub Method1()

       Dim stackTrace As New StackTrace
       Debug.WriteLine("Thread: " & MethodBase.GetCurrentMethod().Name) ' Method1

   End Sub

End Class


Traducción online a C#:
Código (csharp) [Seleccionar]
using System.Diagnostics;
using System.Reflection;
using System.Threading.Tasks;

public sealed class Main : Form {

   public Form1() {
       Shown += Test;
   }

   private void Test() {
       Thread t = new Thread(Method1);
       t.Start();
       Debug.WriteLine("Main: " + MethodBase.GetCurrentMethod().Name); // Test

       Task.Factory.StartNew(() => { Debug.WriteLine("Lambda: " + MethodBase.GetCurrentMethod().Name); }); // _Lambda$__n
   }

   private void Method1() {
       StackTrace stackTrace = new StackTrace();
       Debug.WriteLine("Thread: " + MethodBase.GetCurrentMethod().Name); // Method1
   }

}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


Resultado de ejecución:
CitarMain: Test
Thread: Method1
Lambda: _Lambda$__2




Si lo prefieres también puedes utilizar una propiedad como esta de aquí abajo para tenerla siempre a mano.

El código fuente lo he extraido de mi API ElektroKit:



Vb.Net:
Código (vbnet) [Seleccionar]
''' ----------------------------------------------------------------------------------------------------
''' <summary>
''' Gets the current executing member in the stack trace of the application.
''' </summary>
''' ----------------------------------------------------------------------------------------------------
''' <example> This is a code example.
''' <code>
''' Private Sub TestMethod()
'''
'''     MsgBox(CurrentMember.Name)
'''
''' End Sub
''' </code>
''' </example>
''' ----------------------------------------------------------------------------------------------------
''' <value>
''' The current executing member.
''' </value>
''' ----------------------------------------------------------------------------------------------------
Public Shared ReadOnly Property CurrentMember As MethodBase
   <DebuggerStepThrough>
   Get
       Dim stackTrace As New StackTrace
       Return stackTrace.GetFrame(1).GetMethod()
   End Get
End Property


Traducción online a C#:
Código (csharp) [Seleccionar]
/// ----------------------------------------------------------------------------------------------------
/// <summary>
/// Gets the current executing member in the stack trace of the application.
/// </summary>
/// ----------------------------------------------------------------------------------------------------
/// <value>
/// The current executing member.
/// </value>
/// ----------------------------------------------------------------------------------------------------
public static MethodBase CurrentMember {
get {
StackTrace stackTrace = new StackTrace();
return stackTrace.GetFrame(1).GetMethod();
}
}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


Modo de empleo:

Vb.Net:
Código (vbnet) [Seleccionar]
Imports System.Reflection
Imports System.Threading.Tasks

Public NotInheritable Class Main : Inherits Form

   Private Sub Test() Handles MyBase.Shown

       Dim t As New Thread(AddressOf Method1)
       t.Start()
       Debug.WriteLine("Main: " & CurrentMember.Name) ' Test

       Task.Factory.StartNew(
           Sub()
               Debug.WriteLine("Lambda: " & CurrentMember.Name) ' _Lambda$__n
           End Sub)

   End Sub

   Private Sub Method1()

       Dim stackTrace As New StackTrace
       Debug.WriteLine("Thread: " & CurrentMember.Name) ' Method1

   End Sub

   Public Shared ReadOnly Property CurrentMember As MethodBase
       <DebuggerStepThrough>
       Get
           Dim stackTrace As New StackTrace
           Return stackTrace.GetFrame(1).GetMethod()
       End Get
   End Property

End Class


C#:
Código (csharp) [Seleccionar]
using System.Diagnostics;
using System.Reflection;
using System.Threading.Tasks;

public sealed class Main : Form {

   public Form1() {
       Shown += Test;
   }

   private void Test() {
       Thread t = new Thread(Method1);
       t.Start();
       Debug.WriteLine("Main: " + CurrentMember.Name); // Test
       
       Task.Factory.StartNew(() => { Debug.WriteLine("Lambda: " + CurrentMember.Name); }); // _Lambda$__n
   }

   private void Method1() {
       StackTrace stackTrace = new StackTrace();
       Debug.WriteLine("Thread: " + CurrentMember.Name); // Method1
   }

   public static MethodBase CurrentMember {
       [DebuggerStepThrough()]
       get {
           StackTrace stackTrace = new StackTrace();
           return stackTrace.GetFrame(1).GetMethod();
       }
   }

}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


Resultado de ejecución:
CitarMain: Test
Thread: Method1
Lambda: _Lambda$__2

Saludos!








El Benjo

Muy bien, tienen razón, quizá no he sabido como expresar mi necesidad.

Mi aplicación hace lo siguiente:

Carga librerías dinámicas que encuentre en un directorio en específico y corre (de encontrase) el método 'start()' en cada una de ellas.

El código de las librerías puede correr varios hilos sobre distintos métodos y estos, al ser de terceros, desconozco su nombre.

Ahora la aplicación muestra los hilos que se encuentran dentro del dominio de la aplicación. Lo que necesito pero no he encontrado información para hacerlo es saber el método que está ejecutando cada uno de esos hilos y, de ser posible el ensamblado al que pertenece dicho método.

La única forma que se me ha ocurrido para hacerlo es con la propiedad ProcessThreads al obtener un objeto del proceso actual y sacar la dirección en memoria en cada uno de estos elementos y compararla con la dirección en memoria de cada uno de los métodos contenidos en cada ensamblado. Pero esto se me hace un desperdicio de ciclos de procesador, además de que tendría que explorar de manera recursiva cada clase dentro del ensamblado para obtener la dirección de su método y compararla. Y si el método está declarado en una clase fuera del ensamblado entonces ya no podré obtenerla por este método.

Elektro, el código relevante no puedo mostrarlo porque realmente no existe. Espero que exista una manera de hacer esto que pido: Conocer el nombre de los métodos que ejecuta cada hilo dentro del proceso actual, la clase o ensamblado al que pertenece y saberlo desde un hilo distinto al que lo está ejecutando.

Muchas gracias a todos quienes se tomen la molestia en responder.
www.es.neftis-ai.com

Sí hay un mejor lenguaje de programación y es ese con el que puedes desarrollar tus objetivos.

Eleкtro

#4
En resumidas cuentas tendrías que obtener el stack-trace de ese hilo o hilos, y analizar sus stack-frames para obtener el método. Pero claro, para eso debes administrar/crear tu los hilos... si los hilos son creados por algún método de dicha librería entonces sinceramente no se me ocurre una solución.

De todas formas te muestro un ejemplo por si te sirve de algo...

Observa la propiedad que publiqué, la metodología sería parecida solo que utilizando el constructor que acepta un thread:
Código (csharp) [Seleccionar]
Thread t = Thread.CurrentThread;
StackTrace st = new StackTrace(t, needFileInfo: false);

( esta metodología está considerada obsoleta por Microsoft, sin embargo, tampoco proporcionan ninguna alternativa: https://msdn.microsoft.com/en-us/library/t2k35tat%28v=vs.110%29.aspx)

Un ejemplo algo más extenso, en Vb.Net:
Código (vbnet) [Seleccionar]
Public NotInheritable Class Form1 : Inherits Form

   Private Sub Test() Handles MyBase.Shown

       Dim t As Thread = New Thread(AddressOf Method1)
       t.Start()
       Thread.Sleep(100)
       t.Suspend()

       Dim st As New StackTrace(t, needFileInfo:=False)
       Console.WriteLine(st.GetFrame(2).GetMethod.Name) ' Resultado: Method2

   End Sub

   Public Sub Method1()
       Method2()
   End Sub

   Public Sub Method2()
       Thread.Sleep(Integer.MaxValue)
   End Sub

End Class


PD: ElBenjo, ¿es necesario determinar esa información en tiempo de ejecución por algún motivo?, ¿no te basta con cojer el .Net Reflector y examinar el código fuente de la librería a ver que métodos se llaman en sus threads?.

Saludos!








El Benjo

Elektro, gracias por responder.

No, lo que pasa es que no soy yo el que necesita la información sino mi aplicación. Es decir, que toda la información que necesito es información en tiempo de ejecución y yo no sé (como ya comenté antes) que librerías son las que van a estar ejecutando código ni cual es el código. La aplicación se basa en módulos y los módulos pueden ser escritos por cualquiera, sin embargo la aplicación deber ser capaz de reconocer no sólo la cantidad de hilos que se ejecutan en el proceso sino también el módulo que los creo.

Seguiré buscando a ver si encuentro algo. Muchas gracias, Elektro. Ah, y de paso felicitarte por el trabajo de tu librería. La verdad se nota que le invertiste mucho tiempo y esfuerzo.
www.es.neftis-ai.com

Sí hay un mejor lenguaje de programación y es ese con el que puedes desarrollar tus objetivos.

Eleкtro

#6
Cita de: El Benjo en 19 Febrero 2016, 00:28 AMla aplicación deber ser capaz de reconocer no sólo la cantidad de hilos que se ejecutan en el proceso sino también el módulo que los creo.

Al parecer eso que ahora mencionas es imposible de monitorizar al no poder ser notificado por la propia aplicación, según un respuesta comentario del gran mago y gurú de .Net, Hans Passant:
( desafortunadamente para ti, te recomiendo que no le des más vueltas, si lo dice Hans Passant no hay más que decir, es un genio del conocimiento qué nunca se equivoca. )

Sin embargo, al parecer existe algún proyecto como este de aquí abajo, el cual, supuestamente, desde un proceso externo a tu app, es capaz de monitorizar la creación de threads de tu app.

Cita de: MSEMSE works by quickly attaching to a process when a stack trace is requested and the detaching the second the stack trace has been retrieved.
This way the interference in the normal operation of the process is minimized.

This means it has to done by an external process.
Parece interesante, no lo he testeado ...pero imagino que te podría servir. Y si no te sirve del todo pues... lo bueno es que puedes coger el source del MSE y adaptarlo a tus necesidades.

Otro proyecto parecido sería este:
Tampoco lo he testeado ni se si tiene las mismas funcionalidades que el otro.

PD: Gracias por el comentario del ElektroKit !!

Saludos!








El Benjo

Muchíssimas gracias por la ayuda, Elektro y por tomarte el tiempo para investigar. La verdad es que llevo casi una semana investigado y no he podido encontrar nada. Veré si eso me sirve, y si no, pues adaptaré el código que mencionas. La otra opción es modificar mi aplicación. Sin embargo, somos programadores y como tal, no debería haber un límite en lo que podemos crear, por eso me rehusaré lo más que pueda a modificar los objetivos de mi aplicación.

Un gran saludo, Elektro.
www.es.neftis-ai.com

Sí hay un mejor lenguaje de programación y es ese con el que puedes desarrollar tus objetivos.