Java 虚拟机指令(操作码)集

Java 虚拟机指令(操作码)集

D瓜哥
最近在研究 Java 虚拟机字节码。在 《Java虚拟机规范》 看到一个整理完整的 Java 虚拟机指令集(也叫操作码)列表。转载过来,方便查阅。 关于 Java 虚拟机指令(操作码),准备写一个“探秘”系列: Java 虚拟机操作码探秘:常量指令 — 重点介绍一下关于“常量”指令。 分类 操作码 助记符 指令含义 常量 0 0x00 nop 什么都不做 1 0x01 aconst_null 将 null 推送至栈顶 2 0x02 iconst_m1 将 int 类型 -1 推送至栈顶 3 0x03 iconst_0 将 int 类型 0 推送至栈顶 4 0x04 iconst_1 将 int 类型 1 推送至栈顶 5 0x05 iconst_2 将 int 类型 2 推送至栈顶 6 0x06 iconst_3 将 int 类型 3 推送至栈顶 7 0x07 iconst_4 将 int 类型 4 推送至栈顶 8 0x08 iconst_5 将 int 类型 5 推送至栈顶 9 0x09 lconst_0 将 long 类型 0 推送至栈顶 10 0x0a lconst_1 将 long 类型 1 推送至栈顶 11 0x0b fconst_0 将 float 类型 0 推送至栈顶 12 0x0c fconst_1 将 float 类型 1 推送至栈顶 13 0x0d fconst_2 将 float 类型 2 推送至栈顶 14 0x0e dconst_0 将 double 类型 0 推送至栈顶 15 0x0f dconst_1 将 double 类型 1 推送至栈顶 16 0x10 bipush 将单字节的常量值(-128 ~ 127)推送至栈顶 17 0x11 sipush 将一个短整类型常量值(-32,768 ~ 32,767)推送栈顶 18 0x12 ldc 将 int、 float 或 String 类型常量值从常量池中推送至栈顶 19 0x13 ldc_w 将int、 float 或 String 类型常量值从常量池中推送栈顶(宽索引) 20 0x14 ldc2_w 将 long 或 double 类型常量值从常量池中推送至栈(宽索引) 加载 21 0x15 iload 将指定的 int 类型本地变量推送至栈顶 22 0x16 lload 将指定的 long 类型本地变量推送至栈顶 23 0x17 fload 将指定的 float 类型本地变量推送至栈顶 24 0x18 dload 将指定的 double 类型本地变量推送至栈顶 25 0x19 aload 将指定的引用类型本地变量推送至栈顶 26 0x1a iload_0 将第 1 个 int 类型本地变量推送至栈顶 27 0x1b iload_1 将第 2 个 int 类型本地变量推送至栈顶 28 0x1c iload_2 将第 3 个 int 类型本地变量推送至栈顶 29 0x1d iload_3 将第 4 个 int 类型本地变量推送至栈顶 30 0x1e lload_0 将第 1 个 long 类型本地变量推送至栈顶 31 0x1f lload_1 将第 2 个 long 类型本地变量推送至栈顶 32 0x20 lload_2 将第 3 个 long 类型本地变量推送至栈顶 33 0x21 lload_3 将第 4 个 long 类型本地变量推送至栈顶 34 0x22 fload_0 将第 1 个 float 类型本地变量推送至栈顶 35 0x23 fload_1 将第 2 个 float 类型本地变量推送至栈顶 36 0x24 fload_2 将第 3 个 float 类型本地变量推送至栈顶 37 0x25 fload_3 将第 4 个 float 类型本地变量推送至栈顶 38 0x26 dload_0 将第 1 个 double 类型本地变量推送至栈顶 39 0x27 dload_1 将第 2 个 double 类型本地变量推送至栈顶 40 0x28 dload_2 将第 3 个 double 类型本地变量推送至栈顶 41 0x29 dload_3 将第 4 个 double 类型本地变量推送至栈顶 42 0x2a aload_0 将第 1 个引用类型本地变量推送至栈顶 43 0x2b aload_1 将第 2 个引用类型本地变量推送至栈顶 44 0x2c aload_2 将第 3 个引用类型本地变量推送至栈顶 45 0x2d aload_3 将第 4 个引用类型本地变量推送至栈顶 46 0x2e iaload 将 int 类型数组的指定元素推送至栈顶 47 0x2f laload 将 long 类型数组的指定元素推送至栈顶 48 0x30 faload 将 float 类型数组的指定元素推送至栈顶 49 0x31 daload 将 double 类型数组的指定元素推送至栈顶 50 0x32 aaload 将引用类型数组的指定元素推送至栈顶 51 0x33 baload 将 boolean 或 byte 类型数组的指定元素推送至栈顶 52 0x34 caload 将 char 类型数组的指定元素推送至栈顶 53 0x35 saload 将 short 类型数组的指定元素推送至栈顶 存储 54 0x36 istore 将栈顶 int 类型数值存入指定本地变量 55 0x37 lstore 将栈顶 long 类型数值存入指定本地变量 56 0x38 fstore 将栈顶 float 类型数值存入指定本地变量 57 0x39 dstore 将栈顶 double 类型数值存入指定本地变量 58 0x3a astore 将栈顶引用类型数值存入指定本地变量 59 0x3b istore_0 将栈顶 int 类型数值存入第 1 个本地变量 60 0x3c istore_1 将栈顶 int 类型数值存入第 2 个本地变量 61 0x3d istore_2 将栈顶 int 类型数值存入第 3 个本地变量 62 0x3e istore_3 将栈顶 int 类型数值存入第 4 个本地变量 63 0x3f lstore_0 将栈顶 long 类型数值存入第 1 个本地变量 64 0x40 lstore_1 将栈顶 long 类型数值存入第 2 个本地变量 65 0x41 lstore_2 将栈顶 long 类型数值存入第 3 个本地变量 66 0x42 lstore_3 将栈顶 long 类型数值存入第 4 个本地变量 67 0x43 fstore_0 将栈顶 float 类型数值存入第 1 个本地变量 68 0x44 fstore_1 将栈顶 float 类型数值存入第 2 个本地变量 69 0x45 fstore_2 将栈顶 float 类型数值存入第 3 个本地变量 70 0x46 fstore_3 将栈顶 float 类型数值存入第 4 个本地变量 71 0x47 dstore_0 将栈顶 double 类型数值存入第 1 个本地变量 72 0x48 dstore_1 将栈顶 double 类型数值存入第 2 个本地变量 73 0x49 dstore_2 将栈顶 double 类型数值存入第 3 个本地变量 74 0x4a dstore_3 将栈顶 double 类型数值存入第 4 个本地变量 75 0x4b astore_0 将栈顶引用类型数值存入第 1 个本地变量 76 0x4c astore_1 将栈顶引用类型数值存入第 2 个本地变量 77 0x4d astore_2 将栈顶引用类型数值存入第 3 个本地变量 78 0x4e astore_3 将栈顶引用类型数值存入第 4 个本地变量 79 0x4f iastore 将栈顶 int 类型数值存入指定数组的指定索引位置 80 0x50 lastore 将栈顶 long 类型数值存入指定数组的指定索引位置 81 0x51 fastore 将栈顶 float 类型数值存入指定数组的指定索引位置 82 0x52 dastore 将栈顶 double 类型数值存入指定数组的指定索引位置 83 0x53 aastore 将栈顶引用类型数值存入指定数组的指定索引位置 84 0x54 bastore 将栈顶 boolean 或 byte 类型数值存入指定数组的指定索引位置 85 0x55 castore 将栈顶 char 类型数值存入指定数组的指定索引位置 86 0x56 sastore 将栈顶 short 类型数值存入指定数组的指定索引位置 栈 87 0x57 pop 将栈顶数值弹出(数值不能是 long 或 double 类型的) 88 0x58 pop2 将栈顶的一个 long 或 double 类型的数值或两个其他类型的数值弹出 89 0x59 dup 复制栈顶数值并将复制值压入栈顶 90 0x5a dup_x1 复制栈顶值并将其插入栈顶那两个值的下面 91 0x5b dup_x2 复制栈顶值并将其插入栈顶那两个或三个值的下面 92 0x5c dup2 复制栈顶的一个 long 或 double 类型的值,或两个其他类型的值,并将其压入栈顶 93 0x5d dup2_x1 复制栈顶的一个或两个值,并将其插入栈顶那两个或三个值的下面 94 0x5e dup2_x2 复制栈顶的一个或两个值,并将其插入栈顶那两个、三个或四个值的下面 95 0x5f swap 将栈顶的两个数值互换(数值不能是 long 或 double 类型的) 数学 96 0x60 iadd 将栈顶两 int 类型数值相加并将结果压入栈顶 97 0x61 ladd 将栈顶两 1ong 类型数值相加并将结果压入栈顶 98 0x62 fadd 将栈顶两 float 类型数值相加并将结果压入栈顶 99 0x63 dadd 将栈顶两 double 类型数值相加并将结果压入栈顶 100 0x64 isub 将栈顶两 int 类型数值相减并将结果压入栈顶 101 0x65 lsub 将栈顶两 long 类型数值相减并将结果压入栈顶 102 0x66 fsub 将栈顶两 float 类型数值相减并将结果压入栈顶 103 0x67 dsub 将栈顶两 double 类型数值相减并将结果压入栈顶 104 0x68 imul 将栈顶两 int 类型数值相乘并将结果压入栈顶 105 0x69 lmul 将栈顶两 long 类型数值相乘并将结果压入栈顶 106 0x6a fmul 将栈顶两 float 类型数值相乘并将结果压入栈顶 107 0x6b dmul 将栈顶两 double 类型数值相乘并将结果压入栈顶 108 0x6с idiv 将栈顶两 int 类型数值相除并将结果压入栈顶 109 0x6d ldiv 将栈顶两 long 类型数值相除并将结果压入栈顶 110 0x6e fdiv 将栈顶两 float 类型数值相除并将结果压入栈顶 111 0x6f ddiv 将栈顶两 double 类型数值相除并将结果压入栈顶 112 0x70 irem 将栈顶两 int 类型数值作取模运算并将结果压入栈顶 113 0x71 lrem 将栈顶两 long 类型数值作取模运算并将结果压入栈顶 114 0x72 frem 将栈顶两 float 类型数值作取模运算并将结果压入栈顶 115 0x73 drem 将栈顶两 double 类型数值作取模运算并将结果压入栈顶 116 0x74 ineg 将栈顶 int 类型数值取负并将结果压入栈顶 117 0x75 lneg 将栈顶 long 类型数值取负并将结果压入栈顶 118 0x76 fneg 将栈顶 float 类型数值取负并将结果压入栈顶 119 0x77 dneg 将栈顶 double 类型数值取负并将结果压入栈顶 120 0x78 ishl 将 int 类型数值左移位指定位数并将结果压入栈顶 121 0x79 lshl 将 long 类型数值左移位指定位数并将结果压入栈顶 122 0x7a ishr 将 int 类型数值(有符号)右移位指定位数并将结果压入栈顶 123 0x7b lshr 将 long 类型数值(有符号)右移位指定位数并将结果压入栈顶 124 0x7c iushr 将 int 类型数值(无符号)右移位指定位数并将结果压入栈顶 125 0x7d lushr 将 long 类型数值(无符号)右移位指定位数并将结果压入栈顶 126 0x7e iand 将栈顶两 int 类型数值作“按位与”并将结果压入栈顶 127 0x7f land 将栈顶两 long 类型数值作“按位与”并将结果压入栈顶 128 0x80 ior 将栈顶两 int 类型数值作“按位或”并将结果压入栈顶 129 0x81 lor 将栈顶两 long 类型数值作“按位或”并将结果压入栈顶 130 0x82 ixor 将栈顶两 int 类型数值作“按位异或”并将结果压入栈顶 131 0x83 lxor 将栈顶两 long 类型数值作“按位异或”并将结果压入栈顶 132 0x84 iinc 将指定 int 类型变量增加指定值(i++,i--,i += 2) 转换 133 0x85 i2l 将栈顶 int 类型数值强制转换成 long 类型数值并将结果压入栈顶 134 0x86 i2f 将栈顶 int 类型数值强制转换成 float 类型数值并将结果压入栈顶 135 0x87 i2d 将栈顶 int 类型数值强制转换成 double 类型数值并将结果压入栈顶 136 0x88 l2i 将栈顶 long 类型数值强制转换成 int 类型数值并将结果压入栈顶 137 0x89 l2f 将栈顶 long 类型数值强制转换成 float 类型数值并将结果压入栈顶 138 0x8a l2d 将栈顶 long 类型数值强制转换成 double 类型数值并将结果压入栈顶 139 0x8b f2i 将栈顶 float 类型数值强制转换成 int 类型数值并将结果压入栈顶 140 0x8c f2l 将栈顶 float 类型数值强制转换成 long 类型数值并将结果压入栈顶 141 0x8d f2d 将栈顶 float 类型数值强制转换成 double 类型数值并将结果压入栈顶 142 0x8e d2i 将栈顶 double 类型数值强制转换成 int 类型数值并将结果压入栈顶 143 0x8f d2l 将栈顶 double 类型数值强制转换成 long 类型数值并将结果压入栈顶 144 0x90 d2f 将栈顶 double 类型数值强制转换成 float 类型数值并将结果压入栈顶 145 0x91 i2b 将栈顶 int 类型数值强制转换成 byte 类型数值并将结果压入栈顶 146 0x92 i2c 将栈顶 int 类型数值强制转换成 char 类型数值并将结果压入栈顶 147 0x93 i2s 将栈顶 int 类型数值强制转换成 short 类型数值并将结果压入栈顶 比较 148 0x94 lcmp 比较栈顶两 long 类型数值大小,并将结果(1,0,-1)压入栈顶 149 0x95 fcmpl 比较栈顶两 float 类型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为“NaN”时,将 -1 压入栈顶 150 0x96 fcmpg 比较栈顶两 float 类型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为“NaN”时,将1压入栈顶 151 0x97 dcmpl 比较栈顶两 double 类型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为“NaN”时,将-1压入栈顶 152 0x98 dcmpg 比较栈顶两 double 类型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为“NaN”时,将1压入栈顶 153 0x99 ifeq 当栈顶 int 类型数值等于 0 时跳转 154 0x9a ifne 当栈顶 int 类型数值不等于 0 时跳转 155 0x9b iflt 当栈顶 int 类型数值小于 0 时跳转 156 0x9c ifge 当栈顶 int 类型数值大于等于 0 时跳转 157 0x9d ifgt 当栈顶 int 类型数值大于 0 时跳转 158 0x9e ifle 当栈顶 int 类型数值小于等于 0 时跳转 159 0x9f if_icmpeq 比较栈顶两 int 类型数值大小,当前者等于后者时跳转 160 0xa0 if_icmpne 比较栈顶两 int 类型数值大小,当前者不等于后者时跳转 161 0xa1 if_icmplt 比较栈顶两 int 类型数值大小,当前者小于后者时跳转 162 0xa2 if_icmpge 比较栈顶两 int 类型数值大小,当前者大于等于后者时跳转 163 0xa3 if_icmpgt 比较栈顶两 int 类型数值大小,当前者大于后者时跳转 164 0xa4 if_icmple 比较栈顶两 int 类型数值大小,当前者小于等于后者时跳转 165 0xa5 if_acmpeq 比较栈顶两引用类型数值,当结果相等时跳转 166 0xa6 ifacmpne 比较栈顶两引用类型数值,当结果不相等时跳转 控制 167 0xa7 goto 无条件跳转 168 0xa8 jsr 跳转至指定 16 位 offset 位置,并将 jsr 下一条指令地址压入栈顶 169 0xa9 ret 返回至由指定的局部变量所给出的指令位置(一般与 jsr、jsr_w 联合使用) 170 0xaa tableswitch 用于 switch 条件跳转,case 值连续(变长指令) 171 0xab lookupswitch 用于 switch 条件跳转,case 值不连续(变长指令) 172 0xac ireturn 从当前方法返回 int 173 Oxad lreturn 从当前方法返回 long 174 0xae freturn 从当前方法返回 float 175 0xaf dreturn 从当前方法返回 double 176 0xb0 areturn 从当前方法返回对象引用 177 0xb1 return 从当前方法返回void 引用 178 0xb2 getstatic 获取指定类的静态字段,并将其值压入栈顶 179 0xb3 putstatic 为指定类的静态字段赋值 180 0xb4 getfield 获取指定类的实例字段,并将其值压入栈顶 181 0xb5 putfield 为指定类的实例字段赋值 182 0xb6 invokevirtual 调用实例方法 183 0xb7 invokespecial 调用父类方法、实例初始化方法、私有方法 184 0xb8 invokestatic 调用静态方法 185 0xb9 invokeinterface 调用接口方法 186 0xba invokedynamic 调用动态链接方法 187 0xbb new 创建一个对象,并将其引用值压入栈顶 188 0xbc newarray 创建一个指定原始类型(如int、float 、char等)的数组,并将其引用值压入栈顶 189 0xbd anewarray 创建一个引用型(如类、接口、数组)的数组,并将其引用值压入栈顶 190 0xbe arraylength 获得数组的长度值并压入栈顶 191 0xbf athrow 将栈顶的异常抛出 192 0xcO checkcast 检验类型转换,检验未通过将抛出 ClassCastException 193 0xc1 instanceof 检验对象是否是指定类的实例。如果是,就将 1 压入栈顶,否则将 0 压入栈顶 194 0xc2 monitorenter 获得对象的锁,用于实现同步块 195 0xc3 monitorexit 释放对象的锁,用于实现同步块 扩展 196 0xc4 wide 扩展本地变量索引的宽度 197 0xс5 multianewarray 创建指定类型和指定维度的多维数组(执行该指令时,操作栈中必须包含各维度的长度值),并将其引用值压入栈顶 198 0xc6 ifnull 为nu11时跳转 199 0xc7 ifnonnull 不为nu11时跳转 200 0xc8 goto_w 无条件跳转(宽索引) 201 0xc9 jsr_w 跳转至指定 32 位 offset 位置,并将 jsr_w 下一条指令地址压入栈顶 保留指令 202 Оxca breakpoint 调试时的断点标记 254 Oxfe impdep1 为特定软件面预留的语言后门 255 0xff impdep2 为特定硬件面预留的语言后门 参考资料 Chapter 6. The Java Virtual Machine Instruction Set Java bytecode List of Java bytecode instructions
关于升级 Spring 等依赖的一些经验

关于升级 Spring 等依赖的一些经验

D瓜哥
到公司后,熟悉了一些项目后,发现大部分项目的依赖都比较陈旧,比如某些项目还在使用 Spring 3.x 的版本。所以,在进行需求开发时,也顺手把一些项目的依赖给升级了一下。周五,一个小伙伴问我关于升级 Spring 的经验。正好趁此机会,把一些经验总结一下。 下面的描述以 Java 8 为准,没有在其他版本 Java 上试验过。参考时,请慎重。描述的原则如下: 尽量选择还在维护中的版本,而不是已经 End of Life 的过时版。这样有问题可以及时反馈并得到修复。 Java 8 是目标版本,所以,一定要兼容 Java 8。 Spring Framework 升级 Spring Framework 从 3.2.6.RELEASE 开始提供 BOM。可以利用 BOM 简化 Spring 依赖声明: <!-- D瓜哥 · https://www.diguage.com --> <dependencyManagement> <dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-framework-bom</artifactId> <version>5.3.25</version> <type>pom</type> <scope>import</scope> </dependency> </dependencies> </dependencyManagement> 这样,就不需要重复声明 Spring 依赖的版本,直接使用即可。 Spring Framework Bom 保证了 Spring 自身依赖的版本统一。 这里,关于 Spring 的升级,还有几点需要说明: 从 Spring 3.X 升级到 Spring 4.X+ 后,原来的 MappingJacksonHttpMessageConverter 已经被删除了;直接使用 MappingJackson2HttpMessageConverter 即可。 从 Spring 3.0.0.RELEASE 到 Spring 3.1.4.RELEASE,Spring 有一个 spring-asm,如果不再使用这个区间的 Spring,请把这个依赖删掉。 如果使用了 Apache Velocity 1.X 作为前端模板,由于 Spring 5+ 将相关集成代码删除,所以,只能将 Spring 升级到 4.3.30.RELEASE。相关 BOM 如下: <!-- D瓜哥 · https://www.diguage.com --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-framework-bom</artifactId> <version>4.3.30.RELEASE</version> <type>pom</type> <scope>import</scope> </dependency>
Avro、ProtoBuf、Thrift 的模式演进之法【翻译】

