类库下载 java类库 Java 프로그램 실행에 대한 기본 지식

Java 프로그램 실행에 대한 기본 지식

Oct 29, 2016 am 11:31 AM
java

함수 실행을 위한 JVM 스레드 스택

각 JVM 스레드는 시작 시 전용 스레드 스택을 생성합니다. jvm 스레드 스택은 스택 프레임을 저장하는 데 사용됩니다. jvm 스레드 스택은 C 언어의 스택과 매우 유사하며 로컬 변수 및 일부 작업 결과를 관리하며 함수 호출 및 함수 반환에도 관여합니다. JVM 사양에서 실행 중인 스레드 스택의 크기는 고정되거나 동적으로 할당되거나 특정 규칙에 따라 계산될 수 있습니다. JVM의 스택 구현은 다를 수 있습니다. 일부는 개발자에게 동적 할당을 위해 JVM 스레드 스택의 초기 크기를 제어하는 ​​방법을 제공할 수 있으며 JVM의 최대값과 최소값에 대한 설정도 제공할 수 있습니다.

스레드에 필요한 할당 크기를 계산하면 고정 값이나 설정된 최대 값을 초과하면 jvm에서 StackOverflowError가 발생합니다. 동적으로 할당된 스택의 경우 메모리가 최소값 또는 필수 값을 충족할 만큼 충분한 공간을 제공하지 못하는 경우 JVM은 OutOfMemoryError

스택 프레임을 발생시킵니다. 이는 매개변수를 관리하는 함수 실행 환경으로 이해할 수 있습니다. , 지역 변수, 반환 값 등

각 스택 프레임에는 로컬 변수(로컬 변수)를 관리하는 배열이 포함되어 있습니다. 이 배열의 단위 수는 바이트코드로 컴파일될 때 결정될 수 있습니다. 32비트의 경우 하나의 단위는 boolean, byte, char, short, int, float, reference, returnAddress를 저장할 수 있으며 두 개의 연속 단위는 long 및 double을 저장하는 데 사용될 수 있습니다. 지역 변수 배열의 첨자는 0부터 시작합니다. 일반적으로 0 위치에는 이를 저장하고, 함수의 매개변수, 함수에 나타나는 지역 변수가 차례로 저장됩니다.

각 스택 프레임에는 (LIFO) 작업 스택 데이터 구조(피연산자 스택)도 포함됩니다. 해당 크기는 컴파일 중에 결정될 수도 있습니다. 생성되면 빈 스택이 됩니다. int a+b의 경우 먼저 a를 스택에 넣은 다음 b를 스택에 넣은 다음 두 값을 동시에 팝하고 iadd 명령어를 실행합니다. 그런 다음 결과를 스택에 추가하여 명령어를 완료합니다.

위의 두 가지 주요 구조 외에도 각 스택 프레임에는 상수 풀(런타임 상수 풀), 예외 발생 관리 및 기타 구조가 있습니다. 여기서는 자세히 다루지 않고 다른 정보를 참고하시면 됩니다.

스택 프레임의 작업을 설명하기 위해 간단한 데모를 사용해 보겠습니다. 먼저 이러한 함수를 살펴보겠습니다.

public int comp(float number1, float number2){
        int result ;
        if(number1 < number2)
            result = 1;
        else
            result = 2;
        return result;
    }
로그인 후 복사

함수 내 로직에 해당하는 바이트코드는 다음과 같습니다.

0: fload_1

1: fload_2

2: fcmpg

3: ifge 11

6: icont_1

7: istore_3

8: goto 13

11: icont_2

12: istore_3

13: iload_3

14: ireturn

바이트코드 지침에 대해 간단히 설명하겠습니다.

fload_x: 지역 변수 배열에서 x번째 변수를 가져와 fload를 입력하고 스택에 푸시합니다.

fcmpg: 두 개의 단정밀도 부동 소수점 숫자를 비교합니다. 두 숫자가 결과보다 크면 결과는 1이고, 같으면 결과는 0이며, 결과보다 작으면 결과는 -1입니다.

ifge: 점프 명령; >

iconst_x: 상수 x를 스택에 푸시합니다.

istore_x: 스택을 x번째 지역 변수 배열로 팝합니다.

iload_x: x번째 지역 변수 배열을 읽어서 스택에 푸시합니다. stack;

