JAVA動態代理實現&&動態位元組碼產生(asm)

來源:互聯網
上載者:User
在目前的Java開發包中包含了對動態代理的支援,但是其實現只支援對介面的的實現。其實現主要通過是java.lang.reflect.Proxy類和java.lang.reflect.InvocationHandler介面。Proxy類主要用來擷取動態代理對象,InvocationHandler介面用來限制式呼叫者實現,如下,HelloWorld介面定義的業務方法,HelloWorldImpl是HelloWorld介面的實現,HelloWorldHandler是InvocationHandler介面實現。代碼如下:業務介面:

public interface HelloWorld {        void sayHelloWorld() ;}
業務介面實現:

public class HelloWorldImpl implements HelloWorld {

        public void sayHelloWorld() {               System.out.println("Hello World!");                     }}

 

InvocationHandler實現,需要在介面方法調用前後加入一部份處理工作,這裡僅僅在方法調用前後向後台輸出兩句字串,其代碼如下:

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;

public class HelloWorldHandler implements InvocationHandler {        //要代理的原始對象        private Object objOriginal;        /**         * 建構函式。         * @param obj 要代理的原始對象。         */        public HelloWorldHandler(Object obj) {               this.objOriginal = obj ;        }

        public Object invoke(Object proxy, Method method, Object[] args)                      throws Throwable {                             Object result ;                       //方法調用之前               doBefore();                       //調用原始對象的方法               result = method.invoke(this.objOriginal ,args);                       //方法調用之後               doAfter();                             return result ;        }               private void doBefore() {               System.out.println("before method invoke!");        }               private void doAfter() {               System.out.println("after method invoke!");        }}

 

 

測試代碼:

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Proxy;

public class Test {

        public static void main(String[] args) {

               HelloWorld hw = new HelloWorldImpl();                             InvocationHandler handler = new HelloWorldHandler(hw);                             HelloWorld proxy = (HelloWorld) Proxy.newProxyInstance(                              hw.getClass().getClassLoader(),                              hw.getClass().getInterfaces(),                              handler);

               proxy.sayHelloWorld();        }}

 

 

 

 

Ø         首先擷取一個業務介面的實現對象;Ø         擷取一個InvocationHandler實現,此處是HelloWorldHandler對象;Ø         建立動態代理對象;Ø         通過動態代理對象調用sayHelloWorld()方法,此時會在原始對象HelloWorldImpl. sayHelloWorld()方法前後輸出兩句字串。運行測試類別輸出如下:

before method invoke!Hello World!after method invoke!
此處Test類中的方法調用代碼比較多,在我們的實際應用中可以通過設定檔來來簡化用戶端的調用實現。另外也可以通過動態代理來實現簡單的AOP。
============================================================
動態位元組碼產生(asm)

導讀:
用ASM寫的Hello World。在網上搜尋ASM有關的文章,最後居然又找回Matrix。。汗

ASM2.0位元組碼架構介紹

http://www.matrix.org.cn/resource/article/2006-02-20/ASM+Bytecode+Framework_44220.html

package my;

import java.lang.reflect.Method;

import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

public class Hello {
@SuppressWarnings("unchecked")
public static void main(String[] args) throws Exception {

        ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
        // 類訪問開始:必須
        cw.visit(Opcodes.V1_5, Opcodes.ACC_PUBLIC, "my/Foo", null, "java/lang/Object", null);

        // 至少提供一個建構函式
        MethodVisitor mv = cw.visitMethod(Opcodes.ACC_PUBLIC, "", "()V", null, null);
        // 代碼開始:必須
        mv.visitCode();
        mv.visitVarInsn(Opcodes.ALOAD, 0);
        mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/Object", "", "()V");
        mv.visitInsn(Opcodes.RETURN);
        // 計算棧和局部變數最大空間:必須
        mv.visitMaxs(0, 0);
        // 代碼結束:必須
        mv.visitEnd();

        mv = cw.visitMethod(Opcodes.ACC_PUBLIC | Opcodes.ACC_STATIC, "main",
                        "([Ljava/lang/String;)V", null, null);
        mv.visitCode();
        mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;");
        mv.visitLdcInsn("Hello World!");
        mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println",
                        "(Ljava/lang/String;)V");
        mv.visitInsn(Opcodes.RETURN);
        mv.visitMaxs(0, 0);
        mv.visitEnd();

        // 類結束:必須
        cw.visitEnd();

        final byte[] bs = cw.toByteArray();
        Class clazz = new ClassLoader() {
                protected Class> findClass(String name) throws ClassNotFoundException {
                        return defineClass(name, bs, 0, bs.length);
                }
        }.loadClass("my.Foo");
        Method method = clazz.getMethod("main", new Class[] { String[].class });
        // 數組參數的方法,反射調用方式看起來比較古怪
        method.invoke(null, (Object) new String[0]);

        for (int i = 0; i < bs.length; i++)
                System.out.printf("%d:\t%02X\t%c\n", i, bs[i], (char) bs[i]);

        // OutputStream out = new FileOutputStream("d:/my/Foo.class");
        // out.write(bs);
        // out.close();
        }
}
藉助ASM寫了一個Aqua Data Studio 6.0的破解:

做法很簡單:
1、找到判斷license的方法,修改代碼使總返回true
2、將1個license線程kill掉。

將jar拷到安裝目錄,修改datastudio.bat檔案的最後一行為:
java -javaagent:ads.crack.jar -cp ".\lib\ads.jar;%ADS_PATH%" com.aquafold.datastudio.DataStudio

由於論壇不支援jar檔案上傳,將副檔名改為ads.crack.jar即可。

ads.crack.jar.zip

有時候,如果想要得到程式中某個Class的所有執行個體,也可以用asm修改代碼得到:

package my;

import java.io.FileOutputStream;
import java.io.OutputStream;
import java.lang.reflect.Field;
import java.net.URL;
import java.util.List;

import org.objectweb.asm.ClassAdapter;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.MethodAdapter;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

public class HelloModifyASM {
        public static void main(String[] args) throws Exception {
        URL url = HelloModifyASM.class.getResource("Foo.class");
        ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
        ClassVisitor cv = new ClassAdapter(cw) {

        public void visit(int version, int access, String name, String signature,
                        String superName, String[] interfaces) {
                super.visit(version, access, name, signature, superName, interfaces);

                // 添加欄位:public static List _my_instances;
                super.visitField(Opcodes.ACC_PUBLIC | Opcodes.ACC_STATIC, "_my_instances",
                                "Ljava/util/List;", null, null);

                // 添加靜態初始化塊
                MethodVisitor mv = super.visitMethod(Opcodes.ACC_STATIC, "", "()V", null,null);
                mv.visitCode();
                mv.visitTypeInsn(Opcodes.NEW, "java/util/ArrayList");
                mv.visitInsn(Opcodes.DUP);
                mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/util/ArrayList", "", "()V");
                mv.visitFieldInsn(Opcodes.PUTSTATIC, "my/Foo", "_my_instances", "Ljava/util/List;");
                mv.visitInsn(Opcodes.RETURN);
                mv.visitMaxs(0, 0);
                mv.visitEnd();
        }

        public MethodVisitor visitMethod(int access, String name, String desc,
                        String signature, String[] exceptions) {
                MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);

                // 修改無參的建構函式:
                if (!"".equals(name) || !"()V".equals(desc))
                        return mv;
                return new MethodAdapter(mv) {

                        public void visitInsn(int opcode) {
                                if (opcode == Opcodes.RETURN) {
                                        visitFieldInsn(Opcodes.GETSTATIC, "my/Foo", "_my_instances",
                                                        "Ljava/util/List;");
                                        visitVarInsn(Opcodes.ALOAD, 0);
                                        visitMethodInsn(Opcodes.INVOKEINTERFACE, "java/util/List", "add",
                                                        "(Ljava/lang/Object;)Z");
                                }
                                super.visitInsn(opcode);
                        }
                };
        }
        };

