前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java命令之javap初探

Java命令之javap初探

作者头像
haifeiWu
发布2018-09-11 10:18:44
6700
发布2018-09-11 10:18:44
举报
文章被收录于专栏:haifeiWu与他朋友们的专栏

javap是jdk自带的一个工具在jdk安装目录的/bin下面可以找到,可以对代码反编译,也可以查看java编译器生成的字节码,对代码的执行过程进行分析,了解jvm内部的工作。

下面列举javap命令的常用options及其功能描述,更多功能的使用请自行Google,楼主不做赘述。

用法摘要

代码语言:javascript
复制
-help 帮助
-l 输出行和变量的表
-public 只输出public方法和域
-protected 只输出public和protected类和成员
-package 只输出包,public和protected类和成员,这是默认的
-p -private 输出所有类和成员
-s 输出内部类型签名
-c 输出分解后的代码,例如,类中每一个方法内,包含java字节码的指令,
-verbose 输出栈大小,方法参数的个数
-constants 输出静态final常量

实例分析

javap命令分解一个class文件,它根据options来决定到底输出什么。如果没有使用options,那么javap将会输出该class文件中的包,类里的protected和public域以及类里的所有方法。javap将会把它们输出在标准输出上。来看这个例子,先编译(javac)下面这个类。

代码语言:javascript
复制
package com.thundersoft.metadata.test.kafka;

import org.apache.kafka.clients.consumer.ConsumerRecord;
import org.apache.kafka.clients.consumer.ConsumerRecords;
import org.apache.kafka.clients.consumer.KafkaConsumer;
import org.apache.kafka.clients.producer.KafkaProducer;
import org.apache.kafka.clients.producer.Producer;
import org.apache.kafka.clients.producer.ProducerRecord;
import org.junit.Test;

import java.util.Arrays;
import java.util.Properties;

public class KafkaTest {

    @Test
    public void testProducer() {
        Properties props = new Properties();
        props.put("bootstrap.servers", "192.168.204.30:9092");
        props.put("acks", "all");
        props.put("retries", 0);
        props.put("batch.size", 16384);
        props.put("linger.ms", 1);
        props.put("buffer.memory", 33554432);
        props.put("key.serializer", "org.apache.kafka.common.serialization.StringSerializer");
        props.put("value.serializer", "org.apache.kafka.common.serialization.StringSerializer");

        Producer<String, String> producer = new KafkaProducer<>(props);
        for(int i = 0; i < 100; i++) {
            producer.send(new ProducerRecord<String, String>("my-topic", Integer.toString(i), Integer.toString(i)));
        }

        producer.close();
    }

    @Test
    public void testKafkaConsumer() {
        Properties props = new Properties();
        props.put("bootstrap.servers", "192.168.204.30:9092");
        props.put("group.id", "test");
        props.put("enable.auto.commit", "true");
        props.put("auto.commit.interval.ms", "1000");
        props.put("key.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");
        props.put("value.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");
        KafkaConsumer<String, String> consumer = new KafkaConsumer<>(props);
        consumer.subscribe(Arrays.asList("my-topic"));

        while (true) {
            ConsumerRecords<String, String> records = consumer.poll(100);
            for (ConsumerRecord<String, String> record : records)
                System.out.printf("offset = %s, key = %s, value = %s%n", record.topic(), record.key(), record.value());
        }
    }

    public static void main(String[] args) {
        int a = 2;
        int b = 3;
        int sum = a*b;
        System.out.println(sum);
    }

}

在命令行上键入javap KafkaTest后,输出结果如下

代码语言:javascript
复制
public class com.thundersoft.metadata.test.kafka.KafkaTest {
  public com.thundersoft.metadata.test.kafka.KafkaTest();
  public void testProducer();
  public void testKafkaConsumer();
  public static void main(java.lang.String[]);
}

结合代码分析编译器执行过程

这里只关注main方法内部的代码逻辑,main方法代码如下

代码语言:javascript
复制
public static void main(String[] args) {
       int a = 2;
       int b = 3;
       int sum = a*b;
       System.out.println(sum);
   }

在命令行上键入javap -c KafkaTest后,输出结果如下

代码语言:javascript
复制
public static void main(java.lang.String[]);
   Code:
      0: iconst_2
      1: istore_1
      2: iconst_3
      3: istore_2
      4: iload_1
      5: iload_2
      6: imul
      7: istore_3
      8: getstatic     #47                 // Field java/lang/System.out:Ljava/io/PrintStream;
     11: iload_3
     12: invokevirtual #54                 // Method java/io/PrintStream.println:(I)V
     15: return

如上面代码所,iconst_2 与iconst_3分别代表常量2,3 。istore_1 ,istore_2 分别代表定义两个普通变量,iload_1 , iload_2 分别表示加载istore_1,istore_2 两个变量到数据栈中,imul表示两个变量做乘法运算,结果赋值给变量istore_3,最后将结果输出,程序返回。

在分析这段简单代码的过程中,楼主发现了一个jvm编译命令的网站,分享出来jvm指令

总结

楼主在上面做了一个简单的代码分析的过程,希望可以帮助到有缘人。javap可以用于反编译和查看编译器编译后的字节码。一般用到的不多,不过平时用javap -c比较多,该命令用于列出每个方法所执行的JVM指令,用来解决比较棘手的逻辑出错的bug是个不错的选择。另外通过字节码和源代码的对比,深入分析java的编译原理及代码执行过程,解决各种Java原理级别的问题。

作 者:haifeiWu

原文链接:https://cloud.tencent.com/developer/article/1333326

版权声明:非特殊声明均为本站原创作品,转载时请注明作者和原文链接。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 用法摘要
  • 实例分析
    • 结合代码分析编译器执行过程
    • 总结
    相关产品与服务
    腾讯云代码分析
    腾讯云代码分析(内部代号CodeDog)是集众多代码分析工具的云原生、分布式、高性能的代码综合分析跟踪管理平台,其主要功能是持续跟踪分析代码,观测项目代码质量,助力维护团队卓越代码文化。
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档