ASM 解读

目的: 
程序分析:用于分析程序,动态生成proxy等。 
程序生成:可在内存中生成java类并编译,所谓的just in time complie 
程序转换:优化程序插入debugging或平台观察代码(AOP) 

ASM库提供生成、转换类的2种API,一直API是基于事件一种是基于数的。基于事件的就类似于xml的SAX,而基于树的就像DOM.这2种 API都有各自的有点与缺点。基于事件的API快速并且需要的内存比基于树的少。 

组织结构: 
org.objectweb.asm与org.objectweb.asm.signature是定义基于事件的API并提供了class的读写 组建。 
org.objectweb.asm.util是工具包。 
org.objectweb.asm.commons定义了预定义类的转行器 
org.objectweb.asm.tree定义了基于树的API 
org.objectweb.asm.tree.analysis定义了基于树API的分析框架和预定义类的分析器 

java字节码的结构 
+----------------------------------------------------+ 
|Modifiers, name, super class, interfaces                                            | 
+----------------------------------------------------+ 
|Constant pool: numeric, string and type constants                         | 
+----------------------------------------------------+ 
|Source file name (optional)                                                               | 
+----------------------------------------------------+ 
|Enclosing class reference                           | 
+----------------------------------------------------+ 
|Annotation*                                         | 
+----------------------------------------------------+ 
|Attribute*                                          | 
+----------------------------------------------------+ 
|Inner class* Name                                   | 
+----------------------------------------------------+ 
|Field* Modifiers, name, type                        | 
|       Annotation*                                  | 
|       Attribute*                                   | 
+----------------------------------------------------+ 
|Method* Modifiers, name, return and parameter types | 
|        Annotation*                                 | 
|        Attribute*                                  | 
|        Compiled code                               | 
+----------------------------------------------------+ 
内部名 
在class文件里面使用的是内部名,如String的内部名是 java/lang/String 
类型描述: 
java类型->类型描述 
boolean->Z 
char->C 
byte->B 
short->S 
int->I 
float->F 
long->J 
double->D 
Object->Ljava/lang/Object; 
int[]->[I 
Object[][]->[[Ljava/lang/Object; 
方法描述: 
方法描述是类型描述列表,用于描述方法的参数类型以及返回值类型 
方法声明->方法描述 
void m(int i, float f)->(IF)V 
int m(Object o)->(Ljava/lang/Object;)I 
int[] m(int i, String s)->(ILjava/lang/String;)[I 
Object m(int[] i)->([I)Ljava/lang/Object; 



生成和解析编译好的类文件是基于ClassVisitor接口。这个接口的每个方法对应类文件相同的段名如下: 
public interface ClassVisitor { 
void visit(int version, int access, String name, String signature, 
String superName, String[] interfaces); 
void visitSource(String source, String debug); 
void visitOuterClass(String owner, String name, String desc); 
AnnotationVisitor visitAnnotation(String desc, boolean visible); 
void visitAttribute(Attribute attr); 
void visitInnerClass(String name, String outerName, String innerName, 
int access); 
FieldVisitor visitField(int access, String name, String desc, 
String signature, Object value); 
MethodVisitor visitMethod(int access, String name, String desc, 
String signature, String[] exceptions); 
void visitEnd(); 

使用方法必须符合下面规则: 
visit visitSource? visitOuterClass? ( visitAnnotation | visitAttribute )* 
( visitInnerClass | visitField | visitMethod )* 
visitEnd 
visit必须是第一个被调用的方法,然后调用visitOuterClass,接着调用visitAnnotation和 visitAttribute依次类推,最后调用visitEnd 
ASM提供3个核心基于ClassVisitor的3个核心组件用于产生解析类: 
1、ClassReader解析已经编译好的类文件,调用对应的VisitXxx方法 
2、ClassWriter实现了ClassVisitor接口,生成类 
3、ClassAdapter实现了ClassVisitor接口,代理调用接口的所有方法 
以下是类解析的例子相关信息,通过Visitor打印类

  1. package  com.appspot.coder9527;  
  2.    
  3. import  java.io.IOException;  
  4.    
  5. import  org.objectweb.asm.AnnotationVisitor;  
  6. import  org.objectweb.asm.Attribute;  
  7. import  org.objectweb.asm.ClassReader;  
  8. import  org.objectweb.asm.ClassVisitor;  
  9. import  org.objectweb.asm.FieldVisitor;  
  10. import  org.objectweb.asm.MethodVisitor;  
  11.    
  12. public   class  ClassInfoVisitor  implements  ClassVisitor {  
  13.  @Override   
  14.  public   void  visit( int  version,  int  access, String name, String signature,  
  15.  String superName, String[] interfaces) {  
  16.  System.out.println(name + " extends "  + superName +  " {" );  
  17.  }  
  18.  @Override   
  19.  public  AnnotationVisitor visitAnnotation(String desc,  boolean  visible) {  
  20.  return   null ;  
  21.  }  
  22.  @Override   
  23.  public   void  visitAttribute(Attribute attr) {  
  24.  }  
  25.  @Override   
  26.  public   void  visitEnd() {  
  27.  System.out.println("}" );  
  28.  }  
  29.  @Override   
  30.  public  FieldVisitor visitField( int  access, String name, String desc,  
  31.  String signature, Object value) {  
  32.  System.out.println(" "  + desc +  " "  + name);  
  33.  return   null ;  
  34.  }  
  35.  @Override   
  36.  public   void  visitInnerClass(String name, String outerName,  
  37.  String innerName, int  access) {  
  38.  }  
  39.  @Override   
  40.  public  MethodVisitor visitMethod( int  access, String name, String desc,  
  41.  String signature, String[] exceptions) {  
  42.  System.out.println(" "  + name + desc);  
  43.  return   null ;  
  44.  }  
  45.  @Override   
  46.  public   void  visitOuterClass(String owner, String name, String desc) {  
  47.  }  
  48.  @Override   
  49.  public   void  visitSource(String source, String debug) {  
  50.  }  
  51.  public   static   void  main(String argv[])  throws  IOException {  
  52.  ClassReader reader = new  ClassReader( "com.appspot.coder9527.ClassInfoVisitor" );  
  53.  ClassInfoVisitor visitor = new  ClassInfoVisitor();  
  54.  reader.accept(visitor, 0 );  
  55.  }  
  56. }  
Java代码   收藏代码
  1. package com.appspot.coder9527;  
  2.    
  3. import java.io.IOException;  
  4.    
  5. import org.objectweb.asm.AnnotationVisitor;  
  6. import org.objectweb.asm.Attribute;  
  7. import org.objectweb.asm.ClassReader;  
  8. import org.objectweb.asm.ClassVisitor;  
  9. import org.objectweb.asm.FieldVisitor;  
  10. import org.objectweb.asm.MethodVisitor;  
  11.    
  12. public class ClassInfoVisitor implements ClassVisitor {  
  13.  @Override  
  14.  public void visit(int version, int access, String name, String signature,  
  15.  String superName, String[] interfaces) {  
  16.  System.out.println(name + " extends " + superName + " {");  
  17.  }  
  18.  @Override  
  19.  public AnnotationVisitor visitAnnotation(String desc, boolean visible) {  
  20.  return null;  
  21.  }  
  22.  @Override  
  23.  public void visitAttribute(Attribute attr) {  
  24.  }  
  25.  @Override  
  26.  public void visitEnd() {  
  27.  System.out.println("}");  
  28.  }  
  29.  @Override  
  30.  public FieldVisitor visitField(int access, String name, String desc,  
  31.  String signature, Object value) {  
  32.  System.out.println(" " + desc + " " + name);  
  33.  return null;  
  34.  }  
  35.  @Override  
  36.  public void visitInnerClass(String name, String outerName,  
  37.  String innerName, int access) {  
  38.  }  
  39.  @Override  
  40.  public MethodVisitor visitMethod(int access, String name, String desc,  
  41.  String signature, String[] exceptions) {  
  42.  System.out.println(" " + name + desc);  
  43.  return null;  
  44.  }  
  45.  @Override  
  46.  public void visitOuterClass(String owner, String name, String desc) {  
  47.  }  
  48.  @Override  
  49.  public void visitSource(String source, String debug) {  
  50.  }  
  51.  public static void main(String argv[]) throws IOException {  
  52.  ClassReader reader = new ClassReader("com.appspot.coder9527.ClassInfoVisitor");  
  53.  ClassInfoVisitor visitor = new ClassInfoVisitor();  
  54.  reader.accept(visitor, 0);  
  55.  }  
  56. }  



来看看ClassReader,要了解ClassReader. 
首先要理解要理解Class文件定义的格式 
struct Class_File_Format { 
   u4 magic_number;   //4个字节的魔幻数  是16进制的0xcafe 0xbabe cafe呵呵有点好耍 
   u2 minor_version;  //2个字节的主版本号 
   u2 major_version;  //2个字节次版本号 
   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];//类附加属性数据,包括源文件名等。 
}; 
看看常量表cp_info的定义。 
cp_info { 
  u1 tag; 
  u1 info[]; 

cp_info的tag保存着类型信息,如下: 
类型 值 
CONSTANT_Class 7 
CONSTANT_Fieldref 9 
CONSTANT_Methodref 10 
CONSTANT_InterfaceMethodref 11 
CONSTANT_String 8 
CONSTANT_Integer 3 
CONSTANT_Float 4 
CONSTANT_Long 5 
CONSTANT_Double 6 
CONSTANT_NameAndType 12 
CONSTANT_Utf8 1 
指定的类型又显示了info指向的每种结构,比如CONSTANT_Class这种类型的指向的结构是 
CONSTANT_Class_info { 
  u1 tag; 
  u2 name_index; 

以下是Class_File_Format的结构图 




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
资源包主要包含以下内容: ASP项目源码:每个资源包中都包含完整的ASP项目源码,这些源码采用了经典的ASP技术开发,结构清晰、注释详细,帮助用户轻松理解整个项目的逻辑和实现方式。通过这些源码,用户可以学习到ASP的基本语法、服务器端脚本编写方法、数据库操作、用户权限管理等关键技术。 数据库设计文件:为了方便用户更好地理解系统的后台逻辑,每个项目中都附带了完整的数据库设计文件。这些文件通常包括数据库结构图、数据表设计文档,以及示例数据SQL脚本。用户可以通过这些文件快速搭建项目所需的数据库环境,并了解各个数据表之间的关系和作用。 详细的开发文档:每个资源包都附有详细的开发文档,文档内容包括项目背景介绍、功能模块说明、系统流程图、用户界面设计以及关键代码解析等。这些文档为用户提供了深入的学习材料,使得即便是从零开始的开发者也能逐步掌握项目开发的全过程。 项目演示与使用指南:为帮助用户更好地理解和使用这些ASP项目,每个资源包中都包含项目的演示文件和使用指南。演示文件通常以视频或图文形式展示项目的主要功能和操作流程,使用指南则详细说明了如何配置开发环境、部署项目以及常见问题的解决方法。 毕业设计参考:对于正在准备毕业设计的学生来说,这些资源包是绝佳的参考材料。每个项目不仅功能完善、结构清晰,还符合常见的毕业设计要求和标准。通过这些项目,学生可以学习到如何从零开始构建一个完整的Web系统,并积累丰富的项目经验。
资源包主要包含以下内容: ASP项目源码:每个资源包中都包含完整的ASP项目源码,这些源码采用了经典的ASP技术开发,结构清晰、注释详细,帮助用户轻松理解整个项目的逻辑和实现方式。通过这些源码,用户可以学习到ASP的基本语法、服务器端脚本编写方法、数据库操作、用户权限管理等关键技术。 数据库设计文件:为了方便用户更好地理解系统的后台逻辑,每个项目中都附带了完整的数据库设计文件。这些文件通常包括数据库结构图、数据表设计文档,以及示例数据SQL脚本。用户可以通过这些文件快速搭建项目所需的数据库环境,并了解各个数据表之间的关系和作用。 详细的开发文档:每个资源包都附有详细的开发文档,文档内容包括项目背景介绍、功能模块说明、系统流程图、用户界面设计以及关键代码解析等。这些文档为用户提供了深入的学习材料,使得即便是从零开始的开发者也能逐步掌握项目开发的全过程。 项目演示与使用指南:为帮助用户更好地理解和使用这些ASP项目,每个资源包中都包含项目的演示文件和使用指南。演示文件通常以视频或图文形式展示项目的主要功能和操作流程,使用指南则详细说明了如何配置开发环境、部署项目以及常见问题的解决方法。 毕业设计参考:对于正在准备毕业设计的学生来说,这些资源包是绝佳的参考材料。每个项目不仅功能完善、结构清晰,还符合常见的毕业设计要求和标准。通过这些项目,学生可以学习到如何从零开始构建一个完整的Web系统,并积累丰富的项目经验。
08-10
资源包主要包含以下内容: ASP项目源码:每个资源包中都包含完整的ASP项目源码,这些源码采用了经典的ASP技术开发,结构清晰、注释详细,帮助用户轻松理解整个项目的逻辑和实现方式。通过这些源码,用户可以学习到ASP的基本语法、服务器端脚本编写方法、数据库操作、用户权限管理等关键技术。 数据库设计文件:为了方便用户更好地理解系统的后台逻辑,每个项目中都附带了完整的数据库设计文件。这些文件通常包括数据库结构图、数据表设计文档,以及示例数据SQL脚本。用户可以通过这些文件快速搭建项目所需的数据库环境,并了解各个数据表之间的关系和作用。 详细的开发文档:每个资源包都附有详细的开发文档,文档内容包括项目背景介绍、功能模块说明、系统流程图、用户界面设计以及关键代码解析等。这些文档为用户提供了深入的学习材料,使得即便是从零开始的开发者也能逐步掌握项目开发的全过程。 项目演示与使用指南:为帮助用户更好地理解和使用这些ASP项目,每个资源包中都包含项目的演示文件和使用指南。演示文件通常以视频或图文形式展示项目的主要功能和操作流程,使用指南则详细说明了如何配置开发环境、部署项目以及常见问题的解决方法。 毕业设计参考:对于正在准备毕业设计的学生来说,这些资源包是绝佳的参考材料。每个项目不仅功能完善、结构清晰,还符合常见的毕业设计要求和标准。通过这些项目,学生可以学习到如何从零开始构建一个完整的Web系统,并积累丰富的项目经验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值