一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - Java Agent入門學習之動態(tài)修改代碼

Java Agent入門學習之動態(tài)修改代碼

2020-11-28 12:33chosen0ne Java教程

這篇文章主要給大家分享了Java Agent入門學習之動態(tài)修改代碼的相關(guān)資料,文中介紹的非常詳細,對大家具有一定的參考學習價值,需要的朋友們下面來一起看看吧。

前言

最近用了一下午總算把Java agent給跑通了,本篇文章記錄一下具體的操作步驟,以免遺忘。下面話不多說,來一起看看詳細的介紹:

通過java agent可以動態(tài)修改代碼(替換、修改類的定義),進行AOP。

目標:

為所有添加@ToString注解的類實現(xiàn)默認的toString方法 

需要兩個程序,一個是用來測試的程序,一個agent用于修改代碼。

1. 測試程序

被測試的程序包括:

  - ToString.Java

  - Foo.java

  - Main.java

具體代碼如下:

ToString.java:定義ToString注解

?
1
2
3
4
5
6
7
8
package com.chosen0ne.agent.test;
 
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
 
@Retention(RetentionPolicy.RUNTIME)
public @interface ToString {
}

Foo.java:很簡單用于測試,使用了ToString注解

?
1
2
3
4
5
6
package com.chosen0ne.agent.test;
 
@ToString
public class Foo {
 
}

Main.java:

?
1
2
3
4
5
6
7
8
package com.chosen0ne.agent.test;
 
public class Main {
 public static void main(String[] args) {
  Foo foo = new Foo();
  System.out.println(foo.toString());
 }
}

執(zhí)行Main.java,結(jié)果如下:

?
1
com.chosen0ne.agent.test.Foo@7852e922

可以看到toString返回的是Object的默認實現(xiàn)。

2. Agent程序

java agent程序?qū)嶋H上類似于鉤子,有兩種方式:

  - main函數(shù)開始前

  - 程序運行中

這里主要測試main函數(shù)開始前的情況。類似于main函數(shù),需要實現(xiàn)

?
1
public static void premain(String agentArgs, Instrumentation inst);

這個函數(shù)會在main函數(shù)之前被調(diào)用。可以在premain中,進行字節(jié)碼操作,替換或重新實現(xiàn)一些類。這里使用Byte Buddy庫,在ASM之上提供了更高級的抽象,便于使用。

具體代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
package com.chosen0ne.ByteCode.agent;
 
import java.lang.instrument.Instrumentation;
 
import com.chosen0ne.agent.test.ToString;
 
import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.dynamic.DynamicType.Builder;
import net.bytebuddy.implementation.FixedValue;
import net.bytebuddy.matcher.ElementMatchers;
 
public class ToStringAgent {
 
 public static void premain(String args, Instrumentation instrumentation) {
  System.out.println("print pre main");
  new AgentBuilder.Default()
    .type(ElementMatchers.isAnnotatedWith(ToString.class))
    .transform(new AgentBuilder.Transformer() {
 
     @Override
     public Builder<?> transform(Builder<?> builder,
       TypeDescription typeDescription, ClassLoader classLoader) {
      return builder.method(ElementMatchers.named("toString"))
        .intercept(FixedValue.value("test"));
     }
      
    }).installOn(instrumentation);
 }
}

agent需要打包成jar,并且對于premain的方式需要在MANIFEST.MF中指定Premain-Class,用于指明包含premain函數(shù)的類。具體有兩種方式打包:

 1)直接通過jar命令

編輯生成MANIFEST.MF后,執(zhí)行:

?
1
jar cvfm agent.jar MANIFEST.MF -C . com lib

上述命令打包成的jar包含:

  - com:編譯生成的class文件

  - lib:其依賴的庫

 2)通過maven直接生成:

