From 7f3516ec479ea79960cb1bd2c892668cc9adcff6 Mon Sep 17 00:00:00 2001 From: ehilsdal Date: Fri, 10 Jan 2003 12:32:37 +0000 Subject: updated semantics appendix to 1.1 language (still incomplete) removed useless biblio and glossary made a few docbook rationalizations --- docs/progGuideDB/semantics.xml | 1361 +++++++++++++++++++++------------------- 1 file changed, 717 insertions(+), 644 deletions(-) (limited to 'docs/progGuideDB/semantics.xml') diff --git a/docs/progGuideDB/semantics.xml b/docs/progGuideDB/semantics.xml index cf77a23ed..f137f7497 100644 --- a/docs/progGuideDB/semantics.xml +++ b/docs/progGuideDB/semantics.xml @@ -4,12 +4,11 @@ AspectJ extends Java by overlaying a concept of join points onto the - existing Java semantics and by adding adds four kinds of program elements - to Java: + existing Java semantics and adding a few new program elements to Java: - Join points are well-defined points in the execution of a program. These + A join point is a well-defined point in the execution of a program. These include method and constructor calls, field accesses and others described below. @@ -17,39 +16,39 @@ A pointcut picks out join points, and exposes some of the values in the execution context of those join points. There are several primitive - pointcut designators, new named pointcuts can be defined by the + pointcut designators, and others can be named and defined by the pointcut declaration. - Advice is code that executes at each join point in a pointcut. Advice has + A piece of advice is code that executes at each join point in a pointcut. Advice has access to the values exposed by the pointcut. Advice is defined by before, after, and around declarations. - Introduction and declaration form AspectJ's static crosscutting features, - that is, is code that may change the type structure of a program, by adding - to or extending interfaces and classes with new fields, constructors, or - methods. Introductions are defined through an extension of usual method, - field, and constructor declarations, and other declarations are made with a - new declare keyword. + Inter-type declarations form AspectJ's static crosscutting features, that + is, is code that may change the type structure of a program, by adding to + or extending interfaces and classes with new fields, constructors, or + methods. Some inter-type declarations are defined through an extension of + usual method, field, and constructor declarations, and other declarations + are made with a new declare keyword. - An aspect is a crosscutting type, that encapsulates pointcuts, advice, and + An aspect is a crosscutting type that encapsulates pointcuts, advice, and static crosscutting features. By type, we mean Java's notion: a modular unit of code, with a well-defined interface, about which it is possible to do reasoning at compile time. Aspects are defined by the aspect declaration. - + Join Points - While aspects do define crosscutting types, the AspectJ system does not + While aspects define types that crosscut, the AspectJ system does not allow completely arbitrary crosscutting. Rather, aspects define types that cut across principled points in a program's execution. These principled points are called join points. @@ -61,88 +60,86 @@ join points defined by AspectJ are: - + - - Method call - + + Method call + When a method is called, not including super calls of non-static methods. - - + + - - Method execution - + + Method execution + When the body of code for an actual method executes. - - - + + - - Constructor call - - - When an object is built and a constructor is called, not including - this or super constructor calls. The object being constructed is - returned at a constructor call join point, so it may be accessed - with after returning advice. - - - - - Initializer execution - + + Constructor call + - When the non-static initializers of a class run. + When an object is built and that object's initial constructor is + called (i.e., not for "super" or "this" constructor calls). The + object being constructed is returned at a constructor call join + point, so its return type is considered to be the type of the + object, and the object itself may be accessed with after + returning advice. - - + + - - Constructor execution - + + Constructor execution + When the body of code for an actual constructor executes, after its this or super constructor call. The object being constructed is the currently executing object, and so may be accessed with the this pointcut. No value is returned from - constructor execution join points. + a constructor execution join point, so its return type is + considered to be void. - - + + - - Static initializer execution - + + Static initializer execution + When the static initializer for a class executes. + No value is returned from a static initializer execution join + point, so its return type is considered to be void. - - + + - - Object pre-initialization - + + Object pre-initialization + Before the object initialization code for a particular class runs. This encompasses the time between the start of its first called constructor and the start of its parent's constructor. Thus, the - execution of these join points encompass the join points from the - code found in this() and + execution of these join points encompass the join points of the + evaluation of the arguments of this() and super() constructor calls. + No value is returned from an object pre-initialization join + point, so its return type is considered to be void. - - + + - - Object initialization - + + Object initialization + When the object initialization code for a particular class runs. This encompasses the time between the return of its parent's @@ -150,172 +147,258 @@ includes all the dynamic initializers and constructors used to create the object. The object being constructed is the currently executing object, and so may be accessed with the - this pointcut. No value is returned from - constructor execution join points. + this pointcut. + No value is returned from a constructor execution join point, so + its return type is considered to be void. - - + + - - Field reference - + + Field reference + - When a non-final field is referenced. + When a non-constant field is referenced. + [Note that references to constant fields (static final + fields bound to a constant string object or primitive value) are not + join points, since Java requires them to be inlined.] - - + + - - Field assignment - + + Field set + When a field is assigned to. + Field set join points are considered to have one argument, + the value the field is being set to. + No value is returned from a field set join point, so + its return type is considered to be void. + [Note that the initializations of constant fields (static + final fields where the initializer is a constant string object or + primitive value) are not join points, since Java requires their + references to be inlined.] - - + + - - Handler execution - + + Handler execution + When an exception handler executes. + Handler execution join points are considered to have one argument, + the exception being handled. + No value is returned from a field set join point, so + its return type is considered to be void. - - - + + + + + Advice execution + + + When the body of code for a piece of advice executes. + + + + + - + + + Pointcuts - A pointcut is a program element that picks out join points, as well as - data from the execution context of the join points. Pointcuts are used + A pointcut is a program element that picks out join points and exposes + data from the execution context of those join points. Pointcuts are used primarily by advice. They can be composed with boolean operators to - build up other pointcuts. So a pointcut is defined by one of + build up other pointcuts. The primitive pointcuts and combinators + provided by the language are: + - call(Signature) + call(MethodPattern) - Picks out a method or constructor call join point based on the - static signature at the caller side. + Picks out each method call join point whose signature matches + MethodPattern. + - execution(Signature) + execution(MethodPattern) - Picks out a method or constructor execution join point based on - the static signature at the callee side. + Picks out each method execution join point whose signature matches + MethodPattern. - get(Signature) + get(FieldPattern) - Picks out a field get join point based on the static - signature. Note that references to constant fields (static final + Picks out each field reference join point whose signature matches + FieldPattern. + [Note that references to constant fields (static final fields bound to a constant string object or primitive value) are not - get join points, since Java requires them to be inlined. + join points, since Java requires them to be inlined.] + - set(Signature) + set(FieldPattern) - Picks out a field set join point based on the static - signature. Note that the initializations of constant fields (static + Picks out each field set join point whose signature matches + FieldPattern. + [Note that the initializations of constant fields (static final fields where the initializer is a constant string object or - primitive value) are not set join points, since Java requires their - references to be inlined. + primitive value) are not join points, since Java requires their + references to be inlined.] + + + + + + call(ConstructorPattern) + + Picks out each constructor call join point whose signature matches + ConstructorPattern. - handler(TypePattern) + execution(ConstructorPattern) - Picks out an exception handler of any of the Throwable types - of the type pattern. + Picks out each constructor execution join point whose signature matches + ConstructorPattern. - initialization(Signature) + initialization(ConstructorPattern) - Picks out an object initialization join point based on the - static signature of the starting constructor. + Picks out each object initialization join point whose signature matches + ConstructorPattern. + + + + + preinitialization(ConstructorPattern) + + Picks out each object pre-initialization join point whose signature matches + ConstructorPattern. staticinitialization(TypePattern) - Picks out a static initializer execution join point of any of the types - of the type pattern. + Picks out each static initializer execution join point whose signature matches + TypePattern. + + + + + + handler(TypePattern) + + Picks out each exception handler join point whose signature matches + TypePattern. + + + + + adviceexecution() + + Picks out all advice execution join points. + within(TypePattern) - Picks out all join points where the executing code is defined - in any of the classes of the type pattern. + Picks out each join point where the executing code is defined + in a type matched by TypePattern. + + + + + withincode(MethodPattern) + + Picks out each join point where the executing code is defined + in a method whose signature matches + MethodPattern. - withincode(Signature) + withincode(ConstructorPattern) - Picks out all join points where the executing code is defined - in the method or constructor of the appropriate signature. + Picks out each join point where the executing code is defined + in a constructor whose signature matches + ConstructorPattern. cflow(Pointcut) - Picks out all join points in the control flow of the join - points picked out by the pointcut, including pointcut's join points - themselves. + Picks out each join point in the control flow of any join point + P picked out by + Pointcut, including + P itself. + cflowbelow(Pointcut) - Picks out all join points in the control flow below the join - points picked out by the pointcut. + Picks out each join point in the control flow of any join point + P picked out by + Pointcut, but not + P itself. - this(TypePattern or Id) + this(Type or Id) - Picks out all join points where the currently executing object - (the object bound to this) is an instance of a - type of the type pattern, or of the type of the identifier. - Will not match any join points from static methods. + Picks out each join point where the currently executing object + (the object bound to this) is an instance of + Type, or of the type of + Id (which must be bound in the enclosing + advice or pointcut definition). + Will not match any join points from static contexts. - target(TypePattern or Id) + target(Type or Id) - Picks out all join points where the target object (the object - on which a call or field operation is applied to) is an instance of a - type of the type pattern, or of the type of the - identifier. Will not match any calls, gets, or sets to static - members. + Picks out each join point where the target object (the object + on which a call or field operation is applied to) is an instance of + Type, or of the type of + Id (which must be bound in the enclosing + advice or pointcut definition). + Will not match any calls, gets, or sets of static members. + - args(TypePattern or Id, ...) + args(Type or Id, ...) - Picks out all join points where the arguments are instances of + Picks out each join point where the arguments are instances of a type of the appropriate type pattern or identifier. @@ -323,7 +406,7 @@ PointcutId(TypePattern or Id, ...) - Picks out all join points that are picked out by the + Picks out each join point that is picked out by the user-defined pointcut designator named by PointcutId. @@ -332,7 +415,7 @@ if(BooleanExpression) - Picks out all join points where the boolean expression + Picks out each join point where the boolean expression evaluates to true. The boolean expression used can only access static members, variables exposed by teh enclosing pointcut or advice, and thisJoinPoint forms. In @@ -343,32 +426,34 @@ ! Pointcut - Picks out all join points that are not picked out by the - pointcut. + Picks out each join point that is not picked out by + Pointcut. Pointcut0 Pointcut1 - Picks out all join points that are picked out by both of the - pointcuts. + Picks out each join points that is picked out by both + Pointcut0 and + Pointcut1. Pointcut0 || Pointcut1 - Picks out all join points that are picked out by either of the - pointcuts. + Picks out each join point that is picked out by either + pointcuts. Pointcut0 or + Pointcut1. ( Pointcut ) - Picks out all join points that are picked out by the - parenthesized pointcut. + (parentheses) Picks out each join points picked out by + Pointcut. @@ -469,14 +554,13 @@ aspect B percflow(publicCall()) { - On the right-hand side of advice or pointcut declarations, a regular - Java identifier is allowed in certain pointcut designators in place of - a type or collection of types. - There are four primitive pointcut designators where this is allowed: + On the right-hand side of advice or pointcut declarations, in certain + pointcut designators, a Java identifier is allowed in place of a type + or collection of types. The pointcut designators that allow this are this, target, and - args. In all such - cases, using an identifier rather than a type is as if the type - selected was the type of the formal parameter, so that the pointcut + args. In all such cases, using an identifier rather + than a type does two things. First, it selects join points as based on + the type of the formal parameter. So the pointcut @@ -485,7 +569,8 @@ pointcut intArg(int i): args(i); picks out join points where an int is being passed - as an argument, but furthermore allows advice access to that argument. + as an argument. Second, though, it makes the value of that argument + available to the enclosing advice or pointcut. @@ -537,25 +622,22 @@ pointcut publicCall(Object o): call(public *.*(..)) args(o); AspectJ provides two primitive pointcut designators designed to capture method call and execution join points. - - call(Signature) - execution(Signature) - - - These two pointcuts also pick out constructor call end execution - join points. + + call(MethodPattern) + execution(MethodPattern) + Field-related pointcuts AspectJ provides two primitive pointcut designators designed to - capture field reference and assignment join points: + capture field reference and set join points: - - get(Signature) - set(Signature) - + + get(FieldPattern) + set(FieldPattern) + All set join points are treated as having one argument, the value the @@ -576,15 +658,16 @@ aspect GuardedX { Object creation-related pointcuts - AspectJ provides three primitive pointcut designators designed to + AspectJ provides primitive pointcut designators designed to capture the initializer execution join points of objects. - - call(Signature) - initialization(Signature) - execution(Signature) - + + call(ConstructorPattern) + execution(ConstructorPattern) + initialization(ConstructorPattern) + preinitialization(ConstructorPattern) + Class initialization-related pointcuts @@ -593,9 +676,9 @@ aspect GuardedX { static initializer execution join points. - - staticinitialization(TypePattern) - + + staticinitialization(TypePattern) + Exception handler execution-related pointcuts @@ -604,16 +687,16 @@ aspect GuardedX { execution of exception handlers: - - handler(TypePattern) - + + handler(TypePattern) + All handler join points are treated as having one argument, the value - of the exception being handled, so at a handler join point, that - value can be accessed with an args pointcut. So - an aspect used to put FooException objects into some normal form - before they are handled could be written as + of the exception being handled. That value can be accessed with an + args pointcut. So an aspect used to put + FooException objects into some normal form before + they are handled could be written as @@ -622,6 +705,32 @@ aspect NormalizeFooException { e.normalize(); } } + + + Advice execution-related pointcuts + + + AspectJ provides one primitive pointcut designator to capture + execution of advice + + + + adviceexecution() + + + + This can be used, for example, to filter out any join point in the + control flow of advice from a particular aspect. + + + +aspect TraceStuff { + pointcut myAdvice(): adviceexecution() within(TraceStuff); + + before(): call(* *(..)) !cflow(myAdvice) { + // do something + } +} State-based pointcuts @@ -631,41 +740,45 @@ aspect NormalizeFooException { particular type is executing, being operated on, or being passed around. AspectJ provides primitive pointcuts that capture join points at these times. These pointcuts use the dynamic types of - their objects to discriminate, or pick out, join points. They may - also be used to expose to advice the objects used for - discrimination. + their objects to pick out join points. They may also be used to + expose the objects used for discrimination. - - this(TypePattern or Id) - target(TypePattern or Id) - - + + this(Type or Id) + target(Type or Id) + - The this pointcut picks out all join points where the currently - executing object (the object bound to this) is an - instance of a particular type. The target pointcut picks out all - join points where the target object (the object on which a method is - called or a field is accessed) is an instance of a particular type. - + The this pointcut picks out each join point where + the currently executing object (the object bound to + this) is an instance of a particular type. The + target pointcut picks out each join point where + the target object (the object on which a method is called or a field + is accessed) is an instance of a particular type. Note that + target should be understood to be the object the + current join point is transfering control to. This means that the + target object is the same as the current object at a method execution + join point, for example, but may be different at a method call join + point. - - args(TypePattern or Id or "..", ...) - + + args(Type or Id or "..", ...) + - The args pointcut picks out all join points where the arguments are + The args pointcut picks out each join point where the arguments are instances of some types. Each element in the comma-separated list is - one of three things. If it is a type pattern, then the argument - in that position must be an instance of a type of the type name. If - it is an identifier, then the argument in that position must be an + one of four things. If it is a type name, then the argument in that + position must be an instance of that type. If it is an identifier, + then that identifier must be bound in the enclosing advice or + pointcut declaration, and so the argument in that position must be an instance of the type of the identifier (or of any type if the - identifier is typed to Object). If it is the special wildcard "..", - then any number of arguments will match, just like in signatures. So - the pointcut + identifier is typed to Object). If it is the "*" wildcard, then any + argument will match, and if it is the special wildcard "..", then any + number of arguments will match, just like in signature patterns. So the + pointcut @@ -681,67 +794,77 @@ args(int, .., String) Control flow-based pointcuts - Some concerns cut across the control flow of the program. The cflow - and cflowbelow primitive pointcut designators capture join points - based on control flow. + Some concerns cut across the control flow of the program. The + cflow and cflowbelow primitive + pointcut designators capture join points based on control flow. - - cflow(Pointcut) - cflowbelow(Pointcut) - + + cflow(Pointcut) + cflowbelow(Pointcut) + - The cflow pointcut picks out all join points that occur between the start and the - end of each of the pointcut's join points. + The cflow pointcut picks out all join points that + occur between entry and exit of each join point + P picked out by + Pointcut, including + P itself. Hence, it picks out the join + points in the control flow of the join points + picked out by Pointcut. - The cflowbelow pointcut picks out all join points that occur between - the start and the end of each of the pointcut's join points, but - not including the initial join point of the control flow itself. + The cflowbelow pointcut picks out all join points + that occur between entry and exit of each join point + P picked out by + Pointcut, but not including + P itself. Hence, it picks out the join + points below the control flow of the join points + picked out by Pointcut. Program text-based pointcuts While many concerns cut across the runtime structure of the program, - some must deal with the actual lexical structure. AspectJ allows - aspects to pick out join points based on where their associated code - is defined. + some must deal with the lexical structure. AspectJ allows aspects to + pick out join points based on where their associated code is defined. - - within(TypePattern) - withincode(Signature) - + + within(TypePattern) + withincode(MethodPattern) + withincode(ConstructorPattern) + - The within pointcut picks out all join points where the code - executing is defined in the declaration of one of the types in - TypePattern. This includes the class - initialization, object initialization, and method and constructor - execution join points for the type, as well as any join points - associated with the statements and expressions of the type. It also - includes any join points that are associated with code within any of - the type's nested types. + The within pointcut picks out each join point + where the code executing is defined in the declaration of one of the + types in TypePattern. This includes the + class initialization, object initialization, and method and + constructor execution join points for the type, as well as any join + points associated with the statements and expressions of the type. + It also includes any join points that are associated with code in a + type's nested types, and that type's default constructor, if there is + one. - The withincode pointcut picks out all join points where the code - executing is defined in the declaration of a particular method or - constructor. This includes the method or constructor execution join - point as well as any join points associated with the statements and - expressions of the method or constructor. It also includes any join - points that are associated with code within any of the method or - constructor's local or anonymous types. + The withincode pointcuts picks out each join point + where the code executing is defined in the declaration of a + particular method or constructor. This includes the method or + constructor execution join point as well as any join points + associated with the statements and expressions of the method or + constructor. It also includes any join points that are associated + with code in a method or constructor's local or anonymous types. Dynamic property-based pointcuts - - if(BooleanExpression) - + + if(BooleanExpression) + The if pointcut picks out join points based on a dynamic property. @@ -767,54 +890,80 @@ if(thisJoinPoint.getKind().equals("call")) join points. + Methods + - At a method call join point, the signature is composed of the type used - to access the method, the name of the method, and the the types of the called - method's formal parameters and return value (if any). + Join points associated with methods typically have method signatures, + consisting of a method name, parameter types, return type, the types of + the declared (checked) exceptions, and some type that the method could + be called on (below called the "qualifying type"). - At a method execution join point, the signature is composed of the type - defining the method, the name of the method, and the the types of the executing - method's formal parameters and return value (if any). + At a method call join point, the signature is a method signature whose + qualifying type is the static type used to access + the method. This means that the signature for the join point created + from the call ((Integer)i).toString() is different + than that for the call ((Object)i).toString(), even + if i is the same variable. - - At a constructor call join point, the signature is composed of the type - of the object to be constructed and the types of the - called constructor's formal parameters. + At a method execution join point, the signature a method signature + whose qualifying type is the declaring type of the method. + Fields + - At a constructor execution join point, the signature is composed of the - type defining the constructor and the types of the executing - constructor's formal parameters. + Join points associated with fields typically have field signatures, + consisting of a field name and a field type. A field reference join + point has such a signature, and no parameters. A field set join point + has such a signature, but has a has a single parameter whose type is + the same as the field type. + Constructors + - At an object initialization join point, the signature is composed of - the type being initialized and the types of the formal parameters of - the first constructor entered during the initialization of this type. + Join points associated with constructors typically have constructor + signatures, consisting of a parameter types, the types of the declared + (checked) exceptions, and the declaring type. - At an object pre-initialization join point, the signature is composed - of the type being initialized and the types of the formal parameters of - the first constructor entered during the initialization of this type. + At a constructor call join point, the signature is the constructor + signature of the called constructor. At a constructor execution join + point, the signature is the constructor signature of the currently + executing constructor. - At a field reference or assignment join point, the signature is - composed of the type used to access or assign to the field, the name of - the field, and the type of the field. + At object initialization and pre-initialization join points, the + signature is the constructor signature for the constructor that started + this initialization: the first constructor entered during this type's + initialization of this object. + Others + At a handler execution join point, the signature is composed of the exception type that the handler handles. + + At an advice execution join point, the signature is composed of the + aspect type, the parameter types of the advice, the return type (void + for all but around advice) and the types of the declared (checked) + exceptions. + + + + + + Matching + The withincode, call, execution, get, and @@ -1028,12 +1177,12 @@ handler(java.util.*) -target(com.xerox..*) +within(com.xerox..*) - picks out all join points where the target object is an instance of - defined in any type beginning with "com.xerox.". + picks out all join points where the code is in any type + definition of a type whose name begins with "com.xerox.". Subtype patterns @@ -1109,157 +1258,20 @@ call((Foo+ ! Foo).new(..)) - - - - Pointcuts and Join Points - - It is possible to pick out every different kind of join point with - pointcuts, but some of the less common ones require pointcut - combination. - - - Method call - - -aspect A { - after() returning: call(void foo()) { - System.err.println(thisJoinPoint.getKind()); // should be "method-call" - } -} - - - - - Method execution - - -aspect A { - after() returning: execution(void foo()) { - System.err.println(thisJoinPoint.getKind()); // should be "method-execution" - } -} - - - - - Constructor call - - -aspect A { - after() returning: call(Foo.new()) { - System.err.println(thisJoinPoint.getKind()); // should be "constructor-call" - } -} - - - - - Constructor execution<!-- [add chain up] --> - - -aspect A { - after() returning: execution(Foo.new()) { - System.err.println(thisJoinPoint.getKind()); // should be "constructor-execution" - } -} - - - - - Static initializer execution<!-- [add chain up] --> - - -aspect A { - after() returning: staticinitializer(Foo) { - System.err.println(thisJoinPoint.getKind()); // should be "static-initializar" - } -} - - - - - Object pre-initialization<!-- [add chain up] --> - - This join point will most commonly be seen as the enclosing - execution join point of a particular call, since it cannot be simply - picked out by AspectJ's primitive pointcuts. - - -aspect A { - after() returning: call(Foo) { - System.err.println(thisEnclosingJoinPointStaticPart.getKind()); // should be "pre-initialization" - } -} - - - - - Object initialization<!-- [add chain up] --> - - - -aspect A { - after() returning: initialization(Foo.new()) { - System.err.println(thisEnclosingJoinPointStaticPart.getKind()); // should be "initialization" - } -} - - - - - Field Reference <!-- [add chain up] --> - - - -aspect A { - after() returning: get(Foo.x) { - System.err.println(thisEnclosingJoinPointStaticPart.getKind()); // should be "field-get" - } -} - - - - - Field Assignment <!-- [add chain up] --> - - - -aspect A { - after() returning: set(Foo.x) { - System.err.println(thisEnclosingJoinPointStaticPart.getKind()); // should be "field-set" - } -} - - - - - - Handler Execution <!-- [add chain up] --> - - - -aspect A { - after() returning: handler(FooExn) { - System.err.println(thisEnclosingJoinPointStaticPart.getKind()); // should be "handler" - } -} - - - - - + + + Advice - - before(Formals): Pointcut { Body } - after(Formals) returning [ (Formal) ]: Pointcut { Body } - after(Formals) throwing [ (Formal) ]: Pointcut { Body } - after(Formals) : Pointcut { Body } - Type around(Formals) [ throws TypeList ] : Pointcut { Body } - + + before(Formals): Pointcut { Body } + after(Formals) returning [ (Formal) ]: Pointcut { Body } + after(Formals) throwing [ (Formal) ]: Pointcut { Body } + after(Formals) : Pointcut { Body } + Type around(Formals) [ throws TypeList ] : Pointcut { Body } + Advice defines crosscutting behavior. It is defined in terms of @@ -1470,7 +1482,7 @@ aspect A { The exceptions that each kind of join point in AspectJ may throw are: - + @@ -1511,18 +1523,24 @@ aspect A { - initializer execution, pre-initialization, and initialization + pre-initialization, and initialization any exception that is in the throws clause of all constructors of the initialized class. + + advice execution + + any exception that is in the throws clause of the advice. + + + - Advice precedence @@ -1542,40 +1560,40 @@ aspect A { If the two pieces of advice are defined in different aspects, then there are three cases: - - If aspect A is declared such that it dominates - aspect B, then all advice defined in A has precedence over all advice - defined in - B. + + If aspect A is matched earlier than aspect B in some + declare precedence form, then all advice in + concrete aspect A has precedence over all advice in concrete aspect B + when they are on the same join point. - + Otherwise, if aspect A is a subaspect of aspect B, then all advice defined in A has precedence over all advice defined in - B. So, unless otherwise specified with a - dominates keyword, advice in a subaspect - dominates advice in a superaspect. - - - + B. So, unless otherwise specified with + declare precedence, advice in a subaspect + has precedence over advice in a superaspect. + + + Otherwise, if two pieces of advice are defined in two different aspects, it is undefined which one has precedence. - + - + If the two pieces of advice are defined in the same aspect, then there are two cases: - - If either are after advice, then the one that + + If either are after advice, then the one that appears later in the aspect has precedence over the one that appears - earlier. + earlier. - Otherwise, then the one that appears earlier in the aspect + Otherwise, then the one that appears earlier in the aspect has precedence over the one that appears later. - - - + + + These rules can lead to circularity, such as @@ -1606,7 +1624,7 @@ aspect A { lower precedence from running by throwing an exception. If it returns normally, however, then the advice of the next precedence, or the computation under the join pint if there is no further advice, will run. - + Running after returning advice will run the advice of next precedence, or the computation under the join point if @@ -1655,23 +1673,29 @@ pointcut publicCall(): call(public * *(..)); thisJoinPoint is bound to a complete join point - object, while thisJoinPointStaticPart is bound to a - part of the join point object that includes less information, - but for which no memory allocation is required on each execution of the - advice. + object. + + + + + thisJoinPointStaticPart is bound to a part of the + join point object that includes less information, but for which no + memory allocation is required on each execution of the advice. It is + equivalent to thisJoinPoint.getStaticPart(). thisEnclosingJoinPointStaticPart is bound to the static part of the join point enclosing the current join point. Only the static part of this enclosing join point is available through this - mechanism. + mechanism. - Like standard Java reflection, which uses objects from the - java.lang.reflect hierarchy, join point objects have - types in a type hierarchy. The type of objects bound to + Standard Java reflection uses objects from the + java.lang.reflect hierarchy to build up its + reflective objects. Similarly, AspectJ join point objects have types + in a type hierarchy. The type of objects bound to thisJoinPoint is org.aspectj.lang.JoinPoint, while thisStaticJoinPoint is bound to objects of interface @@ -1681,45 +1705,51 @@ pointcut publicCall(): call(public * *(..)); - + Static crosscutting Advice declarations change the behavior of classes they crosscut, but do not change their static type structure. For crosscutting concerns that do operate over the static structure of type hierarchies, AspectJ provides - forms of introduction. + inter-type member declarations and other declare forms. - - Each introduction form is a member of the aspect defining it, but defines - a new member of another type. - - Member introduction + Inter-type member declarations + + + AspectJ allows the declaration of members by aspects that are + associated with other types. + - A method introduction looks like + An inter-type method declaration looks like - - Modifiers - Type TypePattern + + + [ Modifiers ] + Type OnType . - Id(Formals) - { Body } - - abstract Modifiers - Type TypePattern - . Id(Formals); - - - - - The effect of such an introduction is to make all the types in TypePattern - support the new method. Interfaces in TypePattern will support the new method - as well, even if the method is neither public nor abstract, so the + Id(Formals) + [ ThrowsClause ] + { Body } + + abstract + [ Modifiers ] + Type OnType + . Id(Formals) + [ ThrowsClause ] + ; + + + + + The effect of such a declaration is to make OnType + support the new method. Even if OnType is + an interface. Even if the method is neither public nor abstract. So the following is legal AspectJ code: @@ -1727,7 +1757,7 @@ pointcut publicCall(): call(public * *(..)); interface Iface {} aspect A { - private void Iface.m() { + private void Iface.m() { System.err.println("I'm a private method on an interface"); } void worksOnI(Iface iface) { @@ -1738,72 +1768,87 @@ aspect A { - A constructor introduction looks like + An inter-type constructor declaration looks like - - Modifiers TypePattern.new(Formals) - { Body } - + + + [ Modifiers ] + OnType . new ( + Formals ) + [ ThrowsClause ] + { Body } + - The effect of such an introduction is to make all the types in - TypePattern support the new constructor. You cannot introduce a - constructor onto an interface, so if TypePattern includes an interface - type it is an error. + The effect of such a declaration is to make + OnType support the new constructor. It is + an error for OnType to be an interface. - A field introduction looks like one of + An inter-type field declaration looks like one of - - Modifiers - Type TypePattern.Id = Expression; + + + [ Modifiers ] + Type + OnType . Id + = Expression; - Modifiers - Type TypePattern.Id; - + + [ Modifiers ] + Type + OnType . Id; + - The effect of such an introduction is to make all the types in - TypePattern support the new field. Interfaces in TypePattern will - support the new field as well, even if the field is neither public, - nor static, nor final. + The effect of such a declaration is to make + OnType support the new field. Even if + OnType is an interface. Even if the field is + neither public, nor static, nor final. Any occurrence of the identifier this in the body of - the constructor or method introduction, or in the initializer of a - field introduction, refers to the target type from the - TypePattern rather than to the aspect type. + an inter-type constructor or method declaration, or in the initializer + of an inter-type field declaration, refers to the + OnType object rather than to the aspect + type; it is an error to access this in such a + position from a static inter-type member + declaration. - Access modifiers - Members may be introduced with access modifiers public or private, or - the default package-protected (protected introduction is not - supported). + Inter-type member declarations may be public or private, or have + default (package-protected) visibility. AspectJ does not provide + protected inter-type members. The access modifier applies in relation to the aspect, not in relation - to the target type. So a member that is privately introduced is visible - only from code that is defined within the aspect introducing it. One - that is package-protectedly introduced is visible only from code that - is defined within the introducing aspect's package. + to the target type. So a private inter-type member is visible only from + code that is defined within the declaring aspect. A default-visibility + inter-type member is visible only from code that is defined within the + declaring aspect's package. - Note that privately introducing a method (which AspectJ supports) is - very different from introducing a private method (which AspectJ - previously supported). AspectJ does not allow the introduction of the - private method "void writeObject(ObjectOutputStream)" required to - implement the interface java.io.Serializable. + Note that a declaring a private inter-type method (which AspectJ + supports) is very different from inserting a private method declaration + into another class. The former allows access only from the declaring + aspect, while the latter would allow access only from the target type. + Java serialization, for example, uses the presense of a private method + void writeObject(ObjectOutputStream) for the + implementation of java.io.Serializable. A private + inter-type declaration of that method would not fulfill this + requirement, since it would be private to the aspect, not private to + the target type. @@ -1811,14 +1856,10 @@ aspect A { Conflicts - Introduction may cause conflicts among introduced members and between - introduced members and defined members. - - - - - Assuming otherPackage is not the package defining - the aspect A, the code + Inter-type declarations raise the possibility of conflicts among + locally declared members and inter-type members. For example, assuming + otherPackage is not the package containing the + aspect A, the code @@ -1832,11 +1873,11 @@ aspect A { - adds a field "r" to every type in otherPackage. This - field is only accessible from the code inside of aspect - A. The aspect also adds a - "register" method to every type in - otherPackage. This method is accessible + declares that every type in otherPackage has a field + r. This field, however, is only accessible from the + code inside of aspect A. The aspect also declares + that every type in otherPackage has a method + "register", but makes this method accessible from everywhere. @@ -1844,7 +1885,7 @@ aspect A { If any type in otherPackage already defines a private or package-protected field "r", there is no conflict: The aspect cannot see such a field, and no code in - otherPackage can see the introduced + otherPackage can see the inter-type "r". @@ -1858,9 +1899,9 @@ this.r = r - is an error, since it is ambiguous whether the introduced - "r" or the public "r" should be - used. + is an error, since it is ambiguous whether the private inter-type + "r" or the public locally-defined + "r" should be used. @@ -1875,24 +1916,25 @@ this.r = r resolution rules: - - A subclass can inherit multiple fields from its superclasses, + + A subclass can inherit multiple fields from its superclasses, all with the same name and type. However, it is an error to have an ambiguous - reference to a field. + reference to a field. - A subclass can only inherit multiple + A subclass can only inherit multiple methods with the same name and argument types from its superclasses if only zero or one of them is concrete (i.e., all but one is abstract, or all are abstract). - - + + - Given a potential conflict between inter-type member declarations in - different aspects, if one aspect dominates the other its declaration will - take effect without any conflict notice from compiler. This is true both - when the domination is declared explicitly in a "dominates" clause and - when sub-aspects implicitly dominate their corresponding super-aspect. + Given a potential conflict between inter-type member declarations in + different aspects, if one aspect has precedence over the other its + declaration will take effect without any conflict notice from compiler. + This is true both when the precedence is declared explicitly with + declare precedence as well as when when sub-aspects + implicitly have precedence over their super-aspect. @@ -1901,22 +1943,23 @@ this.r = r Extension and Implementation - An aspect may introduce a superclass or superinterface onto a type, - with the declarations + An aspect may change the inheritance hierarchy of a system by changing + the a superclass of a type or adding a superinterface onto a type, with + the declare parents form. - - declare parents: TypePattern extends TypeList; - declare parents: TypePattern implements TypeList; - + + declare parents: TypePattern extends TypeList; + declare parents: TypePattern implements TypeList; + For example, if an aspect wished to make a particular class runnable, - it might add an appropriate void run() method, but - it should also change the type of the class to specify that it fulfills - the Runnable interface. In order to implement the - methods in the Runnable interface, the - run() method must be publically introduced: + it might define appropriate inter-type void + run() method, but it should also declare that the class + fulfills the Runnable interface. In order to + implement the methods in the Runnable interface, the + inter-type run() method must be public: @@ -1932,18 +1975,20 @@ aspect A { Interfaces with members - Through the use of introduction, interfaces may now carry + Through the use of inter-type members, interfaces may now carry (non-public-static-final) fields and (non-public-abstract) methods that classes can inherit. Conflicts may occur from ambiguously inheriting members from a superclass and multiple superinterfaces. - Because interfaces may carry non-static initializers, the order of - super-interface instantiation is observable. We fix this order with the - following three properties: A supertype is initialized before a - subtype, that initialized code runs only once, and initializers for - supertypes run in left-to-right order. Consider the following hierarchy + Because interfaces may carry non-static initializers, each interface + behaves as if it has a zero-argument constructor containing its + initializers. The order of super-interface instantiation is + observable. We fix this order with the following properties: A + supertype is initialized before a subtype, initialized code runs only + once, and the initializers for a type's superclass are run before the + initializers for its superinterfaces. Consider the following hierarchy where {Object, C, D, E} are classes, {M, N, O, @@ -1970,16 +2015,18 @@ aspect A { + + Warnings and Errors An aspect may specify that a particular join point should never be reached. - - declare error: Pointcut: String; - declare warning: Pointcut: String; - + + declare error: Pointcut: String; + declare warning: Pointcut: String; + If the compiler determines that a join point in Pointcut could possibly be reached, then it @@ -1998,9 +2045,9 @@ aspect A { RuntimeException and thus does not need to be declared. - - declare soft: TypePattern: Pointcut; - + + declare soft: Type: Pointcut; + For example, the aspect @@ -2033,6 +2080,68 @@ aspect A { + + Advice Precedence + + + An aspect may declare a precedence relationship between concrete + aspects with the declare precedence form: + + + + declare precedence : TypePatternList + + + This signifies that if any join point has advice from two concrete + aspects matched by some pattern in + TypePatternList, then the precence of the + advice will be the order of in the list. + + In TypePatternList, the wildcard "*" can + appear at most once, and it means "any type not matched by any other + pattern in the list". + + For example, the constraints that (1) aspects that have Security as + part of their name should have precedence over all other aspects, and (2) + the Logging aspect (and any aspect that extends it) should have + precedence over all non-security aspects, can be expressed by: + + + declare precedence: *..*Security*, Logging+, *; + + + + For another example, the CountEntry aspect might want to count the + entry to methods in the current package accepting a Type object as its + first argument. However, it should count all entries, even those that + the aspect DisallowNulls causes to throw exceptions. This can be + accomplished by stating that CountEntry has precedence over + DisallowNulls. This declaration could be in either aspect, or in + another, ordering aspect: + + + + +aspect Ordering { + declare precedence: CountEntry, DisallowNulls; +} +aspect DisallowNulls { + pointcut allTypeMethods(Type obj): call(* *(..)) args(obj, ..); + before(Type obj): allTypeMethods(obj) { + if (obj == null) throw new RuntimeException(); + } +} +aspect CountEntry { + pointcut allTypeMethods(Type obj): call(* *(..)) args(obj, ..); + static int count = 0; + before(): allTypeMethods(Type) { + count++; + } +} + + + + Statically determinable pointcuts @@ -2045,27 +2154,28 @@ aspect A { discriminate based on dynamic (runtime) context. Therefore, such pointcuts may not be defined in terms of - - cflow - cflowbelow - this - target - args - if - + + cflow + cflowbelow + this + target + args + if + all of which can discriminate on runtime information. - + Aspects - An aspect is a crosscutting type defined by the aspect declaration. The - aspect declaration is similar to the class declaration in that it defines - a type and an implementation for that type. It differs in that the type - and implementation can cut across other types (including those defined by + An aspect is a crosscutting type defined by the aspect + declaration. The aspect declaration is similar to the + class declaration in that it defines a type and an + implementation for that type. It differs in that the type and + implementation can cut across other types (including those defined by other aspect declarations), and that it may not be directly instantiated with a new expression, with cloning, or with serialization. Aspects may have one constructor definition, but if so it must be of a constructor @@ -2074,7 +2184,7 @@ aspect A { Aspects may be defined either at the package level, or as a static nested - aspect, that is, a static member of a class, interface, or aspect. If it + aspect -- that is, a static member of a class, interface, or aspect. If it is not at the package level, the aspect must be defined with the static keyword. Local and anonymous aspects are not allowed. @@ -2143,16 +2253,17 @@ aspect A { Singleton Aspects - - aspect Id { ... } - aspect Id issingleton { ... } - + + aspect Id { ... } + aspect Id issingleton { ... } + - By default, or by using the modifier issingleton, an - aspect has exactly one instance that cuts across the entire program. - That instance is available at any time during program execution with - the static method aspectOf() defined on the aspect + By default (or by using the modifier issingleton) + an aspect has exactly one instance that cuts across the entire + program. That instance is available at any time during program + execution with the static method aspectOf() + defined on the aspect -- so, in the above examples, A.aspectOf() will return A's instance. This aspect instance is created as the aspect's classfile is loaded. @@ -2163,15 +2274,21 @@ aspect A { the running of a program (once its class is loaded), its advice will have a chance to run at all such join points. + + + (In actuality, one instance of the aspect A is made for each version + of the aspect A, so there will be one instantiation for each time A + is loaded by a different classloader.) + Per-object aspects - - aspect Id perthis(Pointcut) { ... } - aspect Id pertarget(Pointcut) { ... } - + + aspect Id perthis(Pointcut) { ... } + aspect Id pertarget(Pointcut) { ... } + If an aspect A is defined @@ -2206,16 +2323,15 @@ aspect A { Both perthis and pertarget aspects may be affected by code the AspectJ compiler controls, as discussed in the appendix. - Per-control-flow aspects - - aspect Id percflow(Pointcut) { ... } - aspect Id percflowbelow(Pointcut) { ... } - + + aspect Id percflow(Pointcut) { ... } + aspect Id percflowbelow(Pointcut) { ... } + If an aspect A is defined @@ -2237,9 +2353,9 @@ aspect A { Aspect privilege - - privileged aspect Id { ... } - + + privileged aspect Id { ... } + Code written in aspects is subject to the same access control rules as @@ -2251,10 +2367,10 @@ aspect A { While these restrictions are suitable for many aspects, there may be - some aspects in which advice or introductions needs to access private + some aspects in which advice or inter-type members needs to access private or protected resources of other types. To allow this, aspects may be declared privileged. Code in priviliged aspects has - access to all members, even private ones. + access to all members, even private ones. @@ -2273,7 +2389,7 @@ privileged aspect A { In this case, if A had not been declared privileged, the field reference - c.i would have resulted in an error signalled by the compiler. + c.i would have resulted in an error signaled by the compiler. @@ -2296,57 +2412,14 @@ privileged aspect A { - A's introduced private field C.i, initially bound to 999, will be + A's private inter-type field C.i, initially bound to 999, will be referenced in the body of the advice in preference to C's privately - declared field, since the A would have access to fields it introduces - even if it were not privileged. + declared field, since the A would have access to its own inter-type + fields even if it were not privileged. - - - Aspect domination - - - aspect Id dominates TypePattern { ... } - - - - An aspect may declare that the advice in it dominates the advice in - some other aspect. Such declarations are like the - strictfp keyword in Java; it applies to the advice - declarations inside of the respective aspects, and states that the - advice declared in the current aspect has more precedence than the - advice in the aspects from TypePattern. - - - - For example, the CountEntry aspect might want to count the entry to - methods in the current package accepting a Type object as its first - argument. However, it should count all entries, even those that the - aspect DisallowNulls causes to throw exceptions. This can be - accomplished by stating that CountEntry dominates DisallowNulls. - - - - -aspect DisallowNulls { - pointcut allTypeMethods(Type obj): call(* *(..)) args(obj, ..); - before(Type obj): allTypeMethods(obj) { - if (obj == null) throw new RuntimeException(); - } -} -aspect CountEntry dominates DisallowNulls { - pointcut allTypeMethods(Type obj): call(* *(..)) args(obj, ..); - static int count = 0; - before(): allTypeMethods(Type) { - count++; - } -} - - - -- cgit v1.2.3