        ClassReader cr = new ClassReader(url.openStream());
        cr.accept(cv, ClassReader.SKIP_DEBUG);
        final byte[] bs = cw.toByteArray();

        OutputStream out = new FileOutputStream("d:/my/Foo.class");
        out.write(bs);
        out.close();

        Class> clazz = new ClassLoader(null) {
                public Class> findClass(String name) throws ClassNotFoundException {
                        if (!"my.Foo".equals(name))
                                return ClassLoader.getSystemClassLoader().loadClass(name);
                        return defineClass(name, bs, 0, bs.length);
                }
        }.loadClass("my.Foo");
        clazz.newInstance();
        clazz.newInstance();

        Field field = clazz.getField("_my_instances");
                List instances = (List) field.get(null);
                System.out.println(instances.size());
                for (Object obj : instances) {
                        System.out.println(obj);
                }
        }
}

問題:如果原始類有、或者沒有靜態初始化塊,處理方法就是不同的。
想了一下,也許可以在visitMethod方法中設定標誌,再在visitEnd方法中進行補充處理(針對沒有的情況)。
另外,如果有多個建構函式,怎樣保證插入的代碼不會重複執行呢?
甚至,想要在原代碼中插入語句,插入位置的尋找也比較費盡,(需要找到不同的RETURN語句的位元組碼)

HelloModifyASM.java

本文轉自
http://www.matrix.org.cn/thread.shtml?topicId=edd2d10c-a79a-11db-8440-755941c7293d&forumId=1

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.