You should be familiar with Communications (RPC in particular) before reviewing this sample. The overall structure is very similar to the RPC sample.
This sample doesn't use any custom interests or notifications, so in other respects it is most similar to BasicNotification.
1) It derives from MSenderProtocol and provides a concrete implementation for its abstractions.
2) It derives from MDelegatingNotifier, so that when used on the dispatcher side of the RPC, it can use the MDispatcherNotifier. For convenience, it implements GetNotifier to create an instance of TNotifier, if it has not already been told to use another notifier. TSender class could be used as is in the previous samples without other changes to them. The flag fOwnsNotifier is used to indicate whether the notifier is one that was set "from outside," or one it created for its own use.
3) It defines SetNotifier so that clients can tell it to use a particular notifier. Note that this is not protocol of MDelegatingNotifier (GetNotifier is), but makes it convenient for TSenderDispatcher to use an instance of TSender as its implementation.
4) It implements GetValueChangedInterest to construct an interest with the notifier returned by GetNotifier, because it is no longer itself a notifier.
1) It derives from MRemoteCaller for RPC.
2) It derives from MRemoteCallerNotifier so that it can relay notifications sent by the MRemoteDispatcherNotifier on the other side of the RPC connection to receivers that have connected to it.
3) It derives from MSenderProtocol and provides a concrete implementation for its abstractions.
4) In its constructor, it sets the MRemoteCallerNotifier to use the MRemoteCaller it has mixed in with. It calls MRemoteCallerEnable, which RPC requires of classes derived from MRemoteCaller.
5) It implements GetValue and SetValue to use RPC to stream the requests and data to the dispatcher.
6) It uses a monitor lock so that the same sender caller may be used by multiple threads (in this example, the main thread and the receiver thread).
1) It derives from MRemoteDispatcher for RPC.
2) It derives from MRemoteDispatcherNotifier so that notifications sent from it are relayed to the MRemoteCallerNotifier on the other side of the RPC connection.
3) In its constructor, it adopts an instance of TSender. It uses the fact that TSender is now an MDelegatingNotifier, and has defined protocol to set the notifier it uses, to tell the adopted TSender to use it as its notifier. Thus, when TSender calls Notify, the MRemoteDispatcherNotifier is used, and notifications pass to the MRemoteCallerNotifier on the other side.
4) It implements FwdAddRemoteInterest and FwdRemoveRemoteInterest to forward these calls to the MRemoteDispatcherNotifier. In the constructor, it uses standard RPC boilerplate to register these requests with the MRemoteDispatcher.
5) It implements SetValueStub and GetValueStub to extract the requests and data from the stream and forward these calls to the TSender implementation. In the constructor, it uses standard RPC boilerplate to register these requests with the MRemoteDispatcher.
Asynchronous communication, as a result of RPC, means that the receiver can receive notifications some time after the notifications were sent. In this example, by the time the TReceiver can process any notifications, the caller has been changed multiple times, so when the receiver tries to query the state of the sender, it is no longer the same as when the notifications were sent.
The notifications are received in a separate thread from the main thread. Both the main thread and the receiver thread use the same caller-- the main thread calls SetValue, and in the receiver thread the receiver uses the same caller to call GetValue. Each call uses the RPC mechanism, but it is an error (an exception will be thrown) to start an RPC call while another is still pending. So TSenderCaller contains a lock, and each call that uses RPC acquires a monitor on the lock before trying to do RPC. In this way the two threads are prevented from attempting RPC at the same time.
main() uses a delay to block this thread so that the thread that dispatches the notifications can run. Otherwise, the main thread might exit before all notifications have been received.