活用ognl表达式
轉(zhuǎn)載:https://github.com/alibaba/arthas/issues/11
前言
Arthas 3.0中使用ognl表達(dá)式替換了groovy來(lái)實(shí)現(xiàn)表達(dá)式的求值功能,解決了groovy潛在會(huì)出現(xiàn)內(nèi)存泄露的問(wèn)題。靈活運(yùn)用ognl表達(dá)式,能夠極大提升問(wèn)題排查的效率。
ognl官方文檔:https://commons.apache.org/proper/commons-ognl/language-guide.html
一個(gè)測(cè)試應(yīng)用
import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Random;/*** @author zhuyong on 2017/9/13.*/ public class Test {public static final Map m = new HashMap<>();public static final Map n = new HashMap<>();static {m.put("a", "aaa");m.put("b", "bbb");n.put(Type.RUN, "aaa");n.put(Type.STOP, "bbb");}public static void main(String[] args) throws InterruptedException {List<Pojo> list = new ArrayList<>();for (int i = 0; i < 40; i ++) {Pojo pojo = new Pojo();pojo.setName("name " + i);pojo.setAge(i + 2);list.add(pojo);}while (true) {int random = new Random().nextInt(40);String name = list.get(random).getName();list.get(random).setName(null);test(list);list.get(random).setName(name);Thread.sleep(1000l);}}public static void test(List<Pojo> list) {}public static void invoke(String a) {System.out.println(a);}static class Pojo {String name;int age;String hobby;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String getHobby() {return hobby;}public void setHobby(String hobby) {this.hobby = hobby;}} }public enum Type {RUN, STOP; }查看第一個(gè)參數(shù)
params是參數(shù)列表,是一個(gè)數(shù)組,可以直接通過(guò)下標(biāo)方式訪(fǎng)問(wèn)
$ watch Test test params[0] -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 26 ms. @ArrayList[@Pojo[Test$Pojo@6e2c634b],@Pojo[Test$Pojo@37a71e93],@Pojo[Test$Pojo@7e6cbb7a],... ]查看數(shù)組中的元素
第一個(gè)參數(shù)是一個(gè)List,想要看List中第一個(gè)Pojo對(duì)象,可以通過(guò)下標(biāo)方式,也可以通過(guò)List的get方法訪(fǎng)問(wèn)。
$ watch Test test params[0][0] -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 14 ms. @Pojo[name=@String[name 0],age=@Integer[2],hobby=null, ]$ watch Test test params[0].get(0) -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 14 ms. @Pojo[name=@String[name 0],age=@Integer[2],hobby=null, ]查看Pojo的屬性
拿到這個(gè)Pojo可以,直接訪(fǎng)問(wèn)Pojo的屬性,如age
$ watch Test test params[0].get(0).age -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 21 ms. @Integer[2]還可以通過(guò)下標(biāo)的方式訪(fǎng)問(wèn)params[0][0]["age"],這個(gè)寫(xiě)法等效于params[0][0].age:
$ watch Test test params[0][0]["name"] -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 53 ms. watch failed, condition is: null, express is: params[0][0][age], ognl.NoSuchPropertyException: com.taobao.arthas.core.advisor.Advice.age, visit /Users/wangtao/logs/arthas/arthas.log for more details.但這樣會(huì)報(bào)錯(cuò),這時(shí)候需要再加一個(gè)引號(hào)
$ watch Test test 'params[0][0]["age"]' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 25 ms. @Integer[2]集合投影
有時(shí)候我們只需要抽取對(duì)象數(shù)組中的某一個(gè)屬性,這種情況可以通過(guò)投影來(lái)實(shí)現(xiàn),比如要將Pojo對(duì)象列表中的name屬性單獨(dú)抽出來(lái),可以通過(guò)params[0].{name}這個(gè)表達(dá)式來(lái)實(shí)現(xiàn)。 ognl會(huì)便利params[0]這個(gè)List取出每個(gè)對(duì)象的name屬性,重新組裝成一個(gè)新的數(shù)組。用法相當(dāng)于Java stream中的map函數(shù)。
$ watch Test test params[0].{name} -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 56 ms. @ArrayList[@String[name 0],@String[name 1],@String[name 2],@String[name 3],null,@String[name 5],@String[name 6],@String[name 7],@String[name 8],@String[name 9], ]集合過(guò)濾
有時(shí)候還需要針對(duì)集合對(duì)象按某種條件進(jìn)行過(guò)濾,比如想找出所有age大于5的Pojo的name,可以這樣寫(xiě)
$ watch Test test "params[0].{? #this.age > 5}.{name}" -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 25 ms. @ArrayList[@String[name 4],@String[name 5],@String[name 6],null,@String[name 8],@String[name 9], ]其中{? #this.age > 5}?相當(dāng)于stream里面的filter,后面的name相當(dāng)于stream里面的map
那如果要找到第一個(gè)age大于5的Pojo的name,怎么辦呢?可以用^或$來(lái)進(jìn)行第一個(gè)或最后一個(gè)的匹配,像下面這樣:
$ watch Test test "params[0].{^ #this.age > 5}.{name}" -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 24 ms. @ArrayList[@String[name 4], ] Command hit execution time limit 1, therefore will be aborted. $ watch Test test "params[0].{$ #this.age > 5}.{name}" -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 43 ms. @ArrayList[@String[name 9], ]多行表達(dá)式
有些表達(dá)式一行之內(nèi)無(wú)法表達(dá),需要多行才能表達(dá),應(yīng)該怎么寫(xiě)的?比如,假設(shè)我們要把所有Pojo的name拿出來(lái),再往里面新加一個(gè)新的元素,在返回新的列表,應(yīng)該如何寫(xiě)?可以通過(guò)中括號(hào)將多個(gè)表達(dá)式串聯(lián)起來(lái),最后一個(gè)表達(dá)式的返回值代表整個(gè)表達(dá)式的最終結(jié)果。臨時(shí)變量可以用#來(lái)表示。
$ watch Test test '(#test=params[0].{name}, #test.add("abc"), #test)' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 28 ms. @ArrayList[@String[name 0],@String[name 1],@String[name 2],@String[name 3],@String[name 4],@String[name 5],@String[name 6],@String[name 7],@String[name 8],null,@String[abc], ]調(diào)用構(gòu)造函數(shù)
調(diào)用構(gòu)造函數(shù),必須要指定要?jiǎng)?chuàng)建的類(lèi)的全類(lèi)名。比如下面的例子中,創(chuàng)建一個(gè)新的list,然后添加一個(gè)新的元素,然后返回添加后的list。
$ watch Test test '(#test=new java.util.ArrayList(), #test.add("abc"), #test)' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 37 ms. @ArrayList[@String[abc], ]訪(fǎng)問(wèn)靜態(tài)變量
可以通過(guò)@class@filed方式訪(fǎng)問(wèn),注意需要填寫(xiě)全類(lèi)名
$ watch Test test '@Test@m' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 35 ms. @HashMap[@String[a]:@String[aaa],@String[b]:@String[bbb], ]調(diào)用靜態(tài)方法
可以通過(guò)@class@method(args)方式訪(fǎng)問(wèn),注意需要填寫(xiě)全類(lèi)名
$ watch Test test '@java.lang.System@getProperty("java.version")' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 42 ms. @String[1.8.0_51]靜態(tài)方法和非靜態(tài)方法結(jié)合,例如想要獲取當(dāng)前方法調(diào)用的TCCL,可以像下面這樣寫(xiě):
$ watch Test test '@java.lang.Thread@currentThread().getContextClassLoader()' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 84 ms. @AppClassLoader[ucp=@URLClassPath[sun.misc.URLClassPath@4cdbe50f],$assertionsDisabled=@Boolean[true], ]訪(fǎng)問(wèn)Map中的元素
Test.n是一個(gè)HashMap,假設(shè)要獲取這個(gè)Map的所有key,ongl針對(duì)Map接口提供了keys,?values這兩個(gè)虛擬屬性,可以像普通屬性一樣訪(fǎng)問(wèn)。
$ watch Test test '@Test@n.keys' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 57 ms. @KeySet[@Type[RUN],@Type[STOP], ]因?yàn)檫@個(gè)Map的Key是一個(gè)Enum,假設(shè)要把key為RUN這個(gè)值的value取出來(lái)應(yīng)該怎么寫(xiě)呢?可以通過(guò)Enum的valueOf方法來(lái)創(chuàng)建一個(gè)Enum,然后get出來(lái),比如下面一樣
$ watch Test test '@Test@n.get(@Type@valueOf("RUN"))' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 168 ms. @String[aaa]或者是下面這樣,通過(guò)迭代器+過(guò)濾的方式:
$ watch Test test '@Test@n.entrySet().iterator.{? #this.key.name() == "RUN"}' -n 1 Press Ctrl+C to abort. Affect(class-cnt:1 , method-cnt:1) cost in 72 ms. @ArrayList[@Node[RUN=aaa], ]附錄: ognl內(nèi)置的ognl的虛擬屬性
- Collection:
- size
- isEmpty
- List:
- iterator
- Map:
- keys
- values
- Set:
- iterator
- Iterator:
- next
- hasNext
- Enumeration:
- next
- hasNext
- nextElement
- hasMoreElements
最后
歡迎在留言區(qū)分享你的牛逼用法,互相交流進(jìn)步~
?
?
總結(jié)
- 上一篇: Spring Data JPA 从入门到
- 下一篇: Encapsulate Collecti