本文来说一下,我们开发好的.java文件是源码文件,并不能交给机器直接执行,需要将其变成字节码甚至是机器码文件。那么静态编译器是如何把源码转化成字节码的呢? 下图为.java源码转化为字节码的过程。...当语义分析完成之后,即可以生成字节码。 字节码必须通过类加载过程加载到JVM环境中后,才可以执行。...字节码执行的三种模式 解释执行 JIT编译执行 JIT编译与解释混合执行(主流JVM默认模式) 何为JIT编译?...JIT的作用就是将Java字节码动态低编译成可以直接发送给处理器指令执行的机器码。大致流程如下: ? 注意:解释执行与编译执行在线上环境微妙的辩证关系。...曾经有这样的故障案例:某程序员在发布平台进行分批发布,在输入发布总批次数的时候,误填写成分两批发布。
是利用Java运算符顺序将式子拆解,然后一步步运算,还是其他什么办法?在思索一会儿之后,决定还是通过字节码指令来看看这两行代码是怎么运行的。...将两行代码拷贝到Test.java中,执行以下指令输出字节码: javac Test.java javap -c Test.class 字节码输出结果如下: 如果是之前对字节码没有了解的话,可以去搜一下字节码指令的资料...,或者去《深入理解Java虚拟机》这本书去找附录b 字节码指令表。.../io/PrintStream.println:(Z)V 60: return 从字节码中可以看到a、b、c、d赋值的时候都是通过invokestatic字节码指令调用了Integer.valueOf...但是不同的是,在给a、b赋值时候字节码指令是bipush,是将单字节的整型常量值(-128 - 127)压入操作数栈顶;给c、d赋值时候字节码指令是sipush,是将int类型的常量值压入操作数栈顶。
是利用Java运算符顺序将式子拆解,然后一步步运算,还是其他什么办法? 在思索一会儿之后,决定还是通过字节码指令来看看这两行代码是怎么运行的。...将两行代码拷贝到Test.java中,执行以下指令将Java源代码转换成字节码: javac Test.java javap -c Test.class 字节码输出结果如下: 如果是之前对字节码没有了解的话...,可以去搜一下字节码指令的资料,或者去《深入理解Java虚拟机》这本书去找「附录b 字节码指令表」。.../io/PrintStream.println:(Z)V 60: return 从字节码中可以看到a、b、c、d赋值的时候都是通过「invokestatic」字节码指令调用了Integer.valueOf...但是不同的是,在给a、b赋值时候字节码指令是bipush,是将单字节的整型常量值(-128 - 127)压入操作数栈顶;给c、d赋值时候字节码指令是sipush,是将int类型的常量值压入操作数栈顶。
很多时候,我们都是从代码层面去学习如何编程,却很少去看看一个个 Java 代码背后到底是什么。今天就让我们从一个最简单的 Hello World 开始看一看 Java 的类文件结构。...Java 虚拟机的类文件结构是一组以 8 位字节为基础的二进制流,各数据项目严格按照顺序紧凑地排列在 Class 文件之中,中间没有添加任何分隔符,这使得整个 Class 文件中存储的内容几乎全都是程序需要的数据...如果你够牛逼,你完全可以写一个编译器,将 C 语言代码编译成符合 Java 虚拟机规范的字节码文件,那么 Java 虚拟机也是可以执行的。...准确地说,Java 虚拟机与字节码文件(Class文件)绑定。 Java类文件结构 Java 虚拟机规范中定义了许多规范,其中有一部分定义了字节码的结构和规范。...在 Class 文件中,字符串是使用 ASCII 码进行编码的,我们将这些十六进制字符转换成对应的 ASCII 码之后,其值为:。 第 8 个常量,是一个字符串常量,转换之后是:()V。
于是我们使用 javac 命令得到它的 class 字节码文件: javac Foo.java 字节码文件都是十六进制的字符集合,我们一般可以用 javap 命令来实现反汇编工作。...其实这个文件就是一系列字节码指令的集合,上面 main 方法中的字节码指令我们可以分两部分来看。 第一部分的字节码指令是这样的: ?...点击图片可以看到每一个字节码指令的详细解释 其实这几行的逻辑对应下面这块源码: if(flag){ System.out.println("Hello, Java!")...; } 而第二部分的字节码指令的分析: ?...; } 看完了这两部分的字节码指令,你会发现只有 iload_1 和 iconst_1 字节码指令,而这两个字节码指令是对 int 类型数据的处理。
从字节码到HIR 正如之前看到的,C1的HIR是一个基于静态单赋值的图IR,由基本块构成控制流图,由静态单赋值指令构成基本块,如图8-1所示。...这些可以改变控制流的字节码,将它们标记为leader字节码,并据此划分出基本块的边界,如代码清单8-8所示。...所谓抽象解释是指C1像模板解释器一样,解释执行基本块对应的字节码,并生成对应的SSA指令。解释过程中需要的局部变量和操作数会放到ValueStack,如图8-2所示。...以图8-2所示为例,假设图中所示是一个基本块,包含了左边的字节码。C1解释执行字节码,并将状态放到ValueStack中。状态包括存放局部变量与函数入参的local和存放临时计算结果的stack。...本文给大家讲解的内容是深入解析java虚拟机:C1编译器,从字节码到HIR 下篇文章给大家讲解的是深入解析java虚拟机:C1编译器,HIR代码优化; 觉得文章不错的朋友可以转发此文关注小编; 感谢大家的支持
方案设计 我们首先要清除代码混淆要实现什么,就是将原代码名称结构和内容使用一系列的规则码替换...artifactId> 9.0 名称混淆 名称混淆指的是把类名,方法名,参数名,变量名等定义的名称进行规则码替换...AnnotationVisitor对象,调用super方法后返回自定义AnnotationVisitor对象递归处理即可 混淆规则 无论混淆哪一部分,我们总是要根据一个名称例如abc混淆后得到一个固定的规则码例如...123 这时候我们会想到md5这种固定输入对应固定输出的信息摘要算法 md5内容太长,我们需要截取某几位进行简化 简化后的规则码在待混淆内容越多时越容易碰撞,需要需要动态调整,简单递归即可,最坏结果就是完整的...注意事项 开发事项 在混淆过程中,需要针对不同的情况进行细节处理,举例如下 混淆名称中有相同部分的优先排序替换长度最长的部分 例如方法名HandleMethod和Handle两部分,Handle对应的规则码为
1 自定义ClassLoader类:MemoryClassLoader public class MemoryClassLoader extends URLC...
下文将从字节码的角度,分析Java中基本类型传参和对象传参。 基本类型传参 以下是处理类Porcess,代码应该已经能够自解释了。...上文已经得到结论,我们从JVM的字节码的角度看一下过程是怎么样的。 首先大致JVM的基本结构,对基本类型,和对象存放的位置有一个大致的了解。下图是JVM的基本组件图。...介绍几个基本的组件 程序计数器: 存储每个线程下一步将执行的JVM指令。...使用javap对字节码进行反编译 基本类型传参字节码 以下是TestPrimitive类在执行function3时的字节码。...从主函数的字节码中可以看到,它的值保存的还是第10行,通过istore_2保存到局部变量第2个索引处的18.
结合这两条指令,我们就可以实现把C语言的循环指令,例如for, while编译成对应的jvm字节码。...完成本节代码后,我们可以把下面的C语言代码编译成java字节码,使之能在jvm上正常运行: void main () { int a[3]; int b[3]; int i;...java字节码,首先需要做的是让数组在定义的时候就直接编译成字节码,而不是当数组被赋值的时候才编译成字节码,也就是说编译器一旦读取到代码int a[3]; 就必须用jvm指令生成构造数组的字节码了,而不能等到读取...从运行结果可以看出,在虚拟机上运行的字节码确实与原来C语言的目的一样,把数组b中的内容赋值并打印出来了。...有了循环指令的编译实现后,我们就可以完成最终章,把实现快速排序的C语言程序全部编译成java字节码,当完成这个内容后,我们整个历时将近两年的java开发编译器课程就将画上完美的句号。
十进制转十六进制的具体方法: 对于整数部分,用被除数反复除以16,除第一次外,每次除以16均取前一次商的整数部分作被除数并依次记下每次的余数。另外,所得到的商的最后一位余数是所求二进制数的最高位。...10进制数转换成十六进制数,这是一个连续除以16的过程:把要转换的数,除以16,得到商和余数,将商继续除以16,直到商为0。最后将所有余数倒序排列,得到数就是转换结果。...这个题应该有好几种方式,这里列出一种 #include int main(){ char b[17]={"0123456789ABCDEF"}; int c[64],d...,i=0,base=16; long n; printf("输入一个数;\n"); scanf("%ld",&n); do{c[i]=n%base;i++;n=n/base...=0); printf("转换成新的数;\n"); for(--i;i>=0;--i) {d=c[i]; printf("%c",b[d]); } }
在本节,我们将用C语言开发快速排序算法,然后利用我们的编译器把它编译成java字节码,让C语言编写的快速排序算法能在java虚拟机上顺利执行,完成本节内容后,编译器可以正确的将下列代码编译成java字节码...原来我们实现函数的编译时,编译器会解读代码,直到函数第一次被调用时,才会把被调函数编译成字节码,但这里,被调函数在执行时会调用它自己,如果对原来的逻辑不加处理,那么编译器会反复的为quicksort函数生成代码...,然后判断这个名字是否被记录过,如果前面有过记录,那么这次进入表明函数发生了递归调用,于是就不再执行函数对应的执行树,如果函数是第一次被调用,那么就执行函数对应的执行树,在执行过程中就可以把函数编译成字节码...上面代码完成后,运行编译器,给定的C语言代码编译出的java汇编代码如下: .class public CSourceToJava .super java/lang/Object .method public...invokestatic CSourceToJava/quicksort([III)V branch1: return .end method .end class 上面的代码转换成jvm字节码运行后结果如下
Windows端的java程序使用jni调用C++编写的库,原来实现过在Android和Linux端通过JNI调用C++程序,在Windows端没有实现过,这里记录下几个关键的点; 1、64位的dll工程...,现在少有32位的平台,所以需要通过VisualStudio编译出64位的dll,注意属性页->C/C++->代码生成/运行库/选择多线程调试(/MTd),参考Linux编译选项的静态链接和动态链接的思路就比较好理解了...两者的区别在于,静态链接将程序所依赖的运行库集成到了可执行文件中,可执行文件运行时不再需要运行库;动态链接没有把程序所依赖的运行库集成到可执行文件中,可执行文件运行时需要运行库。 ...推荐选择/MTd, 这样Java程序就不需要重复链接一些依赖的三方库,或者自己写的静态库;我们实现的场景就是通过一个dll工程封装多个lib库的工程; 2、注意Eclipse工程搜索路径的建立:参考https
2、class 字节码文件介绍 搞清楚了Java代码的跨平台原理,我们接着来介绍为什么编写的Java代码能够被计算机所识别。...Java所有的指令大概有 200 个左右,一个字节(8位)可以存储 256 种不同的信息,我们将一个这样的字节称为字节码(ByteCode)。 ...Class 文件结构中,只有常量池的容量是从 1 开始的,其它的集合类型,都是从 0 开始的。 看上图的十六进制文件,常量池容量计数值为:0x0025,即十进制 37。...需要说明的是,Java代码在进行javac 编译的时候,并不像 C 和 C++ 那样有“连接”这一步骤,而是在虚拟机加载 Class 文件的时候进行动态连接。 ...整个十六进制字节码就不一一进行推导了,下面是各个数据类型的结构: ? ?
比如,Windows 下使用 C 语言编写的程序编译连接后可以生成一个 .exe 的可执行程序,生成的这个可执行程序就是一个二进制程序。那么,这个程序如何用二进制编写呢?...在 Windows 下的可执行程序是 PE 格式的,那么就要了解 PE 格式的数据结构,和 CPU 的机器码;在安卓下的可执行程序中,其格式是 DEX 格式,那么就要了解 DEX 格式的数据结构,以及安卓虚拟机的字节码...(这个字节码不是 CPU 的机器码,DEX 的字节码最终被虚拟机解释成机器码,因此手写 DEX 文件时了解 DEX 格式和其字节码即可),同样的,Java 编译的 Class 文件也和安卓相同,因为它也是基于虚拟机执行的文件...因此,在内存中查看数据时,更多的是使用十六进制,其实从本质上十六进制和二进制是没有区别的,只是表示的方式不同。因此,真正使用二进制来写程序时,是使用十六进制来完成的。...那么,在使用十六进制来编写 Windows 下的可执行程序时,首先需要使用十六进制编辑器构造 PE 文件结构,PE 文件结构主要告诉操作系统,程序加载入内存后,程序的映射起始地址是多少,程序的入口地址是多少
文章目录 一、 Android Studio 模块准备 1、 创建 Android 模块 2、 定义测试类 3、 添加依赖 4、 编译 Android 模块拿到字节码文件 5、 拷贝字节码到 Unity...Studio 打开 Unity 导出的 Android 工程 ) 博客中将 Unity 项目导出为了 Android 项目 , 并在 Android Studio 中编译并运行了该项目 ; 使用的 C#...脚本 , 是在 【Unity3D】Unity 游戏画面帧更新 ( 游戏物体 GameObject 移动 | 借助 Time.deltaTime 进行匀速运动 ) 系列博客中编写的脚本 ; 一、 Android...implementation project(':unityLibrary') implementation project(':mylibrary') } 4、 编译 Android 模块拿到字节码文件...; jar 包路径为: 工程根目录\mylibrary\build\intermediates\compile_library_classes_jar\debug\classes.jar 5、 拷贝字节码到
怎样学习才能从一名Java初级程序员成长为一名合格的架构师,或者说一名合格的架构师应该有怎样的技术知识体系,这是不仅一个刚刚踏入职场的初级程序员也是工作三五年之后开始迷茫的老程序员经常会问到的问题。...一、 JAVA。要想成为JAVA(高级)专家肯定要学习JAVA。一般的程序员或许只需知道一些JAVA的语法结构就可以应付了。但要成为JAVA(高级) 专家,您要对JAVA做比较深入的研究。...如何写出好的代 码往往要借助一些设计模式。当然长期的代码经验积累,只要您用心,会使您形成自己代码风格。相信您的代码也比较符合代码的可重用性,可维护性,可扩展性。...其实真正比较大的项目都是有人专门做数据库的,但往往很多项目要求作为(高级)工程师的您也参与数据库的设计以及SQL的编写。...与在《Java程序员的未来发展之路,你是否感到迷茫?》
根据操作码的长度不同,指令分为单字节操作码指令、双字节操作码指令、三字节操作码指令。...下面是x86架构的CPU指令操作码表: CPU中的指令译码模块拿到手一看,呀,不是指令前缀,是个单字节操作码的mov指令,要往eax寄存器里面塞数据,数据从哪来呢?...C/C++语言编译的程序,最后是直接编译成了CPU的指令,所以跨平台能力差,如果换到ARM架构平台,原来的程序将无法执行,需要重新编译成新的平台的程序。...而Java、Python这类语言,是自己在软件层面的指令集,因为其自身已经开发了针对不同CPU平台的虚拟机、解释器,所以这些语言编写的程序移植性好,真正做到一次编写,到处运行。...总结 我们使用高级语言C、C++编写的程序代码,经过编译器的编译链接,最终变成CPU可以理解的机器指令,随后CPU在执行时通过不断的译码、执行,最终实现高级语言所描述的功能。
从事编程十几年,JAVA、C、C++、Python这四种编程语言都玩过,前三种玩的比较多,python做为兴趣爱好或者玩脚本的时候弄过,编程语言在使用的时候主要还是适合不合适,单片机使用的场景属于功能简单...,成本相对较低,现在也有高配版的单片机,本来单片机是不带系统的,非要硬扯到带系统的,从成本上考量就不是那回事了,成本主要包括两个方面有硬件成本,还有研发难度的成本这都是需要考虑的事情。...单片机开发语言C语言和汇编,别的语言还真不好上手,编程语言能力强大与否关键无外乎是不是能够控制底层的硬件,往往是一些门外汉喜欢编排一些故事,搞成什么编程语言的鄙视链条,真正的程序员底层的编程语言会羡慕上层编程语言的灵活性...,上层的编程语言羡慕底层的编程人员能够控制硬件觉得是一件很神奇的事情,曾经面试过一个做了很多年java的工程师,居然想着拿出两年的时间来学习C语言,最后真还是降低工资标准进来了公司。...,就单片机那点内存压根不是个,所以不同的编程环境不同的搭配,没有必要强求,无论是C/C++编程还是高级语言java,python在使用过程中编程套路都接近 如果真要想在单片机上运行虚拟机还不如直接找个有操作系统的嵌入式设备
当前编译器已经能够把很多C语言的源程序编译成可以在java虚拟机上运行的字节码,但一直存在一个问题是,编译出的字节码存有冗余语句,例如赋值语句: a = 1; 它编译成java字节码后情况如下: aload...经过上面的修改后,在生成java字节码时,就不再会有冗余语句了。现在我们看看,如何把if else 这些分支控制语句转换为java字节码。...branch0: sipush 2 istore 0 out_branch0: sipush 3 istore 0 要比较1和2大小,先要把两个数值压到堆栈上,C语言代码使用的是1字节码时...字节码时怎样的,运行修改代码后的编译器,然后输入上面C语言代码,得到的编译结果如下: .class public CSourceToJava .super java/lang/Object .method...把编译出来的java汇编转换成二进制字节码运行后结果如下: ? 从结果上看,打印出来的b的值是5,由此可见我们编译输出的结果应该是正确的。
领取专属 10元无门槛券
手把手带您无忧上云