Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[java] Merge different increment/decrement expressions #1890

Merged
merged 14 commits into from
Aug 6, 2019

Conversation

oowekyala
Copy link
Member

The current grammar has PreDecrementExpression, PreIncrementExpression, and PostfixExpression to represent mutation expressions.

  • This is asymmetric, and even more so because UnaryExpr has the same precedence as the former two
  • Most usages of those classes don't care about the different node type and try to get all of them.
    • That means, that most of the time, you'd just want to fetch increment and decrement expressions at the same time, whether they're postfix or prefix (also, whether they're ++ or --).

This PR makes two contributions:

  • It merges those three nodes into a single one, IncrementExpression. This simplifies some existing code by placing syntax (post-/prefix) to second rank, and removes the asymmetry. The operator is represented by the new IncrementOp, which adopts constants from UnaryOp (which didn't really belong there).
  • It introduces a subdivision of PrimaryExpression to represent those expressions that can be assigned ( evaluate to lvalues): ASTAssignableExpr.
    • Subclasses are FieldAccess, ArrayAccess, and VariableAccess.
    • VariableReference is renamed to VariableAccess for consistency
    • AssignableExprs have a new @AccessType attribute to check if they're read or written to.

@oowekyala oowekyala added the in:ast About the AST structure or API, the parsing step label Jun 26, 2019
@oowekyala oowekyala added this to the 7.0.0 milestone Jun 26, 2019
@adangel adangel self-assigned this Jul 31, 2019
Copy link
Member

@adangel adangel left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@oowekyala I've added a samples section here: https://github.com/pmd/pmd/wiki/Java_clean_changes#merge-different-increment-and-decrement-expressions
Could you please review? I'm unsure about how the PMD 7 AST looks like.

Only two things I'd change for now:

  • ASTIncrementExpression::getBaseExpression() -> getOperand()
  • TestExtension.kt: variableRef -> variableAccess

Let me know, if you agree and whether I should change it.

*
* </pre>
*/
public final class ASTIncrementExpression extends AbstractJavaExpr implements ASTExpression, LeftRecursiveNode {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not so sure about the name. "IncrementExpression" doesn't sound right, since the node can also be a decrement expression...
But I also don't know a better name.

I had a look at how eclipse jdt exposes this: They have a PrefixExpression and a PostfixExpression. So they combine the grammar in a different way....
Is this something we should do as well? Maybe to keep it in mind (not for this PR though).

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

IMO PrefixExpr + PostfixExpr are not so great because they share the two operators ++ and --, and I'd rather not duplicate them in separate enums.

AFAIK there's no term for "decrement or increment" expression. Maybe "XcrementExpression"? Just kidding 😆

However there is a term for something that is "either postfix or prefix": "unary". Maybe then, it would make sense to merge IncrementExpression into UnaryExpression and have an attribute for whether it's prefix or postfix instead. That way we have a single node, and also a single enum for unary operators too. Wdyt?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure if merging IncrementExpr with UnaryExpr is helpful - In the current use cases in the rules, we would then need to check for the operator, right? (At least in AvoidReassigningLoopVariablesRule)

@adangel adangel merged commit d0b3044 into pmd:java-grammar Aug 6, 2019
@oowekyala oowekyala deleted the grammar-assignable-expression branch August 6, 2019 15:11
oowekyala added a commit to oowekyala/pmd that referenced this pull request Nov 19, 2019
Among the different possible categorisations
of unary expressions, this is probably the
most logical and easiest to document.

Here's a comparison of different possible
categorisations. Note: `_++` is the postfix
increment operator, while `++_` is the prefix
one - idem for decrement. The last one is the
one implemented by this commit.

\## 6.0.x

```
  Unary = { -, + }
  UnaryNotPlusMinus { !, ~ }
  PreIncrement = { ++_ }
  PreDecrement { --_ }
  Postfix { _++, _++ }
```

* Very assymmetric, splits operators based on parsing
concerns

\## Before pmd#1890:

```
  Unary = { -, + , !, ~ }
  PreIncrement = { ++_ }
  PreDecrement { --_ }
  Postfix { _++, _++ }
```

* Minor simplification

\## pmd#1890:

```
  Unary = Prefix \ { ++_, --_ })
  Increment ( { ++ , -- } x (postfix, prefix) )
```
* Names are weird (IncrementExpr may be decrement, Unary != Increment
  even though semantically, Increment \subset Unary)
* No possibility to introduce a supertype (what would it be?)
* But easy to match all increment/decrement expressions

\## JLS (also, Eclipse):

```
  Prefix = { !, ~, -, +, ++_, --_ }
  Postfix = { _++, _-- }
```

* Both can have super interface UnaryExpr
  * This allows matching all increment/decrement expressions easily too
* Easiest to document, JLS like, AST like
* Fits well with `InfixExpr`
@adangel adangel mentioned this pull request Jan 23, 2023
55 tasks
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in:ast About the AST structure or API, the parsing step
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants