Current languages and libraries provide little support for asynchronous and event-based programming. This forces developers to use an explicit continuation passing style by breaking code into many disjointed event-handlers. The lack of an accessible programming model for asynchronous programming is quickly becoming problematic for developers because of the inevitable advent of multi-core computers, distributed computing, and cloud computing, for example for which asynchronous programming is a necessity.
The following presents a simplified summary in order to provide a basic understanding of some novel embodiments described herein. This summary is not an extensive overview, and it is not intended to identify key/critical elements or to delineate the scope thereof. Its sole purpose is to present some concepts in a simplified form as a prelude to the more detailed description that is presented later.
The disclosed architecture leverages the mathematical duality established herein between an asynchronous observable design pattern and a synchronous iterator design pattern. Moreover, this provides a mechanism for processing multiple push-based streams (also referred to as observable collections) and values associated with those streams. This includes, for example, situations where a single push-based stream is directed to multiple subscribers or multiple push-based streams are directed to a single subscriber. Operators are presented that facilitate multi-stream processing based on this proven duality.
As a result of this duality concurrent asynchronous and event-driven programs can be elegantly formulated (e.g., using standard language integrated query (LINQ) query comprehensions). Consequently, asynchronous and event-based programming can be unified into single conceptual framework, based on sound mathematical principles such as monads and duality.
To the accomplishment of the foregoing and related ends, certain illustrative aspects are described herein in connection with the following description and the annexed drawings. These aspects are indicative of the various ways in which the principles disclosed herein can be practiced and all aspects and equivalents thereof are intended to be within the scope of the claimed subject matter. Other advantages and novel features will become apparent from the following detailed description when considered in conjunction with the drawings.
The disclosed architecture begins by proving that the synchronous Iterator (“pull-based” streams) and asynchronous Subject/Observer (“push-based” streams) design patterns are mathematically duals. Leveraging this duality provides an implementation of the standard query operators for implementing the correct causality relationships between nested push-based collections for flattening nested observable collections, of which there are several possibilities to combine the nested observable collections. From this abstract duality, a concrete implementation of standard sequence operators can be derived for observable (“push-based”) collections based on a pair of IObservable and IObserver interfaces that are the mirror images of the enumerable (“pull-based”) collections defined using an IEnumerable and IEnumerator pair of interfaces. As a result of this mathematical duality concurrent asynchronous and event-driven programs can be formulated using, for example, standard query comprehensions, and other types of comprehensions.
Iterator*Subject/Observer Duality
The notion of mathematical duality is a very powerful tool that provides “buy one, get one free” in mathematics and engineering. For example, De Morgan's law exploits the duality between conjunction && and disjunction ∥ to prove that negation! distributes over both conjunction and disjunction:
!(a&&b)==!a∥!b
!(a∥b)==!a&&!b
Another example of duality in computer science is the duality between call-by-value and call-by-name. According to one existing source, duality in category theory can be formally defined as follows:
Let Σ be any statement of the elementary theory of an abstract category. The dual of Σ can be formed as follows:
Replace each occurrence of “domain” in Σ with “codomain”, and vice versa.
Replace each occurrence of g°f=h with f°g=h
Informally, these conditions state that the dual of a statement is formed by reversing arrows and compositions.
The disclosed architecture exploits a similar mathematical duality by relating the notion of asynchronous (push-based) collections to synchronous (pull-based) collections.
Begin with the well-known iterator design pattern for enumerable collections, as embodied in the .NET framework via the pair of IEnumerable<T> and IEnumerator<T> interfaces, and in Java as Iterator<T> and Iterable<T>:
The push-based observable collection is obtained by systematically reversing the signatures of all the members of IEnumerable<T> and IEnumerator<T>, to obtain the following pair of dual interfaces. Only the actual IEnumerator<T> interface is dualized, and the IDisposable aspect retained when dualizing GetEnumerator to Subscribe, since the intention is only to dualize the collection aspect of enumerable collections, but keep the resource management aspect invariant across the two kinds of collections (consider IEnumerator as an intersection of a pair of a IDisposable and a pure IEnumerator interfaces). The Current property or the MoveNext method may throw an exception, which is an implicit return value, and which becomes an explicit parameter of type Exception to the OnError method in the dual interface.
The protocol for the IEnumerator<T> interface is that once MoveNext( ) has returned false, it will return false for each successive call. Hence, in the dual case, the Boolean argument can be encoded to OnCompleted as true by calling it, and as false by not calling it, instead of passing it as an actual argument in each call. For symmetry between OnNext, OnError and OnCompleted, the property OnNext is changed into a method as opposed to a property. Making these final adjustments, the following pair of interfaces are arrived at for observable collections:
In other words, it has been shown that the observer and iterator design patterns are mathematical duals.
The asynchronous nature of observable collections presents challenges in the implementation of the standard sequence operators when compared to the implementation for enumerable collections. Fortunately, the duality between observable and enumerable collections serves as a guide towards the correct implementation.
Reference is now made to the drawings, wherein like reference numerals are used to refer to like elements throughout. In the following description, for purposes of explanation, numerous specific details are set forth in order to provide a thorough understanding thereof. It may be evident, however, that the novel embodiments can be practiced without these specific details. In other instances, well known structures and devices are shown in block diagram form in order to facilitate a description thereof. The intention is to cover all modifications, equivalents, and alternatives falling within the spirit and scope of the claimed subject matter.
The operators 106 are based on the dual relationship established between the observer design pattern and the iterator design pattern. The processing component 104 includes an operator that applies a function to an asynchronous value to create an inner observable collection. The processing component 104 includes an operator that non-deterministically merges asynchronous values of multiple observable collections to the target observable collection 110, while maintaining a causality relationship of the values. The processing component 104 includes an operator that propagates values from a source observable collection (e.g., one of the observable collections 102) to the target observable collection 110 until an asynchronous value occurs on a control observable collection. The processing component 104 can also include an operator that allows values of an observable collection to be shared within one or more subscribers, and an operator that drops a first value of an observable collection based on another value. The processing component 104 also includes an operator that transitions between a nested observable collection to another nested observable collection.
The processing component 104 includes a policy that broadcasts values based on ignoring asynchronous values of one observable collection relative to termination of another observable collection, and a policy that terminates an observable collection in response to creation of another observable collection. The processing component 104 includes policies that record asynchronous values of a next observable collection until a previous observable collection terminates, and plays back the recorded values according to time information associated with the arrival of the values (e.g., on the first observable collection).
One or more of the policies 202 can be applied to each of the mechanisms (as indicated by the “X”). The policies 202 include, but are not limited to, a Replace policy 210, Broadcast policy 212, Record policy 214, Replay policy 216, and variations on the Replay policy 216. The policy variations can include, but are not limited to, recording and replay, for example. Policies such as a ReplayWithTiming policy 218, ReplayWithTime policy 220, ReplayByItems policy 222, ReplayWithTime&Timing policy 224, and ReplayWithTime&Items policy 226, for example, represent variations that can be employed in combination with Replay. Although not illustrated, variations can also be applied to Broadcast, such as broadcast based on time, timing, and so on.
The Replace policy 210 forwards a previous collection until a next collection starts. The Broadcast policy 212 broadcasts values to one or more observers. The Record policy 214 records (buffers) values of an observable collection and the Replay policy 216 plays back the recorded values.
The Record policy 214 and Replay policy 216 can operate in many different ways. For example, if values are recorded with timing differential information that indicates the time between values when recorded. The ReplayWithTiming policy 218 replays the values according to the timing differential information (“timing”) when recorded.
Replay can also be according to a specific time by using the ReplayWithTime policy 220. Replay can also be according to item count by using the ReplayByItems policy 222. Further combinations can include replay initiated at a specific time, and then according to the recorded timing differential information, by using the ReplayWithTiming&Time policy 224. A still further combination can include replaying values at a specific time, and then by item count, by using the ReplayWithTime&Items policy 226.
As previously indicated, other variations can include a RecordByItem policy (not shown) that records a finite number of values, and a RecordWithTiming policy (not shown) that records the timing information (e.g., of time arrival) when recording the values, for example.
Note that there can be many implementations of Select, Flatten, and Share, for example, not all of which are specifically illustrated but follow from the described illustrations and embodiments.
The above pseudocode uses the syntactic sugar of foreach loops and yield return iterators, which a C# compiler can expand into calls to GetEnumerator and MoveNext( )/Current, and a complex state-machine that maintains all control-flow and variable state across interruptions, respectively. Hence, the actual code generated by the compiler is much more complex.
For observable collections of type IObserver<T>, the implementation of Select immediately returns a new target observable of type IObserver<S> that when an observer of type is attached to the target, a new observer of IObserver<T> is attached to the original source. When that observer is notified with a value of type T, it will try to run the selector on the value, and if successful, notify the transformed value of type S, to the target observable collection. If the selector throws an exception or if the source collection signals OnError, the exception is immediately thrown to the target observable collection. If the source observable collection sends a Dispose message, the new observer is removed from the source collection. The pseudocode implementing the Select function on an observable collection can appear as the following:
A naïve observable flatten operation supports the implementation of operators (e.g., LINQ query operators), and is shown as IObservable<T> Flatten(this IObservable<IObservable<T>> src). The push nature of observable collections makes this non-trivial.
Here, the values of each inner collection are grouped when pushed on the target (merged) observable collection. This grouping is obtained by merging the values of a next inner collection (e.g., middle inner collection) only after the values of the previous inner collection (e.g., top inner collection) have been merged based on termination of the associate inner collection. This sorted merge is shown on the target collection 502, where three values from the top inner collection are merged prior to termination of the top inner collection, followed by two values of the middle inner collection following termination of the top inner collection, followed by three values of the bottom inner collection following termination of the middle inner collection.
In an alternative implementation, while processing the top values of the top inner collection 604, if a value comes in on the middle inner collection 602, processing immediately jumps to the middle inner collection 602 regardless of values of the top inner collection 604. The values of the top inner collection 604 (previous) are forgotten (or lost). Still alternatively, the buffered values can be saved with relative timing information (timing) and then be pushed out according to the relative timing information. In yet another alternative implementation, only values that occurred during a previous time period (e.g., last fifteen minutes) or according to a count (e.g., last fifty) are buffered.
Referring back to
Using all these operators, the replacing version of Flatten for observable collections can be defined via the following intimidating expressions (referred to collectively as a Tarzan algorithm) that terminate each nested collection as soon as a next nested collection is produced:
The values are broadcast for sharing to other collections 1318, an existing subscriber 1320 that receives all values of the source and a new subscriber 1322 that only receives values occurring after the subscription time. Values 1324 prior to the subscription time are not shared.
The values are broadcast to an existing subscriber 1416 that receives the values of the source collection. A new subscriber 1418 replaces the existing subscriber 1416 when the new subscription is added; however, the new subscriber does not receive values prior to the subscription time.
More specifically, when a first value 1502 occurs on the source collection, the function F is applied to the first value 1502 to create a first inner collection 1504. The values occurring on the first inner collection 1504 are merged to the target collection, and the temporal information associated with each value arrival in that collection. Concurrently, when a second value 1506 occurs on the source observable collection, a second inner collection 1508 is created using the function F and, the second inner collection 1508 and temporal information are buffered. When the first inner collection 1504 terminates, the buffered second inner collection 1508 is played back and its associated values 1510 are merged to the target collection using the recorded temporal information.
Similarly, while the second inner collection 1508 plays, a third source value 1512 occurs, on the source collection. Application of the function F creates a third inner collection 1514 with arrival interval times for its values 1516. The third inner collection 1514, values 1516 and associated time interval information (also referred to a time differential information or “timing”) is recorded until the second inner observable collection 1508 terminates. Thereafter, the third inner collection 1514, values 1516, and time interval information is played back to the target observable collection.
Included herein is a set of flow charts representative of exemplary methodologies for performing novel aspects of the disclosed architecture. While, for purposes of simplicity of explanation, the one or more methodologies shown herein, for example, in the form of a flow chart or flow diagram, are shown and described as a series of acts, it is to be understood and appreciated that the methodologies are not limited by the order of acts, as some acts may, in accordance therewith, occur in a different order and/or concurrently with other acts from that shown and described herein. For example, those skilled in the art will understand and appreciate that a methodology could alternatively be represented as a series of interrelated states or events, such as in a state diagram. Moreover, not all acts illustrated in a methodology may be required for a novel implementation.
As used in this application, the terms “component” and “system” are intended to refer to a computer-related entity, either hardware, a combination of hardware and software, software, or software in execution. For example, a component can be, but is not limited to being, a process running on a processor, a processor, a hard disk drive, multiple storage drives (of optical, solid state, and/or magnetic storage medium), an object, an executable, a thread of execution, a program, and/or a computer. By way of illustration, both an application running on a server and the server can be a component. One or more components can reside within a process and/or thread of execution, and a component can be localized on one computer and/or distributed between two or more computers. The word “exemplary” may be used herein to mean serving as an example, instance, or illustration. Any aspect or design described herein as “exemplary” is not necessarily to be construed as preferred or advantageous over other aspects or designs.
Referring now to
The computing system 2100 for implementing various aspects includes the computer 2102 having processing unit(s) 2104, a system memory 2106, and a system bus 2108. The processing unit(s) 2104 can be any of various commercially available processors such as single-processor, multi-processor, single-core units and multi-core units. Moreover, those skilled in the art will appreciate that the novel methods can be practiced with other computer system configurations, including minicomputers, mainframe computers, as well as personal computers (e.g., desktop, laptop, etc.), hand-held computing devices, microprocessor-based or programmable consumer electronics, and the like, each of which can be operatively coupled to one or more associated devices.
The system memory 2106 can include volatile (VOL) memory 2110 (e.g., random access memory (RAM)) and non-volatile memory (NON-VOL) 2112 (e.g., ROM, EPROM, EEPROM, etc.). A basic input/output system (BIOS) can be stored in the non-volatile memory 2112, and includes the basic routines that facilitate the communication of data and signals between components within the computer 2102, such as during startup. The volatile memory 2110 can also include a high-speed RAM such as static RAM for caching data.
The system bus 2108 provides an interface for system components including, but not limited to, the memory subsystem 2106 to the processing unit(s) 2104. The system bus 2108 can be any of several types of bus structure that can further interconnect to a memory bus (with or without a memory controller), and a peripheral bus (e.g., PCI, PCIe, AGP, LPC, etc.), using any of a variety of commercially available bus architectures.
The computer 2102 further includes storage subsystem(s) 2114 and storage interface(s) 2116 for interfacing the storage subsystem(s) 2114 to the system bus 2108 and other desired computer components. The storage subsystem(s) 2114 can include one or more of a hard disk drive (HDD), a magnetic floppy disk drive (FDD), and/or optical disk storage drive (e.g., a CD-ROM drive DVD drive), for example. The storage interface(s) 2116 can include interface technologies such as EIDE, ATA, SATA, and IEEE 1394, for example.
One or more programs and data can be stored in the memory subsystem 2106, a removable memory subsystem 2118 (e.g., flash drive form factor technology), and/or the storage subsystem(s) 2114 (e.g., optical, magnetic, solid state), including an operating system 2120, one or more application programs 2122, other program modules 2124, and program data 2126.
The one or more application programs 2122, other program modules 2124, and program data 2126 can include the entities and components of system 100 of
Generally, programs include routines, methods, data structures, other software components, etc., that perform particular tasks or implement particular abstract data types. All or portions of the operating system 2120, applications 2122, modules 2124, and/or data 2126 can also be cached in memory such as the volatile memory 2110, for example. It is to be appreciated that the disclosed architecture can be implemented with various commercially available operating systems or combinations of operating systems (e.g., as virtual machines).
The storage subsystem(s) 2114 and memory subsystems (2106 and 2118) serve as computer readable media for volatile and non-volatile storage of data, data structures, computer-executable instructions, and so forth. Computer readable media can be any available media that can be accessed by the computer 2102 and includes volatile and non-volatile internal and/or external media that is removable or non-removable. For the computer 2102, the media accommodate the storage of data in any suitable digital format. It should be appreciated by those skilled in the art that other types of computer readable media can be employed such as zip drives, magnetic tape, flash memory cards, flash drives, cartridges, and the like, for storing computer executable instructions for performing the novel methods of the disclosed architecture.
A user can interact with the computer 2102, programs, and data using external user input devices 2128 such as a keyboard and a mouse. Other external user input devices 2128 can include a microphone, an IR (infrared) remote control, a joystick, a game pad, camera recognition systems, a stylus pen, touch screen, gesture systems (e.g., eye movement, head movement, etc.), and/or the like. The user can interact with the computer 2102, programs, and data using onboard user input devices 2130 such a touchpad, microphone, keyboard, etc., where the computer 2102 is a portable computer, for example. These and other input devices are connected to the processing unit(s) 2104 through input/output (I/O) device interface(s) 2132 via the system bus 2108, but can be connected by other interfaces such as a parallel port, IEEE 1394 serial port, a game port, a USB port, an IR interface, etc. The I/O device interface(s) 2132 also facilitate the use of output peripherals 2134 such as printers, audio devices, camera devices, and so on, such as a sound card and/or onboard audio processing capability.
One or more graphics interface(s) 2136 (also commonly referred to as a graphics processing unit (GPU)) provide graphics and video signals between the computer 2102 and external display(s) 2138 (e.g., LCD, plasma) and/or onboard displays 2140 (e.g., for portable computer). The graphics interface(s) 2136 can also be manufactured as part of the computer system board.
The computer 2102 can operate in a networked environment (e.g., IP-based) using logical connections via a wired/wireless communications subsystem 2142 to one or more networks and/or other computers. The other computers can include workstations, servers, routers, personal computers, microprocessor-based entertainment appliances, peer devices or other common network nodes, and typically include many or all of the elements described relative to the computer 2102. The logical connections can include wired/wireless connectivity to a local area network (LAN), a wide area network (WAN), hotspot, and so on. LAN and WAN networking environments are commonplace in offices and companies and facilitate enterprise-wide computer networks, such as intranets, all of which may connect to a global communications network such as the Internet.
When used in a networking environment the computer 2102 connects to the network via a wired/wireless communication subsystem 2142 (e.g., a network interface adapter, onboard transceiver subsystem, etc.) to communicate with wired/wireless networks, wired/wireless printers, wired/wireless input devices 2144, and so on. The computer 2102 can include a modem or other means for establishing communications over the network. In a networked environment, programs and data relative to the computer 2102 can be stored in the remote memory/storage device, as is associated with a distributed system. It will be appreciated that the network connections shown are exemplary and other means of establishing a communications link between the computers can be used.
The computer 2102 is operable to communicate with wired/wireless devices or entities using the radio technologies such as the IEEE 802.xx family of standards, such as wireless devices operatively disposed in wireless communication (e.g., IEEE 802.11 over-the-air modulation techniques) with, for example, a printer, scanner, desktop and/or portable computer, personal digital assistant (PDA), communications satellite, any piece of equipment or location associated with a wirelessly detectable tag (e.g., a kiosk, news stand, restroom), and telephone. This includes at least Wi-Fi (or Wireless Fidelity) for hotspots, WiMax, and Bluetooth™ wireless technologies. Thus, the communications can be a predefined structure as with a conventional network or simply an ad hoc communication between at least two devices. Wi-Fi networks use radio technologies called IEEE 802.11x (a, b, g, etc.) to provide secure, reliable, fast wireless connectivity. A Wi-Fi network can be used to connect computers to each other, to the Internet, and to wire networks (which use IEEE 802.3-related media and functions).
What has been described above includes examples of the disclosed architecture. It is, of course, not possible to describe every conceivable combination of components and/or methodologies, but one of ordinary skill in the art may recognize that many further combinations and permutations are possible. Accordingly, the novel architecture is intended to embrace all such alterations, modifications and variations that fall within the spirit and scope of the appended claims. Furthermore, to the extent that the term “includes” is used in either the detailed description or the claims, such term is intended to be inclusive in a manner similar to the term “comprising” as “comprising” is interpreted when employed as a transitional word in a claim.