Java에서 클래스 경로 및 패키지를 사용하는 방법
1. 클래스 경로
자바를 설치하고 신나게 hello world를 작성한 다음 컴파일하고 실행하면 그 두 단어가 앞에 나타날 때까지 기다리지만 안타깝게도 찾을 수 없습니다. 스레드 "main" java.lang.NoSuchMethodError의 HelloWorld 클래스 또는 예외: maain. 컴파일된 클래스가 분명히 존재합니다.
Java 프로그램의 실행 과정을 살펴보겠습니다. Java는 Java 가상 머신을 통해 즉, java 명령을 통해 생성되는 .class 파일을 통해 해석되고 실행된다는 것을 이미 알고 있습니다. javac 컴파일은 가상 머신에서 실행되는 코드를 바이트코드라고 합니다. 가상 머신은 일반적인 의미의 클래스인 클래스로더를 통해 이러한 바이트코드를 로드합니다. 라이브러리와 사용자 자신의 수업은 어디에 있습니까? 또는 기본값(현재 경로)이 있거나 이 변수가 클래스 경로임을 나타내는 사용자 지정 변수가 있거나 런타임 시 매개변수가 가상 머신에 전달됩니다. 컴파일 과정과 실행 과정은 비슷하지만 하나는 찾아서 컴파일하는 것이고, 다른 하나는 찾아서 로드하는 것이다. 실제로 자바 가상머신은 java(또는 java.exe) 프로그램인 java luncher에 의해 초기화된다. ). 가상 머신은
-
1, Boot 클래스: rt.jar 및 i18n.jar의 클래스를 포함하여 Java 플랫폼을 구성하는 클래스를 모두 검색하고 로드합니다.
2, 확장 클래스: - Java 확장 메커니즘을 사용하는 클래스는 모두 확장 디렉터리($JAVA_HOME/jre/lib/ext)에 있는 .jar 아카이브입니다.
3,
사용자 클래스: - 개발자 정의. 클래스 또는 Java 확장 메커니즘을 사용하는 타사 제품이 없습니다. 위에서 언급한 사용자 고유 클래스의 위치를 확인하려면 명령줄에서 -classpath 옵션을 사용하거나 CLASSPATH 환경 변수를 사용해야 합니다. 이런 방식으로, 일반적으로 사용자는 사용자 클래스의 위치만 지정하면 되며, 부팅 클래스와 확장 클래스는 "자동으로" 찾아집니다.
그렇다면 어떻게 해야 할까요? 사용자 클래스 경로는 클래스 파일, .jar, .zip 파일이 포함된 디렉터리 목록입니다. 구체적으로 클래스를 찾는 방법은 패키지 문제와 관련되어 있기 때문에 당분간은 다음과 같이 설명하겠습니다. 이 클래스가 포함되어 있으면 발견되는 것으로 간주됩니다. 구분 기호는 플랫폼에 따라 약간 다릅니다. Unix 계열 시스템은 기본적으로 ":"이며, 가능한 소스는 대부분 ";"입니다. :
1.", 즉, 현재 디렉터리가 기본값입니다.
2. CLASSPATH 환경 변수는 일단 설정되면 기본값을 덮어씁니다. 명령줄 매개변수 -cp 또는 -classpath가 지정되면 위의 두 가지는
4가 됩니다. -jar 매개변수로 지정된 .jar 아카이브는 실행 파일 이후 모든 다른 값을 덮어씁니다. .jar 파일이 생성되면 이 기사의 끝부분에서 언급할 패키지 및 특정 구성 파일과 같은 몇 가지 다른 지식이 필요합니다. 먼저 jdk와 함께 제공되는 몇 가지 예를 살펴보겠습니다. 설명할 HelloWorld의 예입니다. 먼저 다음 가정을 수행합니다.
- 1. 현재 디렉터리는 /HelloWorld(또는 c:HelloWorld, 앞으로는 이전 디렉터리가 사용됨)
- 2.jdk 버전입니다. 1.2.2(Linux에서)
3.PATH 환경 변수가 올바르게 설정되었습니다. (이렇게 하면 모든 디렉터리에서 도구를 사용할 수 있습니다.)
- 4 파일은 HelloWorld.java입니다. :
public class HelloWorld { public static void main(String[] args) { System.out.println("Hello World!\n"); System.exit(0); } }
우선 이 파일을 정확하게 작성해야 합니다. c에 익숙하다면 다음과 같이 작성할 가능성이 매우 높습니다.
public static void main(int argc, String[] argv) { .... }
지금까지는 PATH만 설정했습니다.
[HelloWorld]$ javac HelloWorld.java //这一步不会有多大问题, [HelloWorld]$ java HelloWorld // 这一步可能就会有问题.
[HelloWorld]$ echo $CLASSPATH 或者 c:\HelloWorld>echo %CLASSPATH%CLASSPATH 환경 변수가 설정되어 있는지 확인하세요.
[HelloWorld]$ CLASSPATH= 或者 c:\HelloWorld> set CLASSPATH=비워둔 다음 다시 실행하세요. 이번에는 기본 사용자 클래스 경로가 "."이므로 동일한 문제가 발생하지 않습니다. 또 다른 방법은 CLASSPATH에 "."을 추가하는 것입니다.
[/]$ CLASSPATH=$CLASSPATH:. 或者 c:\HelloWorld> set CLASSPATH=%CLASSPATH%;.행운을 빕니다.2. 프로그램에 타사 클래스 라이브러리 지원이 필요하고 비교적 일반적인 경우 이 방법을 사용할 수 있습니다. 예를 들어 서블릿을 작성하는 데 필요한 서블릿 패키지입니다. 방법은 환경 변수에 CLASSPATH를 추가하는 것입니다. 그러면 직접 컴파일하고 실행할 수 있습니다. 예를 들어 HelloWorld를 루트 디렉터리에서 실행하려면 루트 디렉터리에서 직접 실행하면 됩니다.
$ java HelloWorld 或者 c:\>java HelloWorld
这样肯定会出错, 如果你的 CLASSPATH 没有改动的话. 我想大家应该知道为什么错了吧, 那么怎么改呢? 前面说过, 用户类路径就是一些包含你所需要的类的目录, .jar 档案包, .zip 包. 现在没有生成包, 所以只好把 HelloWorld.class 所在的目录加到 CLASSPAT了, 根据前面的做法, 再运行一次, 看看, 呵呵, 成功了, 换个路径, 又成功了!! 不仅仅可以直接运行其中的类, 当你要 import 其中的某些类时, 同样处理. 不知道你想到没有, 随着你的系统的不断的扩充, (当然了, 都是一些需要 java 的东西).如果都加到这个环境变量里, 那这个变量会越来越臃肿, 虽然环境变量空间可以开很大, 总觉得有些不舒服. 看看下面一个方法.
3, 在命令行参数中指明 classpath. 还是和上面相同的目标, 在任何目录下执行 HelloWorld, 用这个方法怎么实现呢?
[/]$ java -cp /HelloWorld HelloWorld 或者 c:\>java -cp c:\HelloWorld HelloWorld
就可以了. 这是这种方法的最简单的应用了. 当你使用了另外的包的时候, 还可以采用用这种方法.
例如:
$ javac -classpath aPath/aPackage.jar:. myJava.java $ java -cp aPath/aPackage.jar:. myJava 或者 c:\> javac -classpath aPath\aPackage.jar;. myJava.java c:\> java -cp aPath\aPackage.jar;. myJava
这种方法也有一个不方便的的地方就是当第三方包所在的路径较长或者需要两个以上包的时候, 每次编译运行都要写很长, 非常不方便, 这时候可以写脚本来解决.
比如一个例子:
compile (文件, 权限改为可执行, 当前目录) $ cat compile --------------------------- #!/bin/bash javac -classpath aPath\aPackage.jar:anotherPath\anotherPackage.jar:. m yJavva.java --------------------------- run (文件, 权限改为可执行, 当前目录) $cat run --------------------------- #!/bin/bash java -cp aPath\aPackage.jar:anotherPath\anotherPackage.jar:. myJava --------------------------- 或者: compile.bat c:\HelloWorld> type compile.bat ------------------------- javac -classpath aPath\aPackage.jar:anotherPath\anotherPackage.jar:. m yJavva.java ------------------------- run.bat c:\HelloWorld> type run.bat ------------------------ java -cp aPath\aPackage.jar:anotherPath\anotherPackage.jar:. myJava ------------------------
就可以了. 试试看.
前面提到了扩展类, 扩展类是什么呢? java 的扩展类就是应用程序开发者用来扩展核心平台功能的 java 类的包(或者是 native code). 虚拟机能像使用系统类一样使用这些扩展类. 有人建议可以把包放入扩展目录里, 这样, CLASSPATH 也不用设了,也不用指定了, 岂不是很方便? 确实可以正确运行, 但是个人认为这样不好, 不能什么东西都往里搁, 一些标准的扩展包可以, 比如, JavaServlet, Java3D 等等. 可以提个建议, 加一个环境变量, 比如叫 JARPATH, 指定一个目录, 专门存放用户的 jar zip等包, 这个要等 SUN 公司来做了. windows98 下, 我原来安装的时候, 一直装不上, 总是死机, 好不容易装上了, 缺省的是不能运行正确的, 然后把 tool.jar 放入 CLASSPATH 后工作正常. 现在作测试,去掉仍然是正确的. 经过多次测试, 发现如果原来曾装过 jdk 的都很好, 没有装过的,装的时候会死机, 多装几次就可以了. 如果你发现正确安装后, 不能正常工作, 就把tools.jar 加入 CLASSPATH, 试一下.
二、 包 (package)
Java 中的 "包" 是一个比较重要的概念, package 是这样定义的:Definition: A package is a collection of related classes and interfaces that provides access protection and namespace management. 也就是: 一个包就是一些提供访问保护和命名空间管理的相关类与接口的集合. 使用包的目的就是使类容易查找使用, 防止命名冲突, 以及控制访问. 这里我们不讨论关于包的过多的东西, 只讨论和编译, 运行, 类路径相关的东西.至于包的其他内容, 请自己查阅相关文档. 简单一点来说, 包就是一个目录, 下面的子包就是子目录, 这个包里的类就是这个目录下的文件. 我们用一个例子来说明.
首先建目录结构如下: PackageTest/source/, 以后根目录指的是 PackageTest目录, 我们的源程序放在 source 目录下.
源程序如下:
PackageTest.java package pktest; import pktest.subpk.*; public class PackageTest { private String value; public PackageTest(String s) { value = s; } public void printValue() { System.out.println("Value of PackageTest is " + value); } public static void main(String[] args) { PackageTest test = new PackageTest("This is a Test Package"); test.printValue(); PackageSecond second = new PackageSecond("I am in PackageTest"); second.printValue(); PackageSub sub = new PackageSub("I am in PackageTest"); sub.printValue(); System.exit(0); } } PackageSecond.java package pktest; public class PackageSecond { private String value; public PackageSecond(String s) { value = s; } public void printValue() { System.out.println("Value of PackageSecond is " + value); } } PackageSub.java package pktest.subpk; import pktest.*; public class PackageSub { private String value; public PackageSub(String s) { value = s; } public void printValue() { PackageSecond second = new PackageSecond("I am in subpackage."); second.printValue(); System.out.println("Value of PackageSub is " + value); } } Main.java import pktest.*; import pktest.subpk.*; public class Main() { public static void main() { PackageSecond second = new PackageSecond("I am in Main"); second.printValue(); PackageSub sub = new PackageSub("I am in Main"); sub.printValue(); System.exit(0); } }
其中,Main.java是包之外的一个程序,用来测试包外的程序访问包内的类,PackageTest.java属于pktest这个包,也是主程序.PackageSecond.java也属于pktest,PackageSub属于pktest下的subpk包,也就是pktest.subpk.详细使用情况,请参看源程序。
好了, 先把源程序都放在 source 目录下, 使 source 成为当前目录, 然后编译一下, 呵呵, 出错了,Main.java:1: Package pktest not found in import.import pktest.*; 这里涉及到类路径中包是怎么查找的, 前面我们做了一点假设: "只要包含了这个类就算找到了这个类", 现在就有问题了. 其实 jdk 的 工具 javac javajavadoc 都需要查找类, 看见目录, 就认为是包的名字, 对于import 语句来说,一个包对应一个目录. 这个例子中, import pktest.*, 我们知道类路径可以包含一个目录, 那么就以那个目录为根, 比如有个目录 /myclass, 那么就会在查找/myclass/pktest 目录及其下的类. 所有的都找遍, 如果没有就会报错. 由于现在的类路径只有当前目录, 而当前目录下没有 pktest 目录, 所以就会出错. 类路径还可以包含 .jar .zip 文件, 这些就是可以带目录的压缩包, 可以把 .jar .zip文件看做一个虚拟的目录, 然后就和目录一样对待了.
好了, 应该知道怎么做了吧, 修改后的目录结构如下:
PackageTest | |__source Main.java | |__pktest PackageTest.java PackageSecond.java | |__subpk PackageSub.java
然后重新编译, 运行, 哈哈, 通过了. 我们再来运行一下 PackageTest.
[source]$ java pktest/PackageTest
怎么又出错了?
Exception in thread "main" java.lang.NoClassDefFoundError: pktest/PackageTest 是这样的, java 所要运行的是一个类的名字, 它可不管你的类在什么地方, 就象我们前面所讨论的一样来查找这个类, 所以它把 pktest/PackageTest 看成是一个类的名字了, 当然会出错了, 应该这么做,
[source]$ java pktest.PackageTest
大家应该明白道理吧, 我就不多说了. 注意 javac 不一样, 是可以指明源文件路径的, javac 只编译, 不运行, 查找类也只有在源文件中碰到 import 时才会做, 与源文件所在的包没有关系. 似乎还又些不好的地方, 怎么生成的 .class 文件这么分散呀, 看着真别扭. 别急,javac 有一个 -d 命令行参数, 可以指定一个目录, 把生成的 .class 文件按照包给你好好地搁在这个目录里面.
[source]$ mkdir classes [source]$ javac -d classes pktest/PackageTest.java [source]$ javac -d classes Main.java
那么运行怎么运行呢?
[source]$ cd classes [classes]$ java pktest.PackageTest [classes]$ java Main
就可以了. 其实 jdk 的这一套工具小巧简单, 功能强大, 不会用或者用错其实不关工具的事, 关键是明白工具背后的一些原理和必要的知识. 集成环境是很好,但是它屏蔽了很多底层的知识, 不出错还好, 一旦出错, 如果没有这些必要的知识就很难办, 只好上 bbs 问, 别人只告诉了你解决的具体方法, 下一次遇到稍微变化一点的问题又不懂了. 所以不要拘泥于工具, java 的这一套工具组合起来使用, 中小型工程(五六十个类), 还是应付得下来的.
三、 jar 文件
以下把 .jar .zip 都看做是 .jar 文件.
1, 从前面我们可以看出来 jar 文件在 java 中非常重要, 极大地方便了用户的使用. 我们也可以做自己的 .jar 包.
还是使用前面那个例子, Main.java 是包之外的东西, 用了 pktest 包中的类,我们现在就是要把 pktest 做成一个 .jar 包, 很简单, 刚才我们已经把 pktest中的 .class 都集中起来了,
就会生成 mypackage.jar 文件, 测试一下, 刚才我们生成的 Main.class 就在classes 目录下, 所以, 从前面可以知道:
[classes]$ java -cp mypackage.jar:. Main
就可以运行了.
2, 如果你看过 jdk 所带的例子, 你就会知道, .jar 还可以直接运行,
[/demo]$ java -jar aJar.jar
那好, 就那我们的试一试,
看来我们的 jar 和它的 jar 还不一样, 有什么不一样呢? 拿它一个例子出来,重新编译, 生成 .jar 文件, 比较后发现, 是 .jar 压缩包中 META-INF/MANIFEST.MF文件不一样, 多了一行, Main-Class: xxxxx, 再看看出错信息, 原来是没有指定Main-Class, 看看 jar 命令, 发现有一个参数 -m,-m include manifest information from specified manifest file
和出错信息有点关系, 看来它要读一个配制文件. 只好照猫画虎写一个了.
[classes]$ cat myManifest Manifest-Version: 1.0 Main-Class: pktest.PackageTest Created-By: 1.2.2 (Sun Microsystems Inc.) [classes]$ jar -cvfm mypackage.jar myManifest pktest added manifest adding: pktest/(in = 0) (out= 0)(stored 0%) adding: pktest/PackageSecond.class(in = 659) (out= 395)(deflated 40%) adding: pktest/subpk/(in = 0) (out= 0)(stored 0%) adding: pktest/subpk/PackageSub.class(in = 744) (out= 454)(deflated 38%) adding: pktest/PackageTest.class(in = 1041) (out= 602)(deflated 42%) [classes]$ java -jar mypackage.jar Value of PackageTest is This is a Test Package Value of PackageSecond is I am in PackageTest Value of PackageSecond is I am in subpackage. Value of PackageSub is I am in PackageTest
好了, 成功了, 这样就做好了一个可以直接执行的 .jar 文件. 大家可以自己试一试
做一个以 Main 为主程序的可执行的 jar.
下面是一些需要注意的问题:
1, 如果类路径中需要用到 .jar 文件, 必须把 jar 文件的文件名放入类路径, 而不是其所在的目录.
2, 在任何时候, 类名必须带有完全的包名,
3, "." 当前目录最好在你的类路径中.下面是一些常见的编译和运行的模式.
4. To compile HelloWorld.java app in the default package in C:\MyDir, use CD \MyDir C:\jdk1.3\bin\Javac.exe -classpath . HelloWorld.java 5. To run a HelloWorld.class app, in the default package in C:\MyDir, use CD \MyDir C:\jdk1.3\bin\Java.exe -classpath . HelloWorld 6. To run a HelloWorld.class app, in the default package in a jar in C:\MyDir, use CD \MyDir C:\jdk1.3\bin\Java.exe -classpath HelloWorld.jar HelloWorld 7. To compile a HelloWorld.java app in C:\MyPackage, in package MyPackage, use CD \ C:\jdk1.3\bin\Javac.exe -classpath . MyPackage\HelloWorld.java 8. To run a HelloWorld.class app in C:\MyPackage, in package MyPackage, use CD \ C:\jdk1.3\bin\Java.exe -classpath . MyPackage.HelloWorld 9. To run a HelloWorld.class app in C:\MyPackage, in a jar in package MyPackage,, use CD \MyDir C:\jdk1.3\bin\Java.exe -classpath HelloWorld.jar MyPackage.HelloWorl
(注: default package 指的是在程序中不指定任何包). 最后一个小小的建议, 把 sun 的 jdk tools documentation 好好地看一看,把 jdk 的那些工具 java javac javadoc jar javap jdb......好好用一用, 会有好处的. The Simplest Is The Best.
위 내용은 Java에서 클래스 경로 및 패키지를 사용하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

javaispopularforcross-platformdesktopapplicationsduetoits "writeonce, runanywhere"철학

Java에서 플랫폼 별 코드를 작성하는 이유에는 특정 운영 체제 기능에 대한 액세스, 특정 하드웨어와 상호 작용하고 성능 최적화가 포함됩니다. 1) JNA 또는 JNI를 사용하여 Windows 레지스트리에 액세스하십시오. 2) JNI를 통한 Linux 특이 적 하드웨어 드라이버와 상호 작용; 3) 금속을 사용하여 JNI를 통해 MacOS의 게임 성능을 최적화하십시오. 그럼에도 불구하고 플랫폼 별 코드를 작성하면 코드의 이식성에 영향을 미치고 복잡성을 높이며 잠재적으로 성능 오버 헤드 및 보안 위험을 초래할 수 있습니다.

Java는 Cloud-Native Applications, Multi-Platform 배포 및 교차 운용성을 통해 플랫폼 독립성을 더욱 향상시킬 것입니다. 1) Cloud Native Applications는 Graalvm 및 Quarkus를 사용하여 시작 속도를 높입니다. 2) Java는 임베디드 장치, 모바일 장치 및 양자 컴퓨터로 확장됩니다. 3) Graalvm을 통해 Java는 Python 및 JavaScript와 같은 언어와 완벽하게 통합되어 언어 교차 수용 가능성을 향상시킵니다.

Java의 강력한 유형 시스템은 유형 안전, 통합 유형 변환 및 다형성을 통해 플랫폼 독립성을 보장합니다. 1) 유형 안전성 런타임 오류를 피하기 위해 컴파일 시간에 유형 검사를 수행합니다. 2) 통합 유형 변환 규칙은 모든 플랫폼에서 일관성이 있습니다. 3) 다형성 및 인터페이스 메커니즘은 코드가 다른 플랫폼에서 일관되게 행동하게 만듭니다.

JNI는 Java의 플랫폼 독립성을 파괴 할 것입니다. 1) JNI는 특정 플랫폼에 대한 로컬 라이브러리를 요구합니다. 2) 대상 플랫폼에서 로컬 코드를 컴파일하고 연결해야합니다. 3) 운영 체제 또는 JVM의 다른 버전은 다른 로컬 라이브러리 버전을 필요로 할 수 있습니다.

신흥 기술은 위협을 일으키고 Java의 플랫폼 독립성을 향상시킵니다. 1) Docker와 같은 클라우드 컴퓨팅 및 컨테이너화 기술은 Java의 플랫폼 독립성을 향상 시키지만 다양한 클라우드 환경에 적응하도록 최적화되어야합니다. 2) WebAssembly는 Graalvm을 통해 Java 코드를 컴파일하여 플랫폼 독립성을 확장하지만 성능을 위해 다른 언어와 경쟁해야합니다.

다른 JVM 구현은 플랫폼 독립성을 제공 할 수 있지만 성능은 약간 다릅니다. 1. OracleHotspot 및 OpenJDKJVM 플랫폼 독립성에서 유사하게 수행되지만 OpenJDK에는 추가 구성이 필요할 수 있습니다. 2. IBMJ9JVM은 특정 운영 체제에서 최적화를 수행합니다. 3. Graalvm은 여러 언어를 지원하며 추가 구성이 필요합니다. 4. AzulzingJVM에는 특정 플랫폼 조정이 필요합니다.

플랫폼 독립성은 여러 운영 체제에서 동일한 코드 세트를 실행하여 개발 비용을 줄이고 개발 시간을 단축시킵니다. 구체적으로, 그것은 다음과 같이 나타납니다. 1. 개발 시간을 줄이면 하나의 코드 세트 만 필요합니다. 2. 유지 보수 비용을 줄이고 테스트 프로세스를 통합합니다. 3. 배포 프로세스를 단순화하기위한 빠른 반복 및 팀 협업.


핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

SecList
SecLists는 최고의 보안 테스터의 동반자입니다. 보안 평가 시 자주 사용되는 다양한 유형의 목록을 한 곳에 모아 놓은 것입니다. SecLists는 보안 테스터에게 필요할 수 있는 모든 목록을 편리하게 제공하여 보안 테스트를 더욱 효율적이고 생산적으로 만드는 데 도움이 됩니다. 목록 유형에는 사용자 이름, 비밀번호, URL, 퍼징 페이로드, 민감한 데이터 패턴, 웹 셸 등이 포함됩니다. 테스터는 이 저장소를 새로운 테스트 시스템으로 간단히 가져올 수 있으며 필요한 모든 유형의 목록에 액세스할 수 있습니다.

SublimeText3 영어 버전
권장 사항: Win 버전, 코드 프롬프트 지원!

Atom Editor Mac 버전 다운로드
가장 인기 있는 오픈 소스 편집기

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경
