SE 510 Principles and Applications of Software Design Aspect Oriented Programming October 5, 2005 Jeff Webb
Aspect-Oriented Programming Gregor Kiczales, John Lamping, Anurag Mendhekar, Chris Maeda, Cristina Lopes, Jean-Marc Loingtier and John Irwin 1997 Xerox Palo Alto Research Center
Background Information The process of functional decomposition breaks a system down into smaller and smaller parts. –Units of behavior or function. Programming languages provide mechanisms that allow us to define abstractions for these small parts and then compose them into systems.
Generalized Procedure Based Languages Procedural OO Functional
“It feels comfortable to talk about what is encapsulated as a functional unit of the overall system”. ……………………….. We will be considering units of decomposition that are not functional.
The Problem There are many programming problems for which neither procedural nor object-oriented techniques are sufficient to clearly capture some of the important design decisions the program must implement. Examples –Error detection and handling –Security issues –Logging
Without AOP The implementation of those design decisions will be scattered throughout the code, resulting in “tangled code”.
The Problem AOP’s core idea is “separating the business logic in an application from the common services that support it.”
The issues that make these design decisions so hard to clearly capture in code are called ASPECTS The problem is that they CROSS-CUT the systems basic functionality.
Example LISP Image processing functions Some primitive functions are developed and some compound functions Requirements –Easy to develop and maintain –Efficient use of memory
A primitive image function “or!” Public Image or! (Image a, Image b){ Image result = new Image(); For i from 1 to width do{ For j from 1 to height do{ set-pixel (result i j) = or ( (get-pixel a i j) (get-pixel b i j) ); } Return result; }
More functions Functionality Implementation pixelwise logical operations written using loop primitive as or!, and!, not!above. shift image up, down written using loop primitives; slightly different loop structure. difference of two images Function remove! (a b) and!( a (not! b)) top edge of a region Function top-edge! (a) remove!(a (down! a)) bottom edge of a region Function bottom-edge! (a) remove!( a (up! a))
Easy to develop and maintain High level function to return horizontal edge pixels Function horizontal-edge! (a){ Return or! (top-edge!(a) bottom-edge!(a)); }
Hierarchical structure
Not memory efficient Each procedure loops over a number of images Many output images are created frequently, only to be quickly consumed by another loop Results in excessively frequent memory references and storage allocation Leads to cache misses, page faults and terrible performance
Solution to this problem Take a more global approach and hand code the entire process into one, efficient piece of code.
Memory efficient Public Image horizontal-edge! (a){ Image result, a-up, a-down = new Image(); a-up = up!(a); a-down = down!(a); For i from 1 to width do{ For j from 1 to height do{ set-pixel(result i j) = or (and( (get-pixel a i j) (not (get-pixel a-up i j))) and (get-pixel a i j) (not (get-pixel a-down i j)) ) } // j for } // i for Return result; }
Problems Tangled code Difficult to maintain Destroyed the original clear functional structure
Hierarchical structure
Data Flow Diagram and! not! and! or! not! up! a
Memory efficient Public Image horizontal-edge! (a){ Image result, a-up, a-down = new Image(); a-up = up!(a); a-down = down!(a); For i from 1 to width do{ For j from 1 to height do{ set-pixel(result i j) = or (and( (get-pixel a i j) (not (get-pixel a-up i j))) and (get-pixel a i j) (not (get-pixel a-down i j)) ) } // j for } // i for Return result; }
Crux of the problem Two properties are being implemented –1 Functionality –2 Loop fusion ( for the sake of memory ) Both properties originate from primitive filters They must compose differently as filters are composed
Cross-Cutting Functionality composes hierarchically in the traditional way Loop fusion composes by fusing the loops of those primitive filters that : 1.have the same loop structure 2.end up in a producer/consumer relationship at runtime The properties cross-cut one another This cross-cutting phenomena results in the tangled code
Problem with GP languages Only provide one composition mechanism Programmer must do the co-composition manually Leads to complexity and tangling Use two different languages, One for the components and one for the aspects. Solution
Component Can be cleanly encapsulated in a generalized procedure (i.e. object, method, procedure, API) By cleanly, we mean well-localized, and easily accessed and composed as necessary.
Aspect Can not be cleanly encapsulated in a generalized procedure. Aspects tend not to be units of the system’s functional decomposition, but rather to be properties that affect the performance or semantics of the components in systemic ways. Examples of aspects include memory access patterns and synchronization of concurrent objects.
Goal of AOP To support the programmer in cleanly separating components and aspects from each other, by providing mechanisms that make it possible to abstract and compose them to produce the overall system.
Programming with Aspects Rewrite the components so they can be read by the weaver Create an aspect program Weave the two together to produce optimal intermediate code
Component Re-write Public iterater Image pixelwise (Image a, Image b, operator MyOp){ Image result = new Image(); For i from 1 to width do{ For j from 1 to height do{ set-pixel (result i j) = MyOp ( (get-pixel a i j) (get-pixel b i j) ); } Return result; } Function or! (a a){ pixelwise( a b “or”) }
Component weaving Weaver reads the component program and builds a data graph, noting the nodes and edges and! not! and! or! not!
Aspect Language program cond { if (loop-shape(currentNode) == ’pointwise’ and loop-shape(nextNode) == ’pointwise’){ fuse (currentNode, nextNode, ’pointwise’); } // end if } // end cond Where pointwise is an iterator function that takes 2 image parameters and an operator paramater.
Aspect weaving Checks whether two nodes, that are connected by a data flow edge, both have a similar loop structure, and if so it fuses them into a single loop that also has the particular structure, and that has the appropriate merging of the inputs, loop variables and body of the two original loops.
Weaving output Produces an optimally structured C program Note that the real system required about a dozen or so aspects. Program analysis and understanding are to significant for a compiler to be able to use
Results Hand coding the basic functionality of original program required 768 loc (LISP) The hand coded, optimized version implements –Fusion optimization –Memoization of intermediate results –Compile time memory allocation –Specialized intermediate data structures loc The tangled code is extremely difficult to maintain, since small changes to the functionality require mentally untangling and then re-tangling it.
Results Base implementation and three aspect programs = 1039 loc Time efficiency was comparable to hand coded version Space efficiency was better than hand coded version.
Questions ?
References Gregor Kiczales, John Lamping, Anurag Mendhekar, Chris Maeda, Cristina Lopes, Jean-Marc Loingtier and John Irwin. Aspect Oriented Programming, Proceedings of the ECOOP, Volume #32, issue 1ess, March 1997, pages Memoization:
Additional Example Person Owns Dog Taken from :
Person Owns Dog class Person attribute numOfDogsOwned simple accessor method getNumDogsOwned() Person numOfDogsOwned GetNumDogsOwned()
Person Owns Dog /** * not a dog in sight... */ public class Person { private String lastName; private Address address;... public String getLastName() { return lastName; }... }
Person Owns Dog /** * not a person in sight... */ public aspect DogOwnership { public interface IDogOwner {}; private int IDogOwner.numDogsOwned; public int IDogOwner.getNumDogsOwned() { return numDogsOwned; }
Person Owns Dog - We want to create a dog-owning person -We could do this by creating a DogOwningPerson class -However dog-owning isn't limited to people -maybe an Organization can own dogs too?
Keep the concept of dog ownership independent of any one kind of dog owner public aspect PersonOwnsDog { declare parents : Person implements DogOwnership.IDogOwner; } - Now, Person and DogOwnership are independently reusable
Now you can call the getNumDogsOwned method on a Person Person person = new Person(); person.getNumDogsOwned();