ireturn: 함수 끝에서 int 유형을 반환합니다.

자세히 살펴보면 i의 시작 부분은 int를 나타내고 f의 시작 부분은 load를 나타냄을 알 수 있습니다. , load는 로딩을 나타내고, if는 점프를 나타냅니다. 바이트코드의 연산 코드 정의도 특정 의미를 가지며, 세부 사항은 jvm 바이트코드 관련 표준을 번역할 수 있습니다. 특정 호출 comp(1.02,2.02)를 예로 들어 jvm이 스택 프레임 구조에서 실행되는 방식을 살펴보겠습니다.

Java 프로그램 실행에 대한 기본 지식

Java 클래스

바이트코드를 논할 때 .class는 필수 불가결합니다. 클래스의 내용을 자세히 보려면 ​​데모 클래스를 수강하는 것이 좋습니다. 클래스는 두 가지 함수 중 하나는 위의 cmp 함수입니다.

public class Hello {

    public void say(){
        System.out.println("Hello world!");
    }

    public int comp(float number1, float number2){
        int result ;
        if(number1 < number2)
            result = 1;
        else
            result = 2;
        return result;
    }
}
로그인 후 복사
이 클래스를 컴파일하려면 javac -g:none Hello.java를 사용하고, 컴파일된 클래스를 구문 분석하려면 javap -c -v Hello.class를 사용하세요.

Classfile /src/main/java/com/demo/Hello.class
  Last modified 2016-10-28; size 404 bytes
  MD5 checksum 9ac6c800c312d65b568dd2a0718bd2c5public class com.demo.Hello
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:   #1 = Methodref          #6.#14         // java/lang/Object."<init>":()V
   #2 = Fieldref           #15.#16        // java/lang/System.out:Ljava/io/PrintStream;
   #3 = String             #17            // Hello world!
   #4 = Methodref          #18.#19        // java/io/PrintStream.println:(Ljava/lang/String;)V
   #5 = Class              #20            // com/demo/Hello
   #6 = Class              #21            // java/lang/Object
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               say
  #11 = Utf8               comp
  #12 = Utf8               (FF)I
  #13 = Utf8               StackMapTable
  #14 = NameAndType        #7:#8          // "<init>":()V
  #15 = Class              #22            // java/lang/System
  #16 = NameAndType        #23:#24        // out:Ljava/io/PrintStream;
  #17 = Utf8               Hello world!
  #18 = Class              #25            // java/io/PrintStream
  #19 = NameAndType        #26:#27        // println:(Ljava/lang/String;)V
  #20 = Utf8               com/demo/Hello
  #21 = Utf8               java/lang/Object
  #22 = Utf8               java/lang/System
  #23 = Utf8               out
  #24 = Utf8               Ljava/io/PrintStream;
  #25 = Utf8               java/io/PrintStream
  #26 = Utf8               println
  #27 = Utf8               (Ljava/lang/String;)V{  public com.demo.Hello();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:      stack=1, locals=1, args_size=1
         0: aload_0         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return

  public void say();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:      stack=2, locals=1, args_size=1
         0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
         3: ldc           #3                  // String Hello world!
         5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
         8: return

  public int comp(float, float);
    descriptor: (FF)I
    flags: ACC_PUBLIC
    Code:      stack=2, locals=4, args_size=3
         0: fload_1         1: fload_2         2: fcmpg         3: ifge          11
         6: iconst_1         7: istore_3         8: goto          13
        11: iconst_2        12: istore_3        13: iload_3        14: ireturn
      StackMapTable: number_of_entries = 2
        frame_type = 11 /* same */
        frame_type = 252 /* append */
          offset_delta = 1
          locals = [ int ]
}
로그인 후 복사
관련된 새로운 연산 코드를 설명하세요

getstatic:获取镜头变量;
invokevirtual:调用函数;
return:void 函数结束返回;
로그인 후 복사
공용 int comp(float, float) 코드에서 위에서 언급한 단어를 볼 수 있습니다. 섹션 코드 실행의 예입니다. 지각적으로 이해하면 실제로 클래스 파일에는 바이트코드 명령어 외에 상수 풀, 액세스 플래그(공용 등), 클래스 관련 정보(속성, 함수, 상수 등)도 포함되어 있음을 대략적으로 알 수 있습니다. . 이전에 -g:node를 컴파일에 사용했기 때문에 다른 확장 및 디버깅 정보도 다른 모드의 클래스에 포함될 수 있습니다. 공식 클래스 파일 형식은 다음과 같습니다.

