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] SingletonClassReturningNewInstance false positive with double assignment #932

Closed
alixwar opened this issue Feb 22, 2018 · 5 comments · Fixed by #4801
Closed

[java] SingletonClassReturningNewInstance false positive with double assignment #932

alixwar opened this issue Feb 22, 2018 · 5 comments · Fixed by #4801
Labels
a:false-positive PMD flags a piece of code that is not problematic
Milestone

Comments

@alixwar
Copy link

alixwar commented Feb 22, 2018

Affects PMD Version:
5.4.2

Rule:
SingletonClassReturningNewInstance

Description:
The rule incorrectly indicates that the code sample always returns a new instance:

getInstance method always creates a new object and hence does not comply to Singleton Design Pattern behaviour.

Code Sample demonstrating the issue:

public final class Foo {

    private static volatile Foo myFoo;

    public static Foo getInstance() {
        Foo instance = myFoo;
        if (instance == null) {
            synchronized (Foo .class) {
                instance = myFoo;
                if (instance == null) {
                    myFoo= instance = new Foo();
                }
            }
        }
        return instance;
    }

}

Running PMD through: Gradle (SonarQube)

@alixwar alixwar changed the title False positive: Singleton class returning new instance? [java] False positive: Singleton class returning new instance? Feb 22, 2018
@adangel
Copy link
Member

adangel commented Feb 23, 2018

@alixwar You're right - that's a false positive.
I would however slightly simplify the code using the pattern from The "Double-Checked Locking is Broken" Declaration, avoid the local variable:

public final class Foo {
    private static volatile Foo myFoo;

    public static Foo getInstance() {
        if (myFoo == null) {
            synchronized (Foo .class) {
                if (myFoo == null) {
                    instance = new Foo();
                }
            }
        }
        return myFoo;
    }
}

But since the singleton you are using, is static (there is only one Foo instance), there is an even simpler way:

public final class Foo {
    private Foo() {}

    private static class FooSingletonHelper {
        private static final Foo INSTANCE = new Foo();
    }

    public static Foo getInstance() {
        return FooSingletonHelper.INSTANCE;
    }
}

See also Java Singleton Design Pattern Best Practices with Examples

@adangel adangel added a:false-positive PMD flags a piece of code that is not problematic a:bug PMD crashes or fails to analyse a file. labels Feb 23, 2018
@alixwar
Copy link
Author

alixwar commented Feb 26, 2018

@adangel Thanks for the pointers! One detail: the Bill Pugh Singleton Implementation that you recommend triggers another rule violation:

private static final Foo INSTANCE = new Foo();

Instantiates through a private constructor from outside of the constructors class (pmd:AccessorClassGeneration)

@jsotuyod jsotuyod removed the a:bug PMD crashes or fails to analyse a file. label Apr 3, 2018
@liu78778
Copy link

liu78778 commented Apr 8, 2019

@adangel Thanks for the pointers! One detail: the Bill Pugh Singleton Implementation that you recommend triggers another rule violation:

private static final Foo INSTANCE = new Foo();

Instantiates through a private constructor from outside of the constructors class (pmd:AccessorClassGeneration)

I encountered the same situation, when using a static inner class singleton.
(pmd:AccessorClassGeneration)

1

@jsotuyod
Copy link
Member

jsotuyod commented Apr 8, 2019

@alixwar @liu78778 long story short: the solution is to change the visibility of the constructor from private to package-private. It's slightly more open, but is what the compiler ends up doing anyway (through a synthetic constructor).

Now, having said that, I'd strongly recommend you revise wether you actually need the AccessorClassGeneration rule. The rule (along with the companion AccessorMethodGeneration) warn when you access a private field / method / constructor from an inner class (or viceversa). This access is impossible in Java, so the syntax sugar is translated by the compiler into:

  1. creating an "accessor" method (typically named access$001 and so forth) with package-private visibility, delegating to the private method / working as a setter / getter for the private field; or simply a package-private constructor delegating to the private one.
  2. editing the caller to call the new accessor method instead of the original private one

The extra indirection introduced by this method has 2 negative effects:

  1. a slightly worst performance (minimal, and typically removed by the JVM after warmup)
  2. a greater number of methods in the class (something you should not even care about, except for some very specific situations, such as aiming for single-dexing an Android application).

I personally never adopt this rule outside of an Android project myself.

@liu78778
Copy link

liu78778 commented Apr 9, 2019

@jsotuyod Ah, I get it, thank you very much for your replay

@adangel adangel changed the title [java] False positive: Singleton class returning new instance? [java] SingletonClassReturningNewInstance false positive with double assignment Jan 12, 2024
@adangel adangel added this to the 7.0.0 milestone Jan 17, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
a:false-positive PMD flags a piece of code that is not problematic
Projects
None yet
Development

Successfully merging a pull request may close this issue.

4 participants