如何在编译期直接操作 AST? - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
Sign Up Now
For Existing Member  Sign In
ccde8259
V2EX    Java

如何在编译期直接操作 AST?

  •  
  •   ccde8259 Sep 21, 2021 via iPhone 3330 views
    This topic created in 1679 days ago, the information mentioned may be changed or developed.
    不妨假设现在有一个场景,需要大量的模板代码。具体一点,你在使用 gRPC 的时候写完 proto,生成了一车的 class 。
    现在有没有一种比较易用的方式,实现去掉这块模板代码?
    解决这个问题思路其实有两个方向,基于作用时间区分,一个是编译期注入,一个是运行时生成。
    前者的思路需要在编译期根据注解做额外的操作。类似的技术方案是 lombok 和 aspectj 。其中 lombok 本身就是做同一个事情,但没有找到合适的扩展方案。反过来 aspectj 的 CTW 做的事是 weaving,并不支持更多代码生成上的功能。
    后者的思路则要去处理类加载这块的问题,一方面可能需要定制类加载器,hook 桩代码,根据桩代码同时生成所有目标字节码。这种方案受限于类加载器本身的限制,修改类继承之类的操作都不太可能实现,也会拖慢类加载的效率。
    肯定有人会回复:干嘛不干脆重写一个前端?成本考量下,自行添加一丢丢语法糖显然是最便宜的方案。
    在选择使用直接编译期操作 AST 的方案情况下,有没有相对成熟的案例?
    8 replies    2021-09-22 10:56:56 +08:00
    humpy
        1
    humpy  
       Sep 21, 2021
    namelosw
        2
    namelosw  
       Sep 21, 2021
    Java 不知道,不过一般编译期操作 AST 听起来就是完全对等 “宏” 这个概念。
    dcoder
        3
    dcoder  
       Sep 21, 2021
    写 LISP
    billlee
        4
    billlee  
       Sep 21, 2021   1
    annotation processor?
    ch2
        5
    ch2  
       Sep 21, 2021
    魔改前端
    Ediacaran
        6
    Ediacaran  
       Sep 21, 2021 via iPhone
    写个预处理器生成代码
    ychost
        7
    ychost  
       Sep 21, 2021
    就是 Annotation Processor,不过在 Java8 之后的版本 javac 的 API 变化很大,我写了个类似 Lombok 原理实现的编译期的 AOP 库,https://github.com/fast-light/fastaop
    yizmaoaa
        8
    yizmaoaa  
       Sep 22, 2021
    APT+ASM (或者其他的)

    具体可以参考 Quarkus/Micronaut
    About     Help     Advertise     Blog     API     FAQ     Solana     4494 Online   Highest 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 44ms UTC 10:12 PVG 18:12 LAX 03:12 JFK 06:12
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86