通過maven-jar-plugin插件生成jar包,具體配置如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
<build>
 <plugins> 
  <plugin>
   <groupId>org.apache.maven.plugins</groupId>
   <artifactId>maven-jar-plugin</artifactId>
   <version>2.1</version>
   <configuration>
    <archive>
     <manifest>
      <addClasspath>true</addClasspath>
      <classpathPrefix>lib/</classpathPrefix>
      <mainClass>com.chosen0ne.ByteCode.ByteBuddyTest</mainClass>
     </manifest>
     <manifestEntries>
      <Premain-Class>com.chosen0ne.ByteCode.agent.ToStringAgent</Premain-Class>
     </manifestEntries>
    </archive>
   </configuration>
  </plugin>
 </plugins>
</build>

主要通過manifestEntries標簽生成自動的屬性,這里指定了Premain-Class

3. 運行

將生成的agent.jar、依賴的ByteBuddy的jar包和測試程序編譯生成的class文件放到一個路徑下,目錄布局如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
.
├── agent.jar
├── classes
│ └── com
│  └── chosen0ne
│   └── agent
│    └── test
│     ├── Foo.class
│     ├── Main.class
│     └── ToString.class
└── lib
 └── byte-buddy-1.2.3.jar

在當前目錄執(zhí)行命令:

?
1
java -cp classes:lib/byte-buddy-1.2.3.jar -javaagent:agent.jar com.chosen0ne.agent.test.Main

運行結(jié)果如下:

?
1
2
print pre main
test

這里需要注意一點,如果將測試程序也打包成jar包的話,那么在通過-cp指定ByteBuddy庫時會失敗,找不到對應(yīng)的class,錯誤如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
> java -cp classes:lib/byte-buddy-1.2.3.jar -javaagent:agent.jar -jar agent-test-case-0.0.1-SNAPSHOT.jar
Exception in thread "main" java.lang.NoClassDefFoundError: net/bytebuddy/matcher/ElementMatcher
 at java.lang.Class.getDeclaredMethods0(Native Method)
 at java.lang.Class.privateGetDeclaredMethods(Class.java:2688)
 at java.lang.Class.getDeclaredMethod(Class.java:2115)
 at sun.instrument.InstrumentationImpl.loadClassAndStartAgent(InstrumentationImpl.java:327)
 at sun.instrument.InstrumentationImpl.loadClassAndCallPremain(InstrumentationImpl.java:401)
Caused by: java.lang.ClassNotFoundException: net.bytebuddy.matcher.ElementMatcher
 at java.net.URLClassLoader$1.run(URLClassLoader.java:372)
 at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
 at java.security.AccessController.doPrivileged(Native Method)
 at java.net.URLClassLoader.findClass(URLClassLoader.java:360)
 at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
 at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
 at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
 ... 5 more
FATAL ERROR in native method: processing of -javaagent failed

暫時不知道具體原因。。。所以直接以class運行即可

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對服務(wù)器之家的支持。

原文鏈接:http://blog.csdn.net/chosen0ne/article/details/50790372

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 色综合色狠狠天天综合色 | jk制服喷水 | 亚洲国产日韩成人综合天堂 | 国产色婷婷亚洲 | 亚洲色图第一页 | 放荡护士玩3p口述 | 欧美视频一区二区三区在线观看 | 99久久这里只有精品 | 欧美一区二区三区免费观看视频 | 男男羞羞视频网站国产 | 无人区在线观看免费视频国语 | 女性性色生活片免费观看 | 美女全身体光羞羞漫画 | a级精品九九九大片免费看 a级动漫 | 美国videos| 国产一区精品 | 亚洲男人天堂a | 免费看国产精品麻豆 | 日本老妇和子乱视频 | 亚洲成在人网站天堂一区二区 | 亚色九九九全国免费视频 | 9l国产精品久久久久麻豆 | a v在线男人的天堂观看免费 | 日朝欧美亚洲精品 | 日本一本草久p | 国产一级片免费视频 | 欧美一级欧美三级 | 欧美在线观看一区二区三 | 成人网子| 久久久精品成人免费看 | 国产日韩成人 | 99热久久这里只有精品23 | 国产午夜精品理论片 | 日本护士撒尿xxxx欧美 | 人性本色 | 亚洲精品一区二区观看 | 视频在线观看入口一二三2021 | 韩国三级在线 | 欧美日韩在线一区二区三区 | 久久青青草视频在线观 | 精品久久久麻豆国产精品 |