Real Time Data
Real Time Data (or RTD) is data that updates according to it’s own schedule, not just when it is re-evaluated (as is the case for a regular Excel worksheet function).
Examples of real time data include stock prices and other live market data, server loads or the progress of an external task.
Real Time Data has been a first-class feature of Excel since Excel 2002. It uses a hybrid push-pull mechanism where the source of the real time data notifies Excel that new data is available, and then some small time later Excel queries the real time data source for it’s current value and updates the value displayed.
Streaming Data with Jinx
Jinx provides a convenient and simple way to stream real time data to Excel without the complexity of writing (and registering) a Real Time Data COM server.
Real Time Data functions are registered in the same way as other worksheet functions using the @ExcelFunction annotation. Instead of returning a single fixed value, however, they return an instance of an class derived from com.exceljava.jinx.Rtd.
When a function returns an Rtd instance Jinx sets up the real time data subscription in Excel and each time the notify method of the Rtd instance is called Excel is notified that new data is ready.
If multiple function calls from different cells return the same instance of an Rtd class then those cells are subscribed to the same real time data source, so they will all update whenever the notify method is called.
Example Usage
The following example shows a class derived from Rtd that periodically updates its value to the current time.
It uses a ScheduledExecutorService to notify Excel with a new value periodically.
In order to access this real time data in Excel all that’s required is a worksheet function that returns
an instance of this CurrentTimeRtd
class.
Note that the return type of this function is Rtd<String>
.
The generic parameter is the type that eventually gets returned to Excel. If the return type can’t be converted to something Excel can understand then the result will be cached and a handle to that object returned (see cached objects). Array types can also be returned, but they can’t be resized automatically.
When this function is called from Excel the value displayed will periodically update, even though the function jinx.currentTime only gets called once.
=jinx.currentTime('yyyy/MM/dd HH:mm:ss')
Volatile RTD Functions
When Excel opens a workbook it will show the saved values for any formulas until they need recalculating.
For RTD functions this means that the value doesn’t start ticking until the workbook is recalculated, which isn’t always what we want.
Volatile functions (registered with isVolatile = true
using the @ExcelFunction annotation) are
always called when the workbook is opened. We can use that to ensure an RTD function starts
ticking as soon as the workbook is opened.
Usually, marking a worksheet function as volatile also means that it will be called whenever any change happens on the sheet. For RTD functions Jinx takes care of that and will always returns the same Rtd instance if the function is repeatedly called from the same cell with the same arguments.
Restarting RTD Functions
Usually, formulas can be re-evaluated by pressing Ctrl+Alt+F9 in Excel, and that will cause the functions to be re-run.
For RTD functions however, Jinx cached the Rtd instance returned from the RTD function and the cached instance will be returned unless the inputs to the function have changed.
This is necessary for RTD functions to work correctly, but it has the sometimes unwanted effect that recalculating using Ctrl+Alt+F9 does not restart the RTD function since the cached Rtd object is returned.
Instead, in order to restart an RTD function either the inputs need to change, or the formula needs to be re-entered, or the Rtd instance can be removed from the cache using its Rtd.detach() method.
New in Jinx 2.5.0
The Rtd.detach() method can be used to remove a cached Rtd instance. This allows the RTD function to be restarted without having the change the inputs or re-enter the formula.
This is especially useful when using RTD functions for short non-blocking operations, rather than streaming or ticking data.
For example, an RTD function can be used to perform a non-blocking request to an external server in the background and update the result in Excel once the request has completed. Once the request is complete and the Rtd.notify(T) method has been called, calling Rtd.detach() will remove the Rtd instance from Jinx’s cache and next time the function is called it will restart.
Throttle Interval
Excel throttles the rate of updates made via RTD functions. Instead of updating every time it is notified of new data, it waits for a period of time and then updates all cells with new data at once.
The default throttle time is 2,000 milliseconds (2 seconds). This means that even if you are calling notify
on
an Rtd instance more frequently you will not see the value in Excel updating more often than once every
two seconds.
The throttle interval can be changed via the Excel Object API by setting Application.RTD.ThrottleInterval
(in milliseconds). The Excel Object API is what you are using when you write VBA code, but exactly the
same API can be accessed from Java.
Setting the throttle interval is persistent across Excel sessions (meaning that if you close and restart Excel then the value you set the interval to will be remembered).
The following code shows how to set the throttle interval in Java using the jinx-com4j package, which acts as a bridge between your Jinx add-in and the Excel Object API.
For details of jinx-com4j see its github page https://exceljava.github.io/jinx-com4j.
Alternatively it can be set in the registry by modifying the following key. It is a DWORD in milliseconds.
HKEY_CURRENT_USER\Software\Microsoft\Office\10.0\Excel\Options\RTDThrottleInterval