Transactions
GigaSpace
with the different GigaSpaces transaction managers and Spring allow simple declarative definition of transactions. This boils down to the fact that if there is an ongoing transaction running, any operation performed using the GigaSpace
interface joins it, using Spring's rich transaction support.
In order to have GigaSpaces transactional, the transaction manager must be provided as a reference when constructing the GigaSpace bean. For example (using the distributed transaction manager):
<os-core:embedded-space id="space" space-name="space" />
<os-core:distributed-tx-manager id="transactionManager"/>
<os-core:giga-space id="gigaSpace" space="space" tx-manager="transactionManager"/>
<bean id="space" class="org.openspaces.core.space.EmbeddedSpaceFactoryBean">
<property name="name" value="space" />
</bean>
<bean id="transactionManager" class="org.openspaces.core.transaction.manager.DistributedJiniTransactionManager">
<property name="space" ref="space" />
</bean>
<bean id="gigaSpace" class="org.openspaces.core.GigaSpaceFactoryBean">
<property name="space" ref="space" />
<property name="transactionManager" ref="transactionManager" />
</bean>
It is highly recommended to read the transaction management chapter in the Spring reference documentation.
Transaction Provider
OpenSpaces provides a pluggable transaction provider using the following interface:
public interface TransactionProvider {
Transaction getCurrentTransaction(Object transactionalContext, IJSpace space);
int getCurrentTransactionIsolationLevel(Object transactionalContext);
}
OpenSpaces comes with a default transaction provider implementation, which uses Spring and its transaction manager in order to obtain the currently running transactions and automatically use them under transactional operations.
GigaSpaces allows access to the current running transaction using the transaction provider. The following code example shows how the take operation can be performed using IJspace
(users normally won't be required to do so):
gigaSpace.getSpace().take(obj, gigaSpace.getCurrentTransaction(), 1000);
Isolation Level
GigaSpaces supports three isolation levels: READ_UNCOMMITTED
, READ_COMMITTED
and REPEATABLE_READ
(default). When using GigaSpace
, the default isolation level that it will perform under can be defined in the following manner:
<os-core:embedded-space id="space" space-name="space" />
<os-core:giga-space id="gigaSpace" space="space" default-isolation="READ_COMMITTED"/>
<bean id="space" class="org.openspaces.core.space.EmbeddedSpaceFactoryBean">
<property name="name" value="space" />
</bean>
<bean id="gigaSpace" class="org.openspaces.core.GigaSpaceFactoryBean">
<property name="space" ref="space" />
<property name="defaultIsolationLevelName" value="READ_COMMITTED" />
</bean>
IJSpace space = // get Space Where GigaSpaces data is stored. It is the logical cache that holds data objects in memory and might also hold them in layered in tiering. Data is hosted from multiple SoRs, consolidated as a unified data model. either by injection or code creation
GigaSpace gigaSpace = new GigaSpaceConfigurer(space)
.defaultIsolationLevel(TransactionDefinition.ISOLATION_READ_COMMITTED)
.gigaSpace();
In addition, Spring allows you to define the isolation level on the transaction definition itself:
@Transactional(readOnly = true)
public class DefaultFooService implements FooService {
private GigaSpace gigaSpace;
public void setGigaSpace(GigaSpace gigaSpace) {
this.gigaSpace = gigaSpace;
}
public Foo getFoo(String fooName) {
// do something
}
// these settings have precedence for this method
@Transactional(readOnly = false,
propagation = Propagation.REQUIRES_NEW,
isolation = Isolation.READ_COMMITTED)
public void updateFoo(Foo foo) {
// do something
}
}
In the above example, any operation performed using GigaSpace
in the updateFoo
method automatically works under the READ_COMMITTED
isolation level.