知优码

您现在的位置是:首页 > 开源项目 > 项目经验

项目经验

java“找不到符号”或“无法解析符号”错误

2020-12-19项目经验
“找不到符号”错误是什么意思?首先,它是编译错误。这意味着,在Java源代码中有问题,或你编译它的方式有问题。


一、概述


1.“找不到符号”错误是什么意思?
首先,它是编译错误。这意味着,在Java源代码中有问题,或你编译它的方式有问题。
 

二、详解


(1)出现“找不到符号”错误,可能是您的Java源代码包含以下内容:

1. 关键词:如true,false,class,while,等。

2. 文字:如 42和'X'或者 "Hi mum!"。

3. 操作符和其他非字母数字标记:像+,=,{,等等。

4. 标识符:如 Reader,i,toString,processEquibalancedElephants,等。

5. 注释和空格。

“找不到符号”错误与标识符有关。编译代码时,编译器需要确定代码中每个标识符的含义。

“找不到符号”错误表示编译器无法执行此操作。您的代码包含了编译器无法理解的内容。

(2)什么会导致“找不到符号”错误?

首先,只有一个原因。编译器查看了应该定义标识符的所有位置,但找不到该定义。这可能是由多种原因引起的。常见的如下:

对于一般的标识符:

1. 也许您拼错了名字;即 StringBiulder 代替 StringBuilder。Java无法也不会尝试弥补拼写错误或输入错误。

2. 大小写错误。即stringBuilder代替StringBuilder。所有Java标识符均区分大小写。

3. 也许您不恰当地使用了下划线;即mystring和my_string不同。

4. 对于应该引用变量的标识符:

也许您忘记了声明变量。

变量声明在超出作用域范围。(请参见下面的示例)

5. 对于方法或字段名称的标识符:

也许您正在尝试引用在父/祖先类或接口中未声明的继承方法或字段。

也许您正在尝试引用所使用类型中不存在(即尚未声明)的方法或字段;例如"someString".push()2。

也许您正在尝试将方法用作字段,反之亦然;例如"someString".length或someArray.length()。

也许您是错误地对数组而不是数组元素进行操作;例如
 
    String strings[] = ...
    if (strings.charAt(3)) { ... }
    // maybe that should be 'strings[0].charAt(3)'

6.对于类名的标识符:

也许您忘记了导入类。

也许您使用了“星号”导入,但是在您导入的任何软件包中都没有定义该类。

也许您忘记了new:

 
   String s = String();  // should be 'new String()'


7. 对于类型或实例似乎没有成员的情况,您希望它具有:

也许您已经声明了一个嵌套类或泛型参数,该类掩盖了您打算使用的类型。

也许您正在隐藏静态变量或实例变量。

也许您输入了错误的类型;例如由于IDE完成或自动更正。

也许您使用(针对)错误版本的API。

也许您忘记了将对象转换为适当的子类。

问题通常是上述情况的组合。例如,也许您“星标”导入java.io.*,然后尝试使用Filesclass中的java.nionot java.io。或者也许你打算写File...这是的一个课java.io。

这是一个错误的变量作用域如何导致“找不到符号”错误的示例:
 
List<String> strings = ...

for (int i = 0; i < strings.size(); i++) {
    if (strings.get(i).equalsIgnoreCase("fnord")) {
        break;
    }
}
if (i < strings.size()) {
    ...
}

这将使“无法找到符号”错误i的if陈述。虽然我们先前声明i,该声明只是在范围上的for发言和它的身体。语句i中对的引用看不到的声明。它超出范围。ifi

(这里的一个适当的更正可能是将if语句移入循环内部,或者i在循环开始之前进行声明。)

这是一个引起困惑的示例,其中的错字导致看似莫名其妙的“找不到符号”错误:
 
for (int i = 0; i < 100; i++); {
    System.out.println("i is " + i);
}


这将在println呼叫中给您一个编译错误,提示i找不到。但是(我听到你说)我确实声明了!

问题是{ 前的;。Java语言语法在该上下文中将分号定义为空语句。空语句然后成为for循环的主体。因此该代码实际上意味着:
 
for (int i = 0; i < 100; i++); 

// The previous and following are separate statements!!

{
    System.out.println("i is " + i);
}

该{ ... }块不是for循环的主体,因此i该for语句中的上一个声明超出了该块的范围。

这是由错字引起的“找不到符号”错误的另一个示例。
 
int tmp = ...
int res = tmp(a + b);


尽管与先前的声明中,tmp在tmp(...)表达是错误的。编译器将寻找一种称为tmp的方法,但找不到。先前声明tmp的变量位于名称空间中,而不是方法的名称空间中。

在我遇到的示例中,程序员实际上省略了一个运算符。他的意思是这样写的:
 
int res = tmp * (a + b);


如果从命令行进行编译,则编译器可能找不到符号还有另一个原因。您可能只是忘了编译或重新编译其他类。例如,如果您有类,Foo并且Bar在哪里Foo使用Bar。如果您从未编译过Bar并且运行过javac Foo.java,您很有可能发现编译器找不到该符号Bar。简单的答案是编译Foo和Bar在一起; 例如javac Foo.java Bar.java或javac *.java。或者最好还是使用Java构建工具;例如Ant,Maven,Gradle等。
 

三、如何解决这些错误?


一般来说,首先要弄清楚是什么导致了编译错误。

查看编译错误消息指示的文件中的行。

确定错误消息正在谈论的符号。

弄清楚编译器为什么说找不到符号;为什么?往上看!

然后,您考虑一下您的代码应该说什么。最后,您确定需要对源代码进行哪些更正以执行所需的操作。

请注意,并非每个“更正”都是正确的。考虑一下:
 
for (int i = 1; i < 10; i++) {
    for (j = 1; j < 10; j++) {
        ...
    }
}


假设编译器为表示“找不到符号” j。我可以通过多种方式“修复”该问题:

我可以将内部更改for为for (int j = 1; j < 10; j++)-可能是正确的。

我可以在内部循环或外部循环j 之前添加一个声明-可能正确。forfor

我可以更改j为i内for循环-可能是错误的!

等等。

关键是您需要了解您的代码正在尝试做什么才能找到正确的解决方案。

4.不明原因
在某些情况下,“找不到符号”似乎无法解释……直到您仔细观察。

不正确的依赖关系:如果您使用的是IDE或管理构建路径和项目依赖关系的构建工具,则可能是错误的依赖关系;例如,遗漏依赖项,或选择了错误的版本。如果您使用的是构建工具(Ant,Maven,Gradle等),请检查项目的构建文件。如果使用的是IDE,请检查项目的构建路径配置。

您无需重新编译:有时候,新Java程序员可能不了解Java工具链的工作方式,或者没有实现可重复的“构建过程”;例如,使用IDE,Ant,Maven,Gradle等。在这种情况下,程序员最终可能会追尾寻找一个虚幻的错误,该错误实际上是由于未正确地重新编译代码而导致的,等等。

较早的构建问题:较早的构建可能会失败,从而导致JAR文件缺少类。如果使用构建工具,通常会注意到这种故障。但是,如果要从其他人那里获取JAR文件,则取决于它们能否正确构建并注意到错误。如果您怀疑这一点,请使用tar -tvf列出可疑JAR文件的内容。

IDE问题:人们报告说,他们的IDE感到困惑,并且IDE中的编译器找不到存在的类……或相反的情况。

如果IDE配置了错误的JDK版本,则可能会发生这种情况。

如果IDE的缓存与文件系统不同步,则可能发生这种情况。有特定于IDE的方法可以解决此问题。

这可能是一个IDE错误。例如,@ Joel Costigliola描述了Eclipse无法正确处理Maven“测试”树的情况:请参见此答案。

Android问题:在为Android编程时,您遇到与相关的“找不到符号”错误R,请注意,R符号是由context.xml文件定义的。检查您的context.xml文件正确无误,并且位置正确,并且R已生成/编译了相应的类文件。请注意,Java符号区分大小写,因此相应的XML ID也区分大小写。

Android上的其他符号错误可能是由于前面提到的原因引起的;例如,缺少或不正确的依赖项,不正确的程序包名称,在特定API版本中不存在的方法或字段,拼写/键入错误等。

重新定义系统类:我见过一些情况,编译器抱怨这种情况substring是未知符号,如下所示
 
String s = ...
String s1 = s.substring(1);

事实证明,程序员创建了自己的版本,String而他的类版本未定义substring方法。

课程:不要用与通用库类相同的名称来定义自己的类!

象形文字: 如果对源文件使用UTF-8编码,则可能具有看起来相同但实际上不同的标识符,因为它们包含同形文字。请参阅此页面以获取更多信息。

通过将自己限制为ASCII或Latin-1作为源文件编码,并对\uxxxx其他字符使用Java转义,可以避免这种情况。