एक ऐसी ही आवश्यकता थी - इकाई एक सर्वर घटक परीक्षण की पुष्टि करने के लिए यह है कि कॉलबैक प्रतिनिधि आमंत्रण एक उपयुक्त SynchronizationContext पर मार्शल थे और follo के साथ आया था विंग कोड (स्टीफन टब के ब्लॉग पोस्ट http://blogs.msdn.com/b/pfxteam/archive/2012/01/20/10259049.aspx पर आधारित) जो मैं पुनः सरल और अधिक सामान्य हूं क्योंकि यह Post()
/Send()
अनुरोधों को प्रेषित करने के लिए WPF/Winforms/.. पर निर्भर करने के बजाय, अपने आंतरिक थ्रेड का उपयोग करता है।
// A simple SynchronizationContext that encapsulates it's own dedicated task queue and processing
// thread for servicing Send() & Post() calls.
// Based upon http://blogs.msdn.com/b/pfxteam/archive/2012/01/20/10259049.aspx but uses it's own thread
// rather than running on the thread that it's instanciated on
public sealed class DedicatedThreadSynchronisationContext : SynchronizationContext, IDisposable
{
public DedicatedThreadSynchronisationContext()
{
m_thread = new Thread(ThreadWorkerDelegate);
m_thread.Start(this);
}
public void Dispose()
{
m_queue.CompleteAdding();
}
/// <summary>Dispatches an asynchronous message to the synchronization context.</summary>
/// <param name="d">The System.Threading.SendOrPostCallback delegate to call.</param>
/// <param name="state">The object passed to the delegate.</param>
public override void Post(SendOrPostCallback d, object state)
{
if (d == null) throw new ArgumentNullException("d");
m_queue.Add(new KeyValuePair<SendOrPostCallback, object>(d, state));
}
/// <summary> As
public override void Send(SendOrPostCallback d, object state)
{
using (var handledEvent = new ManualResetEvent(false))
{
Post(SendOrPostCallback_BlockingWrapper, Tuple.Create(d, state, handledEvent));
handledEvent.WaitOne();
}
}
public int WorkerThreadId { get { return m_thread.ManagedThreadId; } }
//=========================================================================================
private static void SendOrPostCallback_BlockingWrapper(object state)
{
var innerCallback = (state as Tuple<SendOrPostCallback, object, ManualResetEvent>);
try
{
innerCallback.Item1(innerCallback.Item2);
}
finally
{
innerCallback.Item3.Set();
}
}
/// <summary>The queue of work items.</summary>
private readonly BlockingCollection<KeyValuePair<SendOrPostCallback, object>> m_queue =
new BlockingCollection<KeyValuePair<SendOrPostCallback, object>>();
private readonly Thread m_thread = null;
/// <summary>Runs an loop to process all queued work items.</summary>
private void ThreadWorkerDelegate(object obj)
{
SynchronizationContext.SetSynchronizationContext(obj as SynchronizationContext);
try
{
foreach (var workItem in m_queue.GetConsumingEnumerable())
workItem.Key(workItem.Value);
}
catch (ObjectDisposedException) { }
}
}
स्रोत
2015-07-30 02:12:24
यह बहुत अच्छा और अच्छा और सरल दिखता है। क्या यह वास्तव में उत्तर दिया गया था और पूछे जाने के लगभग 3 साल बाद स्वीकार किया ?! मैं इसे आजमाने की कोशिश कर रहा हूं, लेकिन "क्या यह काम करता है?" –
@ जोनाथन रेनहार्ट - हाँ, यह मेरे देव परीक्षण के दौरान काम करता है, अगर आपको इसके साथ कोई समस्या मिलती है तो कृपया मुझे बताएं ताकि हम जवाब में सुधार कर सकें। – Bond