ClassFile {
    u4             magic;
    u2             minor_version;
    u2             major_version;
    u2             constant_pool_count;
    cp_info        constant_pool[constant_pool_count-1];
    u2             access_flags;
    u2             this_class;
    u2             super_class;
    u2             interfaces_count;
    u2             interfaces[interfaces_count];
    u2             fields_count;
    field_info     fields[fields_count];
    u2             methods_count;
    method_info    methods[methods_count];
    u2             attributes_count;
    attribute_info attributes[attributes_count];
}
로그인 후 복사
magic: 아주 유명한 로고 클래스 파일인 0xCAFEBABE입니다.

minor_version 、major_version :指的是java class 文件的版本,一般说class文件的版本是 XX.xx 其中XX 就是major,xx是minor,比如上面demo中的版本是52.0 代表就是 minor 0,major 51.

constant_pool_count:就是常量池元素个数,cp_info constant_pool[constant_pool_count-1] 就是相关的详细信息了。

access_flags:指的是访问标识例如ACC_PUBLIC、ACC_FINAL、ACC_INTERFACE、ACC_SUPER 写过java的相信看名字应该知道啥意思,ACC是access的缩写。

其他具体的,就不一一介绍了详细可以直接参考官方文档。

动态生成java字节码

当然,你可以直接按照官方的class文件格式来直接写 byte[],然后自定义个 class load 载入编写的byte[]来实现动态生成class。不过,这个要求可能也有点高,必须的非常熟悉class文件格式才能做到。这里demo还是借助 ASM 这个类库来简单演示下,就编写下 上面的Hello 不过里面只实现say的方法。如下:

public class AsmDemo {    public static final String CLASS_NAME = "Hello";    
    public static final AsmDemoLoad load = new AsmDemoLoad();    private static class AsmDemoLoad extends ClassLoader {        public AsmDemoLoad() {            super(AsmDemo.class.getClassLoader());
        }        public Class<?> defineClassForName(String name, byte[] data) {            return this.defineClass(name, data, 0, data.length);
        }
    }    public static byte[] generateSayHello() throws IOException {

        ClassWriter classWriter = new ClassWriter(ClassWriter.COMPUTE_MAXS);
        classWriter.visit(V1_7, ACC_PUBLIC + ACC_SUPER, CLASS_NAME, null, getInternalName(Object.class), null);    
        //默认初始化函数
        Method constructorMethod = Method.getMethod("void <init> ()");
        GeneratorAdapter constructor = new GeneratorAdapter(ACC_PUBLIC, constructorMethod, null, null, classWriter);
        constructor.loadThis();        //每个类都要基础Object
        constructor.invokeConstructor(Type.getType(Object.class), constructorMethod);
        constructor.returnValue();
        constructor.endMethod();

        Method mainMethod = Method.getMethod("void say ()");
        GeneratorAdapter main = new GeneratorAdapter(ACC_PUBLIC, mainMethod, null, null, classWriter);
        main.getStatic(Type.getType(System.class), "out", Type.getType(PrintStream.class));
        main.push("Hello world!");
        main.invokeVirtual(Type.getType(PrintStream.class), Method.getMethod("void println (String)"));
        main.returnValue();
        main.endMethod();        return classWriter.toByteArray();
    }    public static void main(String[] args) throws IllegalAccessException, IllegalArgumentException,
            InvocationTargetException, InstantiationException, NoSuchMethodException, SecurityException, IOException {        byte[] code = AsmDemo.generateSayHello();        //反射构建 hello 类,调用hello方法。
        Class<?> hello = load.defineClassForName(CLASS_NAME, code);
        hello.getMethod("say", null).invoke(hello.newInstance(), null);
    }
}
로그인 후 복사

关于动态生成字节码用途,一定场景下是可以提升效率与性能,因为动态生成的类和普通的载入类并无太大区别。手工优化后的字节码执行可能比编译的要优,可以替代反射使用的许多场景 同时避免反射的性能消耗。很著名的一个例子,fastJSON 就是使用内嵌 ASM 框架动态生成字节码类,来进行序列和反序列化工作,是目前公认最快的json字符串解析。

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

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

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