Avro、ProtoBuf、Thrift 的模式演进之法【翻译】

D瓜哥
前面系统研究了 Hessian 序列化协议。并以此为契机,顺带实例对比了 Hessian、MessagePack 和 JSON 的序列化。早在 2012 年,Martin Kleppmann 就写了一篇文章 《Schema evolution in Avro, Protocol Buffers and Thrift》,也是基于实例,对比了 Avro、ProtoBuf、Thrift 的差别。现在翻译出来,方便做系列研究。 整个“序列化系列”目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。
用自己的小故事致敬白衣天使

用自己的小故事致敬白衣天使

D瓜哥
8·19中国医师节 致敬守护健康的白衣天使 虽然我一直挺健康,但我与医生的故事有好多。按时间顺序,挑选几个来说一说。 早在 2018 年,由于智齿斜长,顶住最后大牙,担心影响到大牙的健康,就到北京大学口腔医院魏公村总院,做智齿拔除手术。由于去排队时间较晚,就被分配了一个急诊号,不知道医生是何许人也。拍片做完检查,医生说我上下有两个智齿,可以考虑一起拔除。麻药敲击拔牙,一气呵成。医生说,你可以走啦。我疑问,不是拔两颗吗?怎么只拔了一颗?医生说,两颗都拔啦。没想到这么迅速。实在是太专业啦!这名医生叫郭华秋。 在 2019 年,在盆友圈看到一个纪录片《医者·脊梁》,认识了一个医生,在业余时间,利用自己的专业知识,深入偏远地区,为这些偏远地区免费义诊,并且建立了一个基金会,为需要手术的脊柱侧弯患者承担必要的费用。经过短暂思考后就决定,要为这个基金会捐款,让他们去救助更多的人。初步定在每年六一儿童节这天,每年捐一千块钱,连续捐十年。这家基金会叫北京海鹰脊柱健康公益基金会。这名医生叫刘海鹰。(他是我的安阳老乡,也是我的大学师兄。真是荣幸之至!)放两张盆友圈截图: 图 1. 2018年捐款 图 2. 2022年捐款 2020年,一场突如其来的疫情,让很多很多医护工作者舍小家为大家,用自己的臂膀组建了守护大家健康的卫生长城。这些医生和护士叫中国医护人员。 感谢他们的付出,感谢他们的辛劳,也感谢他们的专业。有感于此,我在2021年初,就请我认识的每一位医护朋友吃一顿饭。可惜到现在为止,还没有成行。以后有机会一起吃饭的话,希望医护朋友不要可以客气,给我一个表达感激的机会。 图 3. 向医护人员致敬 这是公众号的一篇文章。昨天早晨出门吃早餐,刷朋友圈才知道昨天是医师节。在地铁上写了这篇文章。本来是昨天群发的。由于没有想尝试一次群发多篇文章,就在手机APP上先发表了附加的文章,以为只是发布而不群发。结果,搞了个乌龙,直接群发了,也没办法撤销群发。所以,这篇文章只能今天发表了。
Hessian、Msgpack 和 JSON 实例对比

