Rumah > pembangunan bahagian belakang > C++ > Bagaimanakah Saya Boleh Menunggu Sebarang Acara pada Mana-mana Jenis Menggunakan Kaedah FromEvent Tujuan Am?

Bagaimanakah Saya Boleh Menunggu Sebarang Acara pada Mana-mana Jenis Menggunakan Kaedah FromEvent Tujuan Am?

Mary-Kate Olsen
Lepaskan: 2024-12-31 22:38:19
asal
908 orang telah melayarinya

How Can I Await Any Event on Any Type Using a General-Purpose FromEvent Method?

Kaedah FromEvent Tujuan Umum

Kaedah FromEvent asal memerlukan mencipta kaedah berasingan untuk setiap acara dan kelas yang anda mahu tunggu. Untuk menghapuskan kod boilerplate, pembangun mencari penyelesaian yang lebih umum.

Satu pendekatan melibatkan penggunaan refleksi untuk mendapatkan semula acara dan jenis perwakilannya. Walau bagaimanapun, mengakses parameter perwakilan dan mengubah suai TaskCompletionSource menimbulkan cabaran.

Penyelesaian Tersuai

Berikut ialah penyelesaian tersuai yang menggunakan pelepasan IL dan pengendali acara:

public static class ExtensionMethods
{
    public static Task<object[]> FromEvent<T>(this T obj, string eventName)
    {
        // Create a TaskCompletionSourceHolder to manage the TaskCompletionSource and event handler
        var tcsh = new TaskCompletionSourceHolder();

        // Get the event and its delegate type
        EventInfo eventInfo = obj.GetType().GetEvent(eventName);
        Type eventDelegateType = eventInfo.EventHandlerType;

        // Create a dynamic method to handle the event
        DynamicMethod handler = CreateEventHandler(eventDelegateType);

        // Add the event handler to the target object
        eventInfo.AddEventHandler(obj, handler.CreateDelegate(eventDelegateType, tcsh));

        // Return the Task from the TaskCompletionSourceHolder
        return tcsh.Task;
    }

    private static DynamicMethod CreateEventHandler(Type eventDelegateType)
    {
        // Get the parameter types of the event delegate
        var parameterTypes = GetDelegateParameterTypes(eventDelegateType);

        // Insert the TaskCompletionSourceHolder as the first parameter
        parameterTypes.Insert(0, typeof(TaskCompletionSourceHolder));

        // Create a new dynamic method and IL generator
        DynamicMethod handler = new DynamicMethod("EventHandler", typeof(void), parameterTypes, true);
        ILGenerator ilgen = handler.GetILGenerator();

        // Load the TaskCompletionSourceHolder and create an array to store the arguments
        ilgen.Emit(OpCodes.Ldarg_0);
        ilgen.Emit(OpCodes.Ldc_I4, parameterTypes.Count - 1);
        ilgen.Emit(OpCodes.Newarr, typeof(object));

        // Store each argument in the array
        for (int i = 1; i < parameterTypes.Count; i++)
        {
            ilgen.Emit(OpCodes.Ldloc_0);
            ilgen.Emit(OpCodes.Ldc_I4, i - 1);
            ilgen.Emit(OpCodes.Ldarg, i);
            ilgen.Emit(OpCodes.Stelem_Ref);
        }

        // Call the SetResult method on the TaskCompletionSourceHolder
        ilgen.Emit(OpCodes.Call, typeof(TaskCompletionSourceHolder).GetMethod("SetResult"));

        // Return from the method
        ilgen.Emit(OpCodes.Ret);

        return handler;
    }

    private static List<Type> GetDelegateParameterTypes(Type delegateType)
    {
        var parameters = delegateType.GetMethod("Invoke").GetParameters();
        var parameterTypes = parameters.Select(p => p.ParameterType).ToList();
        return parameterTypes;
    }
}
Salin selepas log masuk

Dengan kaedah ini, anda kini boleh menunggu sebarang acara di mana-mana jenis:

await new MyClass().FromEvent("MyEvent");
Salin selepas log masuk

Faedah Penyelesaian ini

  • Menyokong sebarang jenis acara dengan sebarang nombor atau jenis parameter.
  • Tidak memerlukan penghantaran Tugas kepada jenis pemulangan tertentu.
  • Menghapuskan keperluan untuk mencipta berbilang kaedah FromEvent untuk berbeza jenis dan acara.

Atas ialah kandungan terperinci Bagaimanakah Saya Boleh Menunggu Sebarang Acara pada Mana-mana Jenis Menggunakan Kaedah FromEvent Tujuan Am?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan