Both main and java123 are valid identifiers, main isn't a reserved keyword so it's perfectly acceptable to use, as far as the test goes you should've gotten a point or half a point at least.
A: In Java, all identifiers must begin with a letter, an underscore, or a Unicode currency character. Any other symbol, such as a number, is not valid. Furthermore, an identifier cannot have the same spelling as one of Java's reserved words.
Only alphabetic characters, numeric digits, and the underscore character (_) are legal in an identifier. The first character of an identifier must be alphabetic or an underscore (it cannot be a numeric digit). Upper case letters are considered distinct from lower case letters; that is, identifiers are case sensitive.
You should only use alphabets and numbers while naming anything in Java. The only special characters that are allowed are the dollar sign('$') and the underscore('_'). Even if you can use numbers you cannot use them to start the name of the identifier. You cannot name your variable as 666devil.
public class J {
public static void main(String[] args)
{
String main = "The character sequence \"main\" is an identifier, not a keyword or reserved word.";
System.out.println(main);
}
}
This compiles, and when executed, emits this output:
The character sequence "main" is an identifier, not a keyword or reserved word.
The character sequence main
is an identifier, not a keyword or reserved word.
The relevant section of the JLS is 3.8:
An identifier is an unlimited-length sequence of Java letters and Java digits, the first of which must be a Java letter.
Identifier:
IdentifierChars but not a Keyword or BooleanLiteral or NullLiteral
IdentifierChars:
JavaLetter {JavaLetterOrDigit}
JavaLetter:
any Unicode character that is a "Java letter"
JavaLetterOrDigit:
any Unicode character that is a "Java letter-or-digit"
The character sequence main
fits the above description and is not in the keyword list in Section 3.9.
(The character sequence java1234
is also an identifier, for the same reasons.)
main
is a valid java identifier, and the teacher is wrong.
The relevant documentation is in the Java Language Specification, right here:
Chapter 3. "Lexical Structure", section 3.8. "Identifiers":
https://docs.oracle.com/javase/specs/jls/se10/html/jls-3.html#jls-3.8
It says:
An identifier is an unlimited-length sequence of Java letters and Java digits, the first of which must be a Java letter... An identifier cannot have the same spelling (Unicode character sequence) as a keyword (§3.9), boolean literal (§3.10.3), or the null literal (§3.10.7), or a compile-time error occurs.
Which means that you can prove that it is a valid identifier either by:
As the other answers state
main
is a valid Java identifier, as well as java1234
.
I guess the confusing comes from the fact that the main(String[])
method is often used as entry point by the JVM1. However, that doesn't mean that the token main
itself cannot be used as identifier2.
The specs say so, and the following declarations are also valid:
A field:
private int main;
A local variable:
String main = "";
A method:
void main() { ... }
A class or interface (although a class or interface name starting with lowercase is discouraged):
class main { ... }
A package:
package main;
1: As noted in the comments, the JVM specification itself does not mandate any particular method as entry point, but the widely used java
tool often uses such a method as entry point.
2: I would generally avoid creating a main method other than main(String[])
.
Addendum: I don't feel like this is the place to rant, but here is my correct humble opinion: the identifier main
is just as valid as java1234
, so both must be treated equally valid or wrong. Doing otherwise is intolerable.
This compiles fine on Java 1.8...
public class main {
public String main = "main";
public void main(String main) {
System.out.println("This object is an instance of the class " + this.getClass().getCanonicalName());
System.out.println("The value of the argument \"main\" for this call to the method \"main(String main)\" is " + main);
System.out.println("The value of the field \"main\" is " + this.main);
}
public static void main(String[] args) {
main main = new main();
main.main(main.main + main.main);
}
}
...and when executed produces the output:
This object is an instance of the class main
The value of the argument "main" for this call to the method "main(String main)" is mainmain
The value of the field "main" is main
I threw everything I could at it, and it appears to work. I'd say main is a valid identifier.
package main;
public class main {
static main main;
String Main;
main(String main) {
Main = main;
}
main(main main) {
System.out.println(main.Main);
}
main main(main main) {
return new main(main);
}
public static void main(main...Main) {
main:
for (main main : Main) {
main = (main instanceof Main) ? new main(main): main.main(main);
break main;
}
}
public static void main(String[] args) {
main = new main("main");
main.main(main, main);
main = main.new Main(main) {
main main(main main) {
return ((Main)main).main();
}
};
main.main(main);
main.main(main,main);
}
abstract class Main extends main {
Main(main main) {
super("main");
}
main main() {
main.Main = "Main";
return main;
}
}
}
How main
could not be used as an identifier while it is used as identifier to declare the "main" method ?
For such a classic idiom :
public class Foo{
public static void main(String[] args){
}
}
main
is not a keyword and it would probably never be a keyword in Java for obvious retro compatibility reasons.
About the question, is main
a good identifier ?
First : valid for a compiler doesn't mean necessarily good.
For example the java1234
option that is proposed is also a valid identifier but that should really be avoided.
main
has a very particularly and important meaning : it is used as the entry point method of classes and jars executed by the java
command line.
Using main
for a method name that doesn't fill the criteria to be used by the java
command line would be just misleading while using it as variable name or a class name could make sense.
For example defining the class representing the entry point of an application as the Main
class of the application is acceptable and so using it as variable name too such as :
public class Main {
public static void main(String args[]){
Main main = new Main();
// ...
}
}
In a general way, in Java, multiple characters or "words" are considered valid identifiers for the compiler but are strongly discouraged to be used in the client code (but generated code may do that : nested classes for example) as not readable and/or really misleading.
For example this could be valid for the compiler :
public class Object { // 1
public void foo() {
...
}
}
public class BadChosenIdentifier {
public static void main() { // 2
new BadChosenIdentifier().toString(new Object());
}
public void toString(Object java1234) { // 3, 4
String _result$ = java1234 + " -> to avoid"; // 4
System.out.println(_result$);
}
}
But we don't want :
Object
our class as this is defined in java.lang
(1). main()
if doesn't fill the criteria to be used by the java
command line (2). Object.toString()
method (3). _
, $
or any surprising/unmeaningful characters that go against the shared naming conventions (4). Is it a valid identifier? Yes.
Is it a good identifier? Not if you're using it for anything other than the method that starts at JVM launch.
Is another valid identifier listed? Yes.
Did the test instructions say to choose the best answer?
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With