-
-
Notifications
You must be signed in to change notification settings - Fork 1.5k
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] Improve ConstantsInInterface message to mention alternatives #1205
Comments
Effective Java's item 19 (22 in the 3rd edition, using the name of item that remains unchanged may be better than using the number) states:
If they are not, a constant utility class (private constructor) is preferred. It's not that both are equivalent, but rather that you should use enums if constants describe enumerations, or a constant utility class otherwise. |
I'm sorry, I don't get it. My point is, assuming we have
where do we put Apparently, the point of the quoted book passage is to put |
In that particular case, a utility class is your best option. Moreover, if only the enum is using it, I'd use a inner static utility class to avoid exporting it altogether: public enum Permission {
SMALL_JACKPOT(10*BonusConstants.REWARD),
BIG_JACKPOT(50*BonusConstants.REWARD);
private final int minimum;
[constructor and getter]
private static class BonusConstants {
public static final int REWARD = 5;
private BonusConstants() { }
}
} As for the rule in general, consider the scenario: public interface WeatherConstants {
int SUNNY = 0:
int CLOUDY = 1;
int RAINY = 2;
int THUNDERSTORM = 3;
} In this case, the constant values (0, 1, 2, 3) have no real meaning. This is a poor-man's enumeration, hence, as the book states:
Meaning, we should favor writing: public enum WeatherTypes {
SUNNY,
CLOUDY,
RAINY,
THUNDERSTORM;
} For other scenarios (as your own), the utility class is the best alternative. The book provides a number of reasons for this. The first several deal specially with the scenario where such constants are referenced "locally" by having a concrete class implement it, ie: public class WeatherForecast implements WeatherConstants {
// Do stuff with WeatherConstants
}
That means, I shouldn't care if a class uses a given constant, so declaring it by implementing an interface is breaking encapsulation. The book continues:
There is nothing in the code preventing you from doing this. Just as nothing prevents developers from doing: new WeatherConstants() { }.SUNNY; All of these constitute legal code that will compile and run under Java. So, in general, as a developer, you want to shape your APIs to avoid all such nonsense usages. The utility class fixes all of these, as so do the enum where it applies. |
Excellent explanation, thanks a lot! I'd be very happy to see that going into the failure message description (directly to through a link, maybe to the wiki where you could put this). |
@all-contributors please add @dague1 for doc |
I've put up a pull request to add @dague1! 🎉 |
Hi @dague1 , thanks for contributing. Would you mind to create your PR directly on this repo (pmd/pmd), rather than on a fork (Aliasbai/pmd)? -> AliAsbai#9 |
Hi, @adangel Sure, I will try to do that! |
Affects PMD Version: 6.5.0-SNAPSHOT
Rule: ConstantsInInterface
Description:
suggests to place constants in enums. I assume that doesn't imply to remove the static modifier and was wondering how that's possible since placing anything before the enum listing causes an identifier expected compilation error and placing them after the listing causes an illegal forward reference (compilation) error. Placing them in a private interface as suggested (as only solution) in https://stackoverflow.com/questions/23608885/how-to-define-static-constants-in-a-java-enum causes the
ConstantsInInterface
to fail again. It'd be nice to see the solution from Effective Java in the docs (it shouldn't be an issue to rephrase it and add a citation to the original).Code Sample demonstrating the issue:
I can provide one if it's really necessary.
Running PMD through: Maven
The text was updated successfully, but these errors were encountered: