Python自动生成10000个java类使用APT注解后引发的问题

前言

前面写了一篇关于自己开发的一个基于APT注解的用于RecyclerView复杂楼层的开源框架,框架的原理比较简单,通过注解,在编译期会生成一个ComponentRule.java的文件,然后建立一个映射关系。使用方式简单介绍一下:
1.绑定布局文件

@ComponentType(
        value = ComponentId.SIMPLE,
        layout = R.layout.single_text
)
public class SimpleVH extends Component<SimpleModel> {
    private TextView tvList;
    public SimpleVH(Context context, View itemView) {
        super(context, itemView);
        tvList = itemView.findViewById(R.id.tv_simple);
    }

    @Override
    public void onBind(int pos, SimpleModel item) {
        tvList.setText(item.name);
    }
}

2.绑定Model

@BindType(ComponentId.SIMPLE)
public class SimpleModel {
    public String name;

    public SimpleModel(String name) {
        this.name = name;
    }
}

3.这样在编译的时候就会生成一个ComponentRule.java文件,建立映射关系。文件的内容大概如下:

public class ComponentRule implements IComponentRule  {
    public static final SparseArray<ViewInfo> WIDGET_TYPE;

    public static final Map<Class<?>, SparseArray<ViewInfo>> ATTACH_TYPE;

    public static final Map<Class<?>, Integer> MODEL_TYPE;

    static {
        WIDGET_TYPE = new SparseArray<>();
        MODEL_TYPE = new HashMap();
        ATTACH_TYPE = new HashMap<>();

        putWidget(2,new ViewInfo(2,
                com.study.xuan.emvp.vh.ImageViewHolder.class,2131296309,1, com.study.xuan.emvp.presenter.Contract.ImagePresenter.class));
        putWidget(8,new ViewInfo(8,
                com.study.xuan.emvp.vh.ImgVH.class,-1,2,android.widget.ImageView.class, null));
        ......
        putModel(com.study.xuan.emvp.model.Text.class,0);
    }
}

问题描述

考虑到目前RecyclerView的使用率,一个大型的项目定义的ViewHolder数量可能达到上千个,考虑到如下几个问题:
1.项目的编译速度影响
2.是否会有其他问题
所以这里希望模拟创建1w个ViewHolder类,使用@ComponentType注解,所以用Python写了一个脚本。

import os

def createViewHolder(content,fileName):
    path = '/Users/xuan/Projects/EMvp/app/src/main/java/com/study/xuan/emvp/python'

    if not os.path.exists(path):
        os.makedirs(path)

    name = fileName + '.java'
    file = open(name,'w');
    file.write(content)

    file.close()

    print ('ok')


contentCode = "package com.study.xuan.emvp.python;\n" \
"import android.content.Context;\n" \
"import android.view.View;\n" \
"import android.widget.TextView;\n" \
"import com.xuan.annotation.ComponentType;\n" \
"import com.xuan.eapi.component.Component;\n" \
"import com.study.xuan.emvp.model.Text;\n" \
"@ComponentType(\n" \
"        value = %s,\n" \
"        view = TextView.class,\n" \
"        attach = Text.class" \
")\n" \
"public class PyThonVH%s extends Component {\n" \
"    public PyThonVH%s(Context context, View itemView) {\n" \
"        super(context, itemView);\n" \
"    }\n" \
"    @Override\n" \
"    public void onBind(int pos, Object item) {\n" \
"    }" \
"}"

fileName = 'PyThonVH%s'


for i in range(1,2000):
    createViewHolder(contentCode%((100+i),i,i),fileName%i)

代码也很基础,在当前工程的一个目录下,利用for循环,创建Java文件,文件名就是PyThonVH1,PyThonVH2,PyThonVH3,PyThonVH4...,注解就使用最基础的注解,为了方式ComponentId冲突,这里利用了框架本身提供的多人协作的解决方式attach到一个Model上,然后CompoentId为1,2,3,4...

代码写完了,脚本一运行,成功创建的1w个类



接下来开始验证问题,当build的时候,编译期报了一个意想不到的异常


代码过长异常

代码过长,没有看错,还是第一次遇到这样的异常信息,通过Google查询,得知

JVM规范里对Class文件的规定里有写到每个方法的字节码最多只能有65535字节

其实原理和我们经常遇到的64K异常一样,只不过这会不是方法数量超过导致的,而是方法体大小导致的。
所以解决方式其实也是对应的,拆分

if (commonTypeWidget.size() < LINE_LIMIT) {
            //未超限,不用分割
            writer.write(writeWidget(0, commonTypeWidget.size()));
        } else {
            //分割方法,防止too large code异常
            double splitNum = Math.ceil(commonTypeWidget.size() / LINE_LIMIT);
            for (int i = 0; i < splitNum; i++) {
                int start = (int) (i * LINE_LIMIT);
                int end;
                if (i == splitNum - 1) {
                    end = commonTypeWidget.size();
                } else {
                    end = (int) ((i + 1) * LINE_LIMIT);
                }
                //保存拆分的方法
                splitMethods.add(String.format(FileCreator.COMMON_METHOD_T, i, writeWidget(start,
                        end)));
                writer.write(String.format(FileCreator.COMMON_METHOD_INVOKE, i));
            }
        }

解决方式的核心代码其实就在上面,对映射表的大小和定义的方法体大小(500)比较,如果超过限制,则进行分割,分别查分到splitAttachMethodStep%s()方法内。最后编译的成的文件就变成如下:

static {
        WIDGET_TYPE = new SparseArray<>();
        MODEL_TYPE = new HashMap();
        ATTACH_TYPE = new HashMap<>();

        ...
        splitAttachMethodStep0();
        splitAttachMethodStep1();
        splitAttachMethodStep2();
        splitAttachMethodStep3();
        splitAttachMethodStep4();
        splitAttachMethodStep5();
        splitAttachMethodStep6();
        splitAttachMethodStep7();
        splitAttachMethodStep8();
        splitAttachMethodStep9();
        splitAttachMethodStep10();
        splitAttachMethodStep11();
        splitAttachMethodStep12();
        splitAttachMethodStep13();
        splitAttachMethodStep14();
        splitAttachMethodStep15();
        splitAttachMethodStep16();
        splitAttachMethodStep17();
        splitAttachMethodStep18();
        splitAttachMethodStep19();
        splitAttachMethodStep20();
        putModel(com.study.xuan.emvp.activity.product.Product.class,0);
        putModel(com.study.xuan.emvp.activity.common.SimpleModel.class,9);
        putModel(com.study.xuan.emvp.model.Text.class,0);
    }

当1w个类的时候,编译的速度影响也不是特别的大,最终的编译时间大是20s左右


速度

总结

本篇博客主要讲解了自己写的开源框架时遇到的一个比较有意思的问题,当然这个问题对于一个稳定的框架是必须要考虑的。这里再放上框架源码地址吧,框架支持组件化工程,适合RecyclerView简单或者复杂的楼层样式开发模式,支持多人多页面楼层打通,具有很多的拓展API,欢迎大家提issue讨论~

项目地址:EMvp
欢迎Star👏
欢迎大家提issues提意见~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,392评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,680评论 2 59
  • 走在九月的街道,耳畔传来一首美妙的歌曲,‘愿得一人心,白首不相离’,我用手机摇到这首歌,发到了朋友圈。 片刻,他的...
    云中飘舞阅读 231评论 11 17
  • 既然来了日本,怎么可以不去迪斯尼乐园呢?情人节的那天,我跟同屋的于颖、闫萍和还有闫萍的男朋友冯云松,四个人约好一...
    含月6666阅读 261评论 0 0
  • 昨天是2018年5月29日,晚上参加了马龙飞老师的粉丝见面会,他用短短的时间去告诉我们:一个人在一个环境中呆...
    星空_666阅读 223评论 0 0