| Max API
    8.2.0
    | 
The Max scheduler permits operations to be delayed until a later time.
It keeps track of time in double-precision, but the resolution of the scheduler depends on the user's environment preferences. The scheduler also works in conjunction with a low-priority queue, which permits time-consuming operations that might be initiated inside the scheduler to be executed in a way that does not disrupt timing accuracy.
Most objects interface with the scheduler via a clock (#t_clock) object. A clock is associated with a task function that will execute when the scheduler's current time reaches the clock's time. There is also a function called schedule() that can be used for one-off delayed execution of a function. It creates a clock to do its job however, so if your object is going to be using the scheduler repeatedly, it is more efficient to store references to the clocks it creates so the clocks can be reused.
The scheduler is periodically polled to see if it needs to execute clock tasks. There are numerous preferences Max users can set to determine when and how often this polling occurs. Briefly:
Similar Throttle and Interval settings exist for the low-priority queue as well.
For more information refer to the Timing documentation. While the details might be a little overwhelming on first glance, the important point is that the exact time your scheduled task will execute is subject to variability. Max permits this level of user control over the scheduler to balance all computational needs for a specific application.
There are five steps to using a clock in an external object.
If you want to cancel the execution of a clock for some reason, you can use clock_unset().
Note that if you call clock_delay() on a clock that is already set, its execution time will be changed. It won't execute twice.
A qelem ("queue element") is used to ensure that an operation occurs in the low-priority thread. The task function associated with a #t_qelem is executed when the low-priority queue is serviced, always in the main (user interface) thread. Any qelem that is "set" belongs to the low-priority queue and will be executed as soon as it serviced.
There are two principal things you want to avoid in the high priority thread: first, time-consuming or unpredictable operations such as file access, and second, anything that will block execution for any length of time – for example, showing a dialog box (including a file dialog).
The procedure for using a qelem is analogous to that for using a clock.
If you want to cancel the execution of a qelem for some reason, you can use qelem_unset().
Note that if you call qelem_set() on a qelem that is already set, it won't execute twice. This is a feature, not a bug, as it permits you to execute a low-priority task only as fast as the low-priority queue operates, not at the high-priority rate that the task might be triggered. An example would be that a number box will redraw more slowly than a counter that changes its value. This is not something you need to worry about, even if you are writing UI objects, as Max handles it internally (using a qelem).
The defer function and its variants use a qelem to ensure that a function executes at low-priority. There are three variants: defer(), defer_low(), and defer_medium(). The difference between using defer() and a qelem is that defer() is a one-shot deal – it creates a qelem, sets it, and then gets rid of it when the task function has executed. The effect of this is that if you have some rapid high-priority event that needs to trigger something to happen at low-priority, defer() will ensure that this low-priority task happens every time the high-priority event occurs (in a 1:1 ratio), whereas using a qelem will only run the task at a rate that corresponds to the service interval of the low-priority queue. If you repeatedly defer() something too rapidly, the low-priority queue will become backlogged and the responsiveness of the UI will suffer.
A typical use of defer() is if your object implements a read message to ask the user for a file. Opening the dialog in the timer thread and waiting for user input will likely crash, but even if it didn't, the scheduler would effectively stop.
To use defer(), you write a deferred task function that will execute at low priority. The function will be passed a pointer to your object, plus a symbol and atom list modeled on the prototype for an anything method. You need not pass any arguments to the deferred task if you don't need them, however.
To call the task, use defer() as shown below. The first example passes no arguments. The second passes a couple of long atoms.
Defer copies any atoms you pass to newly allocated memory, which it frees when the deferred task has executed.
defer has two variants, defer_low() and defer_medium(). Here is a comparison:
If executing at high priority, defer() puts the deferred task at the front of the low-priority queue. If not executing at highpriority, defer() calls the deferred task immediately.
At all priority levels, defer_low() puts the deferred task at the back of the low-priority queue.
defer_medium()
If executing at high priority, defer_medium() puts the deferred task at the back of the low-priority queue. If not executing at high priority, defer_medium() calls the deferred task immediately.
The schedule() function is to clocks as defer() is to qelems. Schedule creates a clock for a task function you specify and calls clock_fdelay() on it to make the task execute at a desired time. As with defer(), schedule() can copy arguments to be delivered to the task when it executes.
A schedule() variant, schedule_defer(), executes the task function at low priority after a specified delay.