Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
research:software:reflex:documentation:cache_aspect [2007/07/30 03:05] – admin | research:software:reflex:documentation:cache_aspect [2007/08/24 21:48] (current) – rtoledo | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | ====== Cache Aspect ====== | ||
+ | In this section we will see how to add a cache to the fibonacci class. This aspect is very useful for a wide spectrum of applications, | ||
+ | |||
+ | ===== Cached Values in the Metaobject ===== | ||
+ | |||
+ | First, we will see how to use a metaobject to hold the cached values. Taking this option, we are restricted to use one metaobject for each Fibonacci instance because different instances can have different values for the same parameters (we are forgetting the fact that the cached values are the same among all Fibonacci instances. We are doing this because that is the most common scenario). | ||
+ | |||
+ | As usual, we need our hookset matching the get method: | ||
+ | <code java> | ||
+ | Hookset theHookset = new PrimitiveHookset( | ||
+ | MsgReceive.class, | ||
+ | new NameCS(" | ||
+ | new NameOS(" | ||
+ | ); | ||
+ | </ | ||
+ | And the metaobject definition: | ||
+ | <code java> | ||
+ | MODefinition theMO = new MODefinition.MOClass(CacheHandler.class.getName()); | ||
+ | </ | ||
+ | In the code above we used another way to create the metaobject definition: [[http:// | ||
+ | |||
+ | At the same time, we used the CacheHandler class, which definition is: | ||
+ | <code java> | ||
+ | public static class CacheHandler{ | ||
+ | |||
+ | Map< | ||
+ | |||
+ | public Object lookCache(int anArg, IExecutionPointClosure aClosure){ | ||
+ | if (!itsCache.containsKey(anArg)){ | ||
+ | Long theValue = (Long) aClosure.proceed(); | ||
+ | itsCache.put(anArg, | ||
+ | } | ||
+ | |||
+ | return itsCache.get(anArg); | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | This class keeps the cache of the pre-calculated values in its '' | ||
+ | |||
+ | The code of the link's definition and configuration is: | ||
+ | <code java> | ||
+ | BLink theLink = Links.get(theHookset, | ||
+ | theLink.setControl(Control.AROUND); | ||
+ | theLink.setCall(Control.AROUND, | ||
+ | new Parameter[]{ | ||
+ | new StdParameters.IndexedArgument(0), | ||
+ | Parameter.CLOSURE | ||
+ | }); | ||
+ | theLink.setScope(Scope.OBJECT); | ||
+ | </ | ||
+ | |||
+ | In the previous code we set the control attribute as AROUND, because we do not want to execute the original operation if we have already cached the result. Then, we set the call descriptor: the lookCache method should be called with two arguments, the fibonacci number and the closure object. | ||
+ | |||
+ | The source code of the link provider that registers this links is avalilable [[http:// | ||
+ | |||
+ | < | ||
+ | Windows | ||
+ | %java -classpath | ||
+ | " | ||
+ | reflex.examples.tutorial.CacheMOAspect | ||
+ | |||
+ | Linux | ||
+ | %java -classpath | ||
+ | " | ||
+ | reflex.examples.tutorial.CacheMOAspect | ||
+ | </ | ||
+ | |||
+ | The output should be the same as executing the Fibonacci class without the aspect. To test the proper working of the cache, modify the '' | ||
+ | <code java> | ||
+ | public static class CacheHandler{ | ||
+ | |||
+ | Map< | ||
+ | |||
+ | public Object lookCache(int anArg, IExecutionPointClosure aClosure){ | ||
+ | System.out.print(" | ||
+ | |||
+ | if (!itsCache.containsKey(anArg)){ | ||
+ | System.out.println(" | ||
+ | Long theValue = (Long) aClosure.proceed(); | ||
+ | itsCache.put(anArg, | ||
+ | } | ||
+ | else{ | ||
+ | System.out.println(" | ||
+ | } | ||
+ | |||
+ | itsCache.get(anArg); | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | You should see something like: | ||
+ | < | ||
+ | lookCache(5) | ||
+ | lookCache(4) | ||
+ | lookCache(3) | ||
+ | lookCache(2) | ||
+ | lookCache(1) | ||
+ | lookCache(0) | ||
+ | lookCache(1) | ||
+ | lookCache(2) | ||
+ | lookCache(3) | ||
+ | The fibonacci[5] is 5 | ||
+ | |||
+ | lookCache(6) | ||
+ | lookCache(5) | ||
+ | lookCache(4) | ||
+ | lookCache(3) | ||
+ | lookCache(2) | ||
+ | lookCache(1) | ||
+ | lookCache(0) | ||
+ | lookCache(1) | ||
+ | lookCache(2) | ||
+ | lookCache(3) | ||
+ | lookCache(4) | ||
+ | The fibonacci[6] is 8 | ||
+ | </ | ||
+ | |||
+ | As you can see, two different '' | ||
+ | |||
+ | In the following section we will see how to change this relationship cardinality (in this example we have 1:1 between Fibonacci and CacheHandler instances) to n:1 as it is a more scalable solution. | ||
+ | |||
+ | ===== Cached Values in the Fibonacci Instances ===== | ||
+ | |||
+ | In the previous section we created a configuration where there is one instance of a '' | ||
+ | |||
+ | In this section, we will use the Fibonacci instances themselves to hold their cached values and just one '' | ||
+ | |||
+ | When we have just one instance of a metaobject in the system, we say that the metaobject has '' | ||
+ | . | ||
+ | If we want to add the cache map to the Fibonacci class, we have to use a structural metaobject, which is defined here: | ||
+ | <code java> | ||
+ | public class CacheElementsSMO implements SMetaobject{ | ||
+ | |||
+ | public void handleClass(RClass aClass) throws MOPException{ | ||
+ | try{ | ||
+ | MemberFactory theMemberFactory = API.getMemberFactory(); | ||
+ | |||
+ | theMemberFactory.addField( | ||
+ | " | ||
+ | ); | ||
+ | |||
+ | theMemberFactory.addMethod( | ||
+ | " | ||
+ | ); | ||
+ | |||
+ | aClass.addInterface(CacheHolder.class.getName()); | ||
+ | } | ||
+ | catch (Exception e){ | ||
+ | //This should not happen | ||
+ | throw new RuntimeException(e); | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Here we add three cache-related elements to the Fibonacci class: | ||
+ | * A '' | ||
+ | * A '' | ||
+ | * A '' | ||
+ | |||
+ | |||
+ | <note important> | ||
+ | Notice that when we add a method or a field to a class, the only way to access it at runtime is making use of reflection: | ||
+ | <code java> | ||
+ | ClasstheFibClass = fibInstance.getClass(); | ||
+ | FieldtheField = theFibClass.getField(" | ||
+ | Map theCache = (Map) theField.get(fibInstance); | ||
+ | </ | ||
+ | |||
+ | But, the code above is error-prone because we are using the string '' | ||
+ | Then, in this example we added two more elements besides the field: an access method and an interface, so we use the following code to access the field: | ||
+ | <code > | ||
+ | Map theCache = ((CacheHolder) fibInstance).getCache(); | ||
+ | </ | ||
+ | |||
+ | It is worth to notice that making type-safe the access to introduced fields and methods is a good practice (we have discovered it ourselves) as it avoids the use of reflection and the problems that comes with it: security issues, name mistakes, performance penalties, etc. | ||
+ | </ | ||
+ | |||
+ | The code of the CacheHolder interface is: | ||
+ | <code java> | ||
+ | public interface CacheHolder{ | ||
+ | |||
+ | public Map getCache(); | ||
+ | } | ||
+ | </ | ||
+ | With this structural metaobject, we need to define a structural link that will apply to the Fibonacci class: | ||
+ | <code java> | ||
+ | SLink theSLink = Links.get( | ||
+ | new NameCS(" | ||
+ | theSMO | ||
+ | ); | ||
+ | </ | ||
+ | At the moment, we have transformed our Fibonacci class to hold the already-calculated values. Now we have to add the behavior to use these values. As in previous examples, we will define a hookset, a metaobject definition and a behavioral link. | ||
+ | |||
+ | The hookset code is: | ||
+ | <code java> | ||
+ | Hookset theHookset = new PrimitiveHookset( | ||
+ | MsgReceive.class, | ||
+ | new NameCS(" | ||
+ | new NameOS(" | ||
+ | ); | ||
+ | </ | ||
+ | The code above is the same as in the first implementation of the cache because the results of get are still the values we want to cache. | ||
+ | |||
+ | Then, the code of the metaobject definifion is: | ||
+ | <code java> | ||
+ | MODefinition theMO = new MODefinition.SharedMO(new GlobalCacheHandler()); | ||
+ | </ | ||
+ | Here we came back to '' | ||
+ | <code java> | ||
+ | public static class GlobalCacheHandler{ | ||
+ | |||
+ | public Object lookCache(Object aContext, int anArg, | ||
+ | IExecutionPointClosure aClosure){ | ||
+ | |||
+ | CacheHolder theHolder = ((CacheHolder) aContext); | ||
+ | Map< | ||
+ | |||
+ | if (!theCache.containsKey(anArg)){ | ||
+ | Long theValue = (Long) aClosure.proceed(); | ||
+ | theCache.put(anArg, | ||
+ | } | ||
+ | |||
+ | return theCache.get(anArg); | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | Here we have a slightly modified version of the lookCache method: it does not use its own itsCache field, it asks the context object (the Fibonacci instance) for it through the getCache method. Here you can see in practice what we were refering to when we said " | ||
+ | |||
+ | Finally, the definition and configuration of the behavioral link is: | ||
+ | <code java> | ||
+ | BLink theLink = Links.get(theHookset, | ||
+ | theLink.setControl(Control.AROUND); | ||
+ | theLink.setCall(Control.AROUND, | ||
+ | new Parameter[]{ | ||
+ | Parameter.CONTEXT_OBJECT, | ||
+ | new StdParameters.IndexedArgument(0), | ||
+ | Parameter.CLOSURE | ||
+ | })); | ||
+ | </ | ||
+ | There are only two changes in the link configuration between this approach and the previous one (the one that used the metaobject to hold the cached values): | ||
+ | * We pass a reference to the context object in order to, at the end, make the cache map accessible: '' | ||
+ | * There is not a theLink.setScope(Scope.OBJECT) because our metaobject is global (there is not a '' | ||
+ | |||
+ | The complete code of the link provider that add these two links (the structural one and the behavioral one) is [[http:// | ||
+ | < | ||
+ | Windows | ||
+ | %java -classpath | ||
+ | " | ||
+ | reflex.examples.tutorial.GlobalCacheMOAspect | ||
+ | |||
+ | Linux | ||
+ | %java -classpath | ||
+ | " | ||
+ | | ||
+ | </ | ||
+ | You should see the same output as run the Fibonacci application without Reflex. If you want to test the cache, modify the GlobalCacheHandler class this way: | ||
+ | <code java> | ||
+ | public static class GlobalCacheHandler{ | ||
+ | |||
+ | public Object lookCache(Object aContext, int anArg, | ||
+ | IExecutionPointClosure aClosure){ | ||
+ | |||
+ | System.out.print(" | ||
+ | |||
+ | CacheHolder theHolder = ((CacheHolder) aContext); | ||
+ | Map< | ||
+ | if (!theCache.containsKey(anArg)){ | ||
+ | System.out.println(" | ||
+ | Long theValue = (Long) aClosure.proceed(); | ||
+ | theCache.put(anArg, | ||
+ | } | ||
+ | else{ | ||
+ | System.out.println(" | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | When executed, you should see something like the following output: | ||
+ | < | ||
+ | lookCache(5) | ||
+ | lookCache(4) | ||
+ | lookCache(3) | ||
+ | lookCache(2) | ||
+ | lookCache(1) | ||
+ | lookCache(0) | ||
+ | lookCache(1) | ||
+ | lookCache(2) | ||
+ | lookCache(3) | ||
+ | The fibonacci[5] is 5 | ||
+ | |||
+ | lookCache(6) | ||
+ | lookCache(5) | ||
+ | lookCache(4) | ||
+ | lookCache(3) | ||
+ | lookCache(2) | ||
+ | lookCache(1) | ||
+ | lookCache(0) | ||
+ | lookCache(1) | ||
+ | lookCache(2) | ||
+ | lookCache(3) | ||
+ | lookCache(4) | ||
+ | The fibonacci[6] is 8 | ||
+ | </ | ||
+ | As you can see, we obtained the same results, but making use of just one metaobject to coordinate all caches. |