-
-
Notifications
You must be signed in to change notification settings - Fork 751
Understanding Broadcaster
A Broadcaster implements the publish/subscribe paradigm. An application can subscribe to one or many Broadcaster to get notified about events. By default, a single Broadcaster is created by the framework and associated with every new AtmosphereResource. The default Broadcaster's id is always "/*", which means that if you invoke
Future<Object> f = broadcaster.broadcast("hello");
all AtmosphereResource will gets notified and will have a chance to handle the event. By default, broadcaster.broadcast("hello") is an asynchronous operation and won't block. If you want to get notified when the asynchronous execution complete, you can use the returned Future
Future<Object> f = broadcaster.broadcast("hello");
f.get(); // blocking until broadcast completed.
You can associate a Broadcaster with an AtmosphereResource by doing
atmosphereResource.setBroadcaster(broadcaster);
You can associate an AtmosphereResource to several Broadcasters by doing
broadcaster.addAtmosphereResource(atmosphereResource);
As an example, implementing a Twitter like application using the Atmosphere Framework would consist of creating one Broadcaster per Twitter account:
Broadcaster b = BroadcasterFactory.getDefault().get("jfarcand");
Broadcaster b2 = BroadcasterFactory.getDefault().get("atmo_framework");
If user 'jfarcand" wants to be notified when 'atmo_framework' publish a tweet, you just associate the AtmosphereResource representing 'jfarcand' to the 'atmo_framework' Broadcaster:
b2.addAtmosphereResource(atmosphereResource);
You can create channel (or Broadcaster) on the fly, using the BroadcasterFactory class.
The MetaBroadcaster utility class can be really helpful when you want to broadcast messages to all or a subset of existing Broadcaster. For example, executing
MetaBroadcaster.getDefault().broadcast("/", "hello world");
will broadcast the "hello world" message to all AtmosphereResource associated to all Broadcaster. To be more specific, assume you create
BroadcasterFactory.getDefault().get("/a");
BroadcasterFactory.getDefault().get("/a/a1");
BroadcasterFactory.getDefault().get("/a/a2");
BroadcasterFactory.getDefault().get("/b");
BroadcasterFactory.getDefault().get("/c");
Doing
MetaBroadcaster.getDefault().broadcast("/a/*", "hello world");
is equivalent of doing:
// Retrieve the Broadcaster named "/a/a1"
BroadcasterFactory.getDefault().lookup("/a/a1", true).broadcast("hello world");
// Retrieve the Broadcaster named "/a/a2"
BroadcasterFactory.getDefault().lookup("/a/a2", true).broadcast("hello world");
A unique instance of BroadcasterConfig is always associated with a Broadcaster. You can use a BroadcasterConfig to set ExecutorServices, add BroadcastFilter and set a BroadcasterCache.
Configuring the default Broadcaster
By default, Atmosphere is using the DefaultBroadcaster and JerseyBroadcaster if atmosphere-jersey is used. You can either extends those Broadcaster or write your own. You can configure it by doing: In web.xml
<init-param>
<param-name>org.atmosphere.cpr.broadcasterClass</param-name>
<param-value>org....</param-value>
</init-param>
or in atmosphere.xml
<applicationConfig>
<param-name>org.atmosphere.cpr.broadcasterClass</param-name>
<param-value>org...</param-value>
</applicationConfig>
You can also annotate your Broadcaster implementation by using the BroadcasterService annotation:
@BroadcasterService
public class MyBroadcaster implements Broadcaster {...}
Atmosphere is able to auto discover the following Broadcaster when available on the classpath
That means you don't have to specify them by default.
By default, a Broadcaster always creates three ExecutorServices: one for supporting asynchronous broadcast, one for supporting asynchronous write and one for scheduling tasks. If you don't need asynchronous I/O, it is recommended you use the SimpleBroadcaster or SimpleJerseyBroadcaster. Both Broadcaster aren't using any ExecutorServices.
If your application creates a lot of Broadcaster, you may experiment some Out Of Memory error because too many instance of ExecutorServices has been created, e.g number of broadcaster * 3. If that's the case, you can configure Atmosphere to share ExecutorServices amongst Broadcaster. In that case only two ExecutorServices will be created:
Broadcaster b = BroadcasterFactory.getDefault().get();
b.getBroacasterConfig().setExecutorServices(...).setAsyncWriteService(...);
In web.xml
<init-param>
<param-name>org.atmosphere.cpr.broadcaster.shareableThreadPool</param-name>
<param-value>true</param-value>
</init-param>
or in atmosphere.xml
<applicationConfig>
<param-name>org.atmosphere.cpr.broadcaster.shareableThreadPool</param-name>
<param-value>true</param-value>
</applicationConfig>
Instead of creating several Broadcaster, you can always reduce the number by instead adding BroadcasterFilter and PerRequestBroadcasterFilter to a Broadcaster. You can then filter which broadcasted messages get delivered to which AtmosphereResource (client).
Broadcaster b = BroadcasterFactory.getDefault().get();
b.getBroacasterConfig().addFilter(...);
Another way to prevent or reduce memory usage is by configuring a BroadcasterFactoryLifecyclePolicy. Supported policy are:
- IDLE: Release all resources associated with the Broadcaster when the idle time expires. Suspended AtmosphereResource will NOT get resumed and instead be closed right away.
- IDLE_DESTROY: Release all resources associated with the Broadcaster when the idle time expires and destroy the Broadcaster. This operation remove the Broadcaster from it's associated BroadcasterFactory. Suspended AtmosphereResource will NOT get resumed and instead be closed right away.
- IDLE_RESUME:Release all resources associated with the Broadcaster when the idle time expires. All associated AtmosphereResource WILL BE resumed and this broadcaster destroyed.
- EMPTY: If there is no AtmosphereResource associated with the Broadcaster release all resources.
- EMPTY_DESTROY: If there is no AtmosphereResource associated with the Broadcaster, release all resources and destroy the broadcaster. This operation remove the Broadcaster from it's associated BroadcasterFactory
- NEVER: Never release or destroy the Broadcaster from it's associated BroadcasterFactory
The default is NEVER, which means that a fair amount of Broadcaster may "polute' the BroadcasterFactory if not handled properly. BroadcasterFactoryLifecyclePolicy can be configured
You can configure the policy on a Broadcaster directly:
Broadcaster b = BroadcasterFactory.getDefault().get();
b.setBroadcasterLifeCyclePolicy(BroadcasterLifeCyclePolicy.IDLE);
You can also associate BroadcasterLifeCyclePolicyListener to a Broadcaster so you get notified when a policy is exectuted.
b.addBroadcasterLifeCyclePolicyListener(new BroadcasterLifeCyclePolicyListener() {...});
In web.xml
<init-param>
<param-name>org.atmosphere.cpr.broadcasterLifeCyclePolicy</param-name>
<param-value>IDLE</param-value>
</init-param>
or in atmosphere.xml
<applicationConfig>
<param-name>org.atmosphere.cpr.broadcasterLifeCyclePolicy</param-name>
<param-value>IDLE</param-value>
</applicationConfig>
- Understanding Atmosphere
- Understanding @ManagedService
- Using javax.inject.Inject and javax.inject.PostConstruct annotation
- Understanding Atmosphere's Annotation
- Understanding AtmosphereResource
- Understanding AtmosphereHandler
- Understanding WebSocketHandler
- Understanding Broadcaster
- Understanding BroadcasterCache
- Understanding Meteor
- Understanding BroadcastFilter
- Understanding Atmosphere's Events Listeners
- Understanding AtmosphereInterceptor
- Configuring Atmosphere for Performance
- Understanding JavaScript functions
- Understanding AtmosphereResourceSession
- Improving Performance by using the PoolableBroadcasterFactory
- Using Atmosphere Jersey API
- Using Meteor API
- Using AtmosphereHandler API
- Using Socket.IO
- Using GWT
- Writing HTML5 Server-Sent Events
- Using STOMP protocol
- Streaming WebSocket messages
- Configuring Atmosphere's Classes Creation and Injection
- Using AtmosphereInterceptor to customize Atmosphere Framework
- Writing WebSocket sub protocol
- Configuring Atmosphere for the Cloud
- Injecting Atmosphere's Components in Jersey
- Sharing connection between Browser's windows and tabs
- Understanding AtmosphereResourceSession
- Manage installed services
- Server Side: javadoc API
- Server Side: atmosphere.xml and web.xml configuration
- Client Side: atmosphere.js API