자바의 완전수 자바의 완전수 Aug 30, 2024 pm 04:28 PM

Java의 완전수 가이드. 여기서는 정의, Java에서 완전 숫자를 확인하는 방법, 코드 구현 예제에 대해 논의합니다.

자바의 웨카 자바의 웨카 Aug 30, 2024 pm 04:28 PM

Java의 Weka 가이드. 여기에서는 소개, weka java 사용 방법, 플랫폼 유형 및 장점을 예제와 함께 설명합니다.

Java의 스미스 번호 Java의 스미스 번호 Aug 30, 2024 pm 04:28 PM

Java의 Smith Number 가이드. 여기서는 정의, Java에서 스미스 번호를 확인하는 방법에 대해 논의합니다. 코드 구현의 예.

Java Spring 인터뷰 질문 Java Spring 인터뷰 질문 Aug 30, 2024 pm 04:29 PM

이 기사에서는 가장 많이 묻는 Java Spring 면접 질문과 자세한 답변을 보관했습니다. 그래야 면접에 합격할 수 있습니다.

Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Feb 07, 2025 pm 12:09 PM

Java 8은 스트림 API를 소개하여 데이터 컬렉션을 처리하는 강력하고 표현적인 방법을 제공합니다. 그러나 스트림을 사용할 때 일반적인 질문은 다음과 같은 것입니다. 기존 루프는 조기 중단 또는 반환을 허용하지만 스트림의 Foreach 메소드는이 방법을 직접 지원하지 않습니다. 이 기사는 이유를 설명하고 스트림 처리 시스템에서 조기 종료를 구현하기위한 대체 방법을 탐색합니다. 추가 읽기 : Java Stream API 개선 스트림 foreach를 이해하십시오 Foreach 메소드는 스트림의 각 요소에서 하나의 작업을 수행하는 터미널 작동입니다. 디자인 의도입니다

Java의 날짜까지의 타임스탬프 Java의 날짜까지의 타임스탬프 Aug 30, 2024 pm 04:28 PM

Java의 TimeStamp to Date 안내. 여기서는 소개와 예제와 함께 Java에서 타임스탬프를 날짜로 변환하는 방법에 대해서도 설명합니다.

캡슐의 양을 찾기위한 Java 프로그램 캡슐의 양을 찾기위한 Java 프로그램 Feb 07, 2025 am 11:37 AM

캡슐은 3 차원 기하학적 그림이며, 양쪽 끝에 실린더와 반구로 구성됩니다. 캡슐의 부피는 실린더의 부피와 양쪽 끝에 반구의 부피를 첨가하여 계산할 수 있습니다. 이 튜토리얼은 다른 방법을 사용하여 Java에서 주어진 캡슐의 부피를 계산하는 방법에 대해 논의합니다. 캡슐 볼륨 공식 캡슐 볼륨에 대한 공식은 다음과 같습니다. 캡슐 부피 = 원통형 볼륨 2 반구 볼륨 안에, R : 반구의 반경. H : 실린더의 높이 (반구 제외). 예 1 입력하다 반경 = 5 단위 높이 = 10 단위 산출 볼륨 = 1570.8 입방 단위 설명하다 공식을 사용하여 볼륨 계산 : 부피 = π × r2 × h (4

미래를 창조하세요: 완전 초보자를 위한 Java 프로그래밍 미래를 창조하세요: 완전 초보자를 위한 Java 프로그래밍 Oct 13, 2024 pm 01:32 PM

Java는 초보자와 숙련된 개발자 모두가 배울 수 있는 인기 있는 프로그래밍 언어입니다. 이 튜토리얼은 기본 개념부터 시작하여 고급 주제를 통해 진행됩니다. Java Development Kit를 설치한 후 간단한 "Hello, World!" 프로그램을 작성하여 프로그래밍을 연습할 수 있습니다. 코드를 이해한 후 명령 프롬프트를 사용하여 프로그램을 컴파일하고 실행하면 "Hello, World!"가 콘솔에 출력됩니다. Java를 배우면 프로그래밍 여정이 시작되고, 숙달이 깊어짐에 따라 더 복잡한 애플리케이션을 만들 수 있습니다.

See all articles