Hessian、Msgpack 和 JSON 实例对比

D瓜哥
前段时间,翻译了 Hessian 2.0 的序列化协议,发布在了 Hessian 2.0 序列化协议(中文版)。但是,其中有很多言语不详之处。所以,接下来会用几篇文章来详细解释并实践一下 Hessian 序列化协议,以求做到知其然知其所以然。 目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。 本文用实际来对比一下 JSON、Hessian 和 MessagePack 的区别。
Hessian 协议解释与实战(五):对象与映射

Hessian 协议解释与实战(五):对象与映射

D瓜哥
前段时间,翻译了 Hessian 2.0 的序列化协议,发布在了 Hessian 2.0 序列化协议(中文版)。但是,其中有很多言语不详之处。所以,接下来会用几篇文章来详细解释并实践一下 Hessian 序列化协议,以求做到知其然知其所以然。目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。 在上一篇文章 Hessian 协议解释与实战(四):数组与集合 中研究了数组和集合的处理方式。接下来介绍对象和映射的处理。 基础工具方法 基础工具方法就不再赘述,请直接参考 Hessian 协议解释与实战(一):基础工具方法 中提到的几个方法。
Hessian 协议解释与实战(四):数组与集合

Hessian 协议解释与实战(四):数组与集合

D瓜哥
前段时间,翻译了 Hessian 2.0 的序列化协议,发布在了 Hessian 2.0 序列化协议(中文版)。但是,其中有很多言语不详之处。所以,接下来会用几篇文章来详细解释并实践一下 Hessian 序列化协议,以求做到知其然知其所以然。目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。 在上一篇文章 Hessian 源码分析(Java) 对 Hessian 的 Java 实现做了一个概要的分析,对处理流程以及整体架构做了一个简单的分析。接下来,回到主题,继续来解释 Hessian 序列化协议。这篇文章,我们来重点分析一下数组与集合相关的操作。
Hessian 源码分析(Java)

Hessian 源码分析(Java)

D瓜哥
前面通过几篇文章,解释并实践了一下 Hessian 的序列化协议。文章目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。 该系列第四篇文章准备详细介绍一下 Hessian 对对象、链表以及 Map 等处理。但是,越调试代码,越发觉得应该先对 Hessian 的实现做一个源码分析。于是,就有了本文。 这里有几点需要声明一下: 在上面“解释与实战”系列文章中提到的代码就不再重复说明。 通过“解释与实战”系列文章,大家应该可以领略到,处理序列化有大量的细节。但是,本文并不打算涉及。本文重点是介绍 Hessian 的 Java 实现的架构蓝图。相当于给指明一条路,沿着这条路,大家就可以探索 Hessian 的各种细节。 本文的介绍,全部基于 Hessian 4.0.60 的源码。由于没有找到 Hessian 的仓库,D瓜哥从 Hessian 的网站下,下载了源码包,解压后发布在了 GitHub 上: Hessian — The source code of Hessian Library.。
Hessian 协议解释与实战(三):字符串

Hessian 协议解释与实战(三):字符串

D瓜哥
前段时间,翻译了 Hessian 2.0 的序列化协议,发布在了 Hessian 2.0 序列化协议(中文版)。但是,其中有很多言语不详之处。所以,接下来会用几篇文章来详细解释并实践一下 Hessian 序列化协议,以求做到知其然知其所以然。目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。 在上一篇文章 Hessian 协议解释与实战(二):长整型、二进制数据与 Null 中研究了长整型、二进制数据与 null 等三种数据类型的处理方式。接下来,我们再来介绍字符串的处理情况。
Hessian 协议解释与实战(二):长整型、二进制数据与 Null

Hessian 协议解释与实战(二):长整型、二进制数据与 Null

D瓜哥
前段时间,翻译了 Hessian 2.0 的序列化协议,发布在了 Hessian 2.0 序列化协议(中文版)。但是,其中有很多言语不详之处。所以,接下来会用几篇文章来详细解释并实践一下 Hessian 序列化协议,以求做到知其然知其所以然。目录如下: Hessian 2.0 序列化协议(中文版) — Hessian 序列化协议的中文翻译版。根据后面的“协议解释与实战”系列文章,增加了协议内容错误提示。 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数 — 介绍布尔型数据、日期类型、浮点类型数据和整数类型数据等四种类型的数据的处理。 Hessian 协议解释与实战(二):长整型、二进制数据与 Null — 介绍长整数类型数据、二进制数据和 null 等三种类型的数据的处理。 Hessian 协议解释与实战(三):字符串 — 专门介绍了关于字符串的处理。由于字符串需要铺垫的基础知识比较多,处理细节也有繁琐,所以单独成篇来介绍。 Hessian 源码分析(Java) — 开始第四篇分析之前,先来介绍一下 Hessian 的源码实现。方便后续展开说明。 Hessian 协议解释与实战(四):数组与集合 — 铺垫了一些关于实例对象的处理,重点介绍关于数组和集合的相关处理。 Hessian 协议解释与实战(五):对象与映射 — 重点介绍关于对象与映射的相关处理。 Hessian、Msgpack 和 JSON 实例对比 — 用实例对比 JSON、Hessian 和 MessagePack 的区别。 Avro、ProtoBuf、Thrift 的模式演进之路 — 翻译的 Martin Kleppmann 的文章,重点对比了 Avro、ProtoBuf、Thrift 的序列化处理思路。 在上一篇文章 Hessian 协议解释与实战(一) 中研究了布尔型数据、日期类型、浮点类型数据、整数类型数据等四种数据类型的处理方式。接下来,我们再来介绍长整数类型数据、二进制数据和 null 的处理情况。 基础工具方法 基础工具方法就不再赘述,请直接参考 Hessian 协议解释与实战(一):布尔、日期、浮点数与整数:基础工具方法 中提到的几个方法。