목차
바이트코드란 무엇입니까?
바이트코드 보기
바이트코드 분석
바이트코드 분석 활용
리버스 엔지니어링
Java java지도 시간 재미와 이익을 위해 Java 바이트코드를 읽는 방법

재미와 이익을 위해 Java 바이트코드를 읽는 방법

Oct 22, 2024 pm 01:03 PM

Java 바이트코드의 세계로 여행을 떠나시나요? 이 기사에서는 시작하기 위해 알아야 할 모든 것을 다룹니다.

재미와 이익을 위해 Java 바이트코드를 읽는 방법

바이트코드란 무엇입니까?

1995년, Java 프로그래밍 창시자인 Sun Microsystems는 언어로 과감한 주장을 펼쳤습니다. 그들은 Java를 사용하면 "한 번만 작성하면 어디에서나 실행할 수 있다"고 말했습니다. 이는 컴파일된 바이너리가 모든 시스템 아키텍처에서 실행될 수 있다는 것을 의미하며, 이는 C가 할 수 없는 일이며 오늘날까지도 Java 작성의 핵심 테넌트로 남아 있습니다.

이러한 크로스 플랫폼 기능을 달성하기 위해 Java는 다음을 사용합니다. 컴파일할 때 독특한 접근 방식을 사용합니다. 소스 코드에서 기계어 코드(각 시스템 아키텍처에 따라 다름)로 직접 이동하는 대신 Java는 프로그램을 바이트코드라는 중간 형식으로 컴파일합니다. 바이트코드는 특정 기계어에 얽매이지 않고 특정 하드웨어 아키텍처에 종속되지 않는 명령어 세트입니다. 이러한 추상화는 Java 이식성의 핵심입니다.

Java 바이트코드 명령을 해석하고 실행하는 프로그램을 JVM(Java Virtual Machine)이라고 합니다. JVM은 각 바이트코드 명령어를 실행 중인 특정 시스템 아키텍처의 기본 기계어 코드로 변환합니다. 종종 "JIT(Just-In-Time)" 컴파일이라고도 하는 이 프로세스를 통해 Java 바이트코드가 특정 플랫폼에서 최대한 효율적으로 실행될 수 있습니다.

바이트코드 보기

바이트코드는 하지만 JVM에만 유용한 것은 아닙니다. Java 클래스의 바이트코드는 리버스 엔지니어링, 성능 최적화, 보안 연구 및 기타 정적 분석 기능에 유용하기 때문에 JDK에는 여러분과 제가 이를 검사하는 데 도움이 되는 유틸리티가 함께 제공됩니다.

예를 살펴보려면 바이트코드의 경우 `boolean` 기본 유형을 각각 unbox 및 box하는 `java.lang.Boolean`, `booleanValue` 및 `valueOf(boolean)`의 다음 두 가지 메소드를 고려하십시오.

public boolean booleanValue() {
    return value;
}

public static Boolean valueOf(boolean b) {
    return (b ? TRUE : FALSE);
}

` 사용 JDK와 함께 제공되는 javap` 명령을 사용하면 각각에 대한 바이트코드를 볼 수 있습니다. 다음과 같이 `-c` 명령과 클래스의 정규화된 이름을 사용하여 `javap`을 실행하면 됩니다.

javap -c java.lang.Boolean

결과는 `에 있는 모든 공개 메소드에 대한 바이트코드입니다. java.lang.Boolean`. 여기서는 `booleanValue` 및 `valueOf(boolean)`에 대한 바이트코드만 복사했습니다.

public boolean booleanValue();
    code:
    0: aload_0
    1: getfield		#7                  // Field value:Z
    4: ireturn
    
public static java.lang.Boolean valueOf(boolean);
    Code:       
    0: iload_0
    1: ifeq          	10
    4: getstatic     	#27                 // Field TRUE:Ljava/lang/Boolean;
    7: goto          	13
    10: getstatic     	#31                 // Field FALSE:Ljava/lang/Boolean;
    13: areturn

바이트코드 분석

얼핏 보면 완전히 새로운 언어를 배울 수 있습니다. 그러나 각 명령어의 기능과 Java가 스택과 함께 작동한다는 사실을 배우면 금방 이해가 됩니다.

`booleanValue`에 대한 3개의 바이트코드 명령을 예로 들어 보겠습니다.

  • `aload_n`은 로컬 변수에 대한 참조를 스택에 배치하는 것을 의미합니다. 클래스 인스턴스에서 `aload_0`은 `this`를 참조합니다.

  • `getfield`는 `this`(스택의 하위 항목)에서 멤버 변수를 읽고 해당 항목을 배치하는 것을 의미합니다. 스택의 값

    • `#7`은 상수 풀의 참조 인덱스를 나타냅니다.

    • `// 필드 값:Z`는 `#7`이 가리키는 것은 `boolean` 유형의 `value`라는 필드(Z)

  • `ireturn`은 기본 값을 팝한다는 의미입니다. 스택에서 꺼내서 반환

간단히 말하면 이 세 가지 명령어는 인스턴스의 `value` 필드를 조회하여 반환합니다.

두 번째 예로, 다음 메소드인 `valueOf(boolean)`:

  • `iload_n`은 기본 지역 변수를 스택에 배치하는 것을 의미합니다. `iload_0`은 첫 번째 메소드 매개변수를 참조합니다(첫 번째 메소드 매개변수는 기본 매개변수이므로)

  • `ifeq    n`은 스택에서 값을 팝하고 그것이 true인지 확인하는 것을 의미합니다. 그렇다면 다음 줄로 진행하고, 그렇지 않으면 `n` 줄로 점프합니다.

  • `getstatic #n`은 정적 멤버를 스택으로 읽는다는 의미입니다

    • `#27`은 상수 풀에 있는 정적 멤버의 인덱스를 나타냅니다.

    • `// 필드 TRUE:Ljava/lang/Boolean`은 `#27`이 무엇을 참조하는지 알려줍니다. , `Boolean

  • `goto n` 유형의 `TRUE`라는 정적 멤버는 이제 바이트코드에서 `n` 줄로 점프한다는 의미입니다

  • `areturn`은 스택에서 참조를 꺼내서 반환한다는 의미입니다

즉, 이 지침에서는 true인 경우 첫 번째 메서드 매개변수를 가져오라고 말합니다. , 그런 다음 `Boolean.TRUE`를 반환합니다. 그렇지 않으면 `Boolean.FALSE`를 반환합니다.

바이트코드 분석 활용

이 기능은 리버스 엔지니어링, 성능 최적화 및 보안 연구에 도움이 될 수 있다고 앞서 언급했습니다. 이제 이에 대해 확장해 보겠습니다.

리버스 엔지니어링

타사 라이브러리나 비공개 소스 구성 요소로 작업할 때 바이트코드 분석은 강력한 도구가 됩니다. 바이트코드를 디컴파일하면 이러한 라이브러리의 내부 작동 방식을 엿볼 수 있어 통합, 문제 해결 및 호환성 보장에 도움이 됩니다.

독점 또는 비공개 소스 Java 코드가 있는 상황에서는 바이트코드를 읽는 것이 유일한 방법일 수 있습니다. 그 기능을 이해하는 방법. 바이트코드 분석을 사용하면 비공개 소스 애플리케이션의 동작을 리버스 엔지니어링하고 이해하여 상호 운용성 또는 사용자 정의를 촉진할 수 있습니다.

실제 사례를 통해 저는 최근 타사 패키지 탱글 분석 도구를 Ci 시스템에 통합하려고 했습니다. 불행하게도 공급업체는 비공개 소스였으며 독점 ​​UI를 통해 라이브러리에 액세스하는 방법에 대한 문서만 가지고 있었습니다. 바이트코드를 분석함으로써 기본 분석 엔진의 예상되는 입력과 출력을 리버스 엔지니어링할 수 있었습니다.


위 내용은 재미와 이익을 위해 Java 바이트코드를 읽는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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

핫 AI 도구

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

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)

뜨거운 주제

해시 맵은 자바에서 내부적으로 어떻게 작동합니까? 해시 맵은 자바에서 내부적으로 어떻게 작동합니까? Jul 15, 2025 am 03:10 AM

해시 맵은 Java의 해시 테이블을 통해 키 값 쌍 스토리지를 구현하며, 그 핵심은 데이터 위치를 빠르게 배치하는 데 있습니다. 1. 먼저 키의 hashcode () 메소드를 사용하여 해시 값을 생성하고 비트 작업을 통해 배열 인덱스로 변환합니다. 2. 다른 객체가 동일한 해시 값을 생성하여 충돌을 일으킬 수 있습니다. 현재 노드는 링크 된 목록의 형태로 장착됩니다. JDK8 후 링크 된 목록이 너무 길고 (기본 길이 8) 효율을 향상시키기 위해 빨간색과 검은 색 트리로 변환됩니다. 3. 사용자 정의 클래스를 키로 사용하는 경우 equals () 및 hashcode () 메소드를 다시 작성해야합니다. 4. 해시 맵은 용량을 동적으로 확장합니다. 요소 수가 용량을 초과하고 하중 계수 (기본 0.75)를 곱하면 확장 및 재사용; 5. 해시 맵은 스레드 안전이 아니며 Multithreaded에서 Concu를 사용해야합니다.

Java의 캐릭터 인코딩 문제를 처리하는 방법은 무엇입니까? Java의 캐릭터 인코딩 문제를 처리하는 방법은 무엇입니까? Jul 13, 2025 am 02:46 AM

Java의 문자 인코딩 문제를 처리하려면 키는 각 단계에서 사용되는 인코딩을 명확하게 지정하는 것입니다. 1. 텍스트를 읽고 쓰는 시점에 항상 인코딩을 지정하고 InputStreamReader 및 OutputStreamWriter를 사용하고 시스템 기본 인코딩에 의존하지 않도록 명시 적 문자 세트를 전달하십시오. 2. 네트워크 경계에서 문자열을 처리 할 때 양쪽 끝이 일관되도록하고 올바른 컨텐츠 유형 헤더를 설정하고 라이브러리와 인코딩을 명시 적으로 지정하십시오. 3. String.getBytes () 및 Newstring (byte [])을주의해서 사용하고 플랫폼 차이로 인한 데이터 손상을 피하기 위해 항상 Standardcharsets.utf_8을 수동으로 지정하십시오. 요컨대,

Java의 비교기 비교기 Java의 비교기 비교기 Jul 13, 2025 am 02:31 AM

Java에서는 비교 가능성이 기본 정렬 규칙을 내부적으로 정의하는 데 사용되며 비교기는 여러 정렬 로직을 외부로 정의하는 데 사용됩니다. 1. 클래스 자체가 구현 한 인터페이스입니다. 비교 () 메소드를 다시 작성하여 자연 순서를 정의합니다. 문자열 또는 정수와 같은 고정되고 가장 일반적으로 사용되는 분류 방법이있는 클래스에 적합합니다. 2. 비교기는 외부 정의 된 기능 인터페이스이며, 동일한 클래스에 여러 정렬 방법이 필요한 상황에 적합한 Compare () 메소드를 통해 구현되며, 클래스 소스 코드를 수정할 수 없거나 정렬 로직이 종종 변경됩니다. 둘의 차이점은 비교할 수있는 분류 논리 만 정의 할 수 있으며 클래스 자체를 수정하면서 비교할 필요가 있다는 것입니다.

Java의지도를 반복하는 방법은 무엇입니까? Java의지도를 반복하는 방법은 무엇입니까? Jul 13, 2025 am 02:54 AM

Java에는지도를 가로 지르는 세 가지 일반적인 방법이 있습니다. 1. Entryset을 사용하여 키와 값을 동시에 얻으십시오. 이는 대부분의 시나리오에 적합합니다. 2. 키즈 또는 값을 사용하여 각각 키 또는 값을 가로 지르십시오. 3. 코드 구조를 단순화하려면 Java8의 foreach를 사용하십시오. Entryset은 모든 키 값 쌍이 포함 된 세트 세트를 반환하고 각 루프는 맵을 가져옵니다. 열 객체는 키와 값에 자주 액세스하기에 적합합니다. 키나 값 만 필요한 경우 각각 Keyset () 또는 values ()를 호출하거나 키를 가로 질러 Map.Get (키)를 통해 값을 얻을 수 있습니다. Java 8은 foreach ((키, 값)-& gt를 사용할 수 있습니다

Java의 '정적'키워드는 무엇입니까? Java의 '정적'키워드는 무엇입니까? Jul 13, 2025 am 02:51 AM

injava, thestatickeywordmeansamembeLongstotheclassitself, nottoinstances.StaticvariablesAresharedAcrossAllInstances 및 OutObjectCreation, 유용한 ForgloBalTrackingorConstants.StaticMethodsOperateateAteAteClassEvel, canceCcessnon-StaticMbers, statice

Windows에서 Java_Home 환경 변수를 설정하는 방법 Windows에서 Java_Home 환경 변수를 설정하는 방법 Jul 18, 2025 am 04:05 AM

TOSETJAVA_HOMEONWINDOWS, FIRSTLOCATETEJDKINSTALLATIONPATH (예 : C : \ ProgramFiles \ java \ jdk-17), thencreateasystemenvaria blenamedjava_homewiththatpath.next, updatePathVariableByadding%java \ _home%\ bin, andverifythesetupusingjava-versionandjavac-v

JDBC로 Java의 거래를 처리하는 방법은 무엇입니까? JDBC로 Java의 거래를 처리하는 방법은 무엇입니까? Aug 02, 2025 pm 12:29 PM

JDBC 트랜잭션을 올바르게 처리하려면 먼저 자동 커밋 모드를 끄고 여러 작업을 수행 한 다음 결과에 따라 커밋 또는 롤백을 수행해야합니다. 1. 트랜잭션을 시작하려면 Conn.SetAutoCommit (False)에게 전화하십시오. 2. 인서트 및 업데이트와 같은 여러 SQL 작업을 실행합니다. 3. 모든 작업이 성공한 경우 Conn.commit ()에게 전화하여 데이터 일관성을 보장하기 위해 예외가 발생하면 Conn.Rollback ()에게 전화하십시오. 동시에, 재 시도는 리소스를 관리하고, 예외를 올바르게 처리하고, 연결 유출을 피하기 위해 긴밀한 연결을 사용하는 데 사용해야합니다. 또한 연결 풀을 사용하고 부분적으로 롤백을 달성하기 위해 저장 포인트를 설정하고 성능을 향상시키기 위해 거래를 가능한 한 짧게 유지하는 것이 좋습니다.

Java 가상 스레드 성능 벤치마킹 Java 가상 스레드 성능 벤치마킹 Jul 21, 2025 am 03:17 AM

가상 스레드는 동시성과 IO 집약적 시나리오에서 상당한 성능 이점을 가지고 있지만 테스트 방법과 해당 시나리오에주의를 기울여야합니다. 1. 정확한 테스트는 실제 비즈니스, 특히 IO 차단 시나리오를 시뮬레이션하고 JMH 또는 Gatling과 같은 도구를 사용하여 플랫폼 스레드를 비교해야합니다. 2. 처리량 간격은 분명하며, 일정이 가볍고 효율적이기 때문에 10 만 동시 요청보다 여러 배에서 10 배나 높을 수 있습니다. 3. 테스트 중에, 높은 동시성 수치를 맹목적으로 추구하고, 비 차단 IO 모델에 적응하고, 대기 시간 및 GC와 같은 모니터링 지표에주의를 기울일 필요가있다. 4. 실제 애플리케이션에서는 웹 백엔드, 비동기 작업 처리 및 많은 동시 IO 시나리오에 적합하지만 CPU 집약적 작업은 플랫폼 스레드 또는 포크 플랫폼에 여전히 적합합니다.

See all articles