android自定义lint规则

项目地址 CustomLintRules

网上有很多讲解的文章,不再赘述。比如浅谈 Android 自定义 Lint 规则的实现。了解基础的同学可以直接看工程。
关于工程结构,简单梳理一下。

图1

截止发文,共有5个模块。不过重点只关注applintjar即可。
lintjar模块是自定义的lint规则。

网上其他文章用的lint库貌似比较老旧,我目前用的是:

dependencies {
    compile 'com.android.tools.lint:lint-api:26.3.1'
    compile 'com.android.tools.lint:lint-checks:26.3.1'
}

算是最新的。如无异常,应该会持续更新。新版本的api和老的有一定出入,并且部分用到了kotlin。

目前有如下几个检测项:

LogDetector

public class LogDetector extends Detector implements Detector.UastScanner {
    public static final Issue ISSUE = Issue.create(
            "LogUsage",
            "避免调用android.util.Log",
            "请勿直接调用android.util.Log,应该使用统一工具类",
            Category.SECURITY, 5, Severity.ERROR,
            new Implementation(LogDetector.class, Scope.JAVA_FILE_SCOPE));

    @Override
    public List<String> getApplicableMethodNames() {
        return Arrays.asList("v", "d", "i", "w", "e", "wtf");
    }

    @Override
    public void visitMethodCall(@NotNull JavaContext context,
                                @NotNull UCallExpression node,
                                @NotNull PsiMethod method) {
        if (context.getEvaluator().isMemberInClass(method, "android.util.Log")) {
            context.report(ISSUE, node, context.getLocation(node), "避免调用android.util.Log");
        }
    }
}

不多说了,其他文章里都是以这个为例子讲解的。

NewThreadDetector

检测直接new Thread的操作。也是非常简单。

public class NewThreadDetector extends Detector implements Detector.UastScanner {

    public static final Issue ISSUE = Issue.create(
            "NewThread",
            "避免自己创建Thread",
            "请勿直接调用new Thread(),建议使用统一的线程管理工具类",
            Category.PERFORMANCE, 5, Severity.ERROR,
            new Implementation(NewThreadDetector.class, Scope.JAVA_FILE_SCOPE));

    @Override
    public List<String> getApplicableConstructorTypes() {
        return Collections.singletonList("java.lang.Thread");
    }

    @Override
    public void visitConstructor(@NotNull JavaContext context,
                                 @NotNull UCallExpression node,
                                 @NotNull PsiMethod constructor) {
        context.report(ISSUE, node, context.getLocation(node),
                "避免自己创建Thread");
    }
}

ConcurrentModifyDetector

检查ConcurrentModificationException的规则,应对的场景相对要简单一些。代码稍微复杂些,不再粘贴,可以直接看源码。检测结果如下:

图2

可以看到,排除了break和return两种情况,检测出了第一个for循环可能出现ConcurrentModificationException

DrawableAttrDetector

检测自定义的drawable文件是否使用了如下格式的属性:
android:color="?attr/XXX"android:drawable="?attr/XXX"。由于自定义drawable使用?attr是api21才有的特性,并且编译器不会给出提示,在开发中会埋下隐患,所以设计了这个规则。检测结果如下:

图3

ModuleAccessibleDetector

android的annotation包中,有个RestrictTo的注解,可以限定类、方法等的使用范围。

@Retention(CLASS)
@Target({ANNOTATION_TYPE,TYPE,METHOD,CONSTRUCTOR,FIELD,PACKAGE})
public @interface RestrictTo {

    /**
     * The scope to which usage should be restricted.
     */
    Scope[] value();

    enum Scope {
        /**
         * Restrict usage to code within the same library (e.g. the same
         * gradle group ID and artifact ID).
         */
        LIBRARY,

        /**
         * Restrict usage to code within the same group of libraries.
         * This corresponds to the gradle group ID.
         */
        LIBRARY_GROUP,

        /**
         * Restrict usage to code within the same group ID (based on gradle
         * group ID). This is an alias for {@link #LIBRARY_GROUP}.
         *
         * @deprecated Use {@link #LIBRARY_GROUP} instead
         */
        @Deprecated
        GROUP_ID,

        /**
         * Restrict usage to tests.
         */
        TESTS,

        /**
         * Restrict usage to subclasses of the enclosing class.
         * <p>
         * <strong>Note:</strong> This scope should not be used to annotate
         * packages.
         */
        SUBCLASSES,
    }
}

Scope定义了限定的几种范围。

  • TESTS限定只有测试代码可以访问
  • SUBCLASSES限定只有子类可以访问

LIBRARYLIBRARY_GROUP,以下是我个人的理解,如有误请指出。
这两个范围需要打成aar | jar包才能起作用,以源码方式使用不起作用。
比如对于库'com.google.code.gson:gson:2.8.2'LIBRARY限定只有该库中的代码可以访问;LIBRARY_GROUP限定groupId是'com.google.code.gson'的库可以访问,比如'com.google.code.gson:data:1.0.0(虚构的库)。

为了使以源码方式使用时也能起作用,这里在工程的annotation模块定义了ModuleAccessible注解类,并使用ModuleAccessibleDetector进行lint检测。

PrintStackTraceDetector

项目中有一些Throwable.printStackTrace()引发的OOM的报告,为了防止伙伴们直接调用该方法,故做了检测,并且加入了自动修复功能。

public class PrintStackTraceDetector extends com.android.tools.lint.detector.api.Detector
        implements com.android.tools.lint.detector.api.Detector.UastScanner {

    public static final Issue ISSUE = Issue.create(
            "PrintStackTraceUsage",
            "避免直接调用Throwable.printStackTrace()",
            "直接调用Throwable.printStackTrace()可能引起OOM",
            Category.LINT, 5, Severity.ERROR,
            new Implementation(PrintStackTraceDetector.class, Scope.JAVA_FILE_SCOPE));

    @Override
    public List<String> getApplicableMethodNames() {
        return Arrays.asList("printStackTrace");
    }

    @Override
    public void visitMethodCall(@NotNull JavaContext context,
                                @NotNull UCallExpression node,
                                @NotNull PsiMethod method) {
        if (context.getEvaluator().isMemberInClass(method, "java.lang.Throwable")) {
            /*
            报告该问题
             */
            context.report(ISSUE,
                    method,
                    context.getLocation(node),
                    "直接调用Throwable.printStackTrace()可能引起OOM,使用自定义方法替代",
                    getLintFix(context, node));
        }
    }

    /**
     * lint自动修复
     */
    private LintFix getLintFix(@NotNull JavaContext context,
                               @NotNull UCallExpression node) {
        /*
            先检查当前文件是否import了我们需要的类
             */
        boolean hasImport = false;
        List<UImportStatement> list = context.getUastFile().getImports();
        for (UImportStatement statement : list) {
            UElement element = statement.getImportReference();
            if ("com.sollian.customlintrules.utils.LogUtils".endsWith(element.asRenderString())) {
                hasImport = true;
                break;
            }
        }

       /*
        第一个修复,替换方法调用
        */
        LintFix fix = fix().replace()
                .all()
                .with("LogUtils.printStaceTrace(" + node.getReceiver().asRenderString() + ')')
                .autoFix()
                .build();

        /*
         第二个修复,import LogUtils类
         */
        LintFix importFix = null;
        if (!hasImport) {
            UImportStatement statement = list.get(list.size() - 1);
            String lastImport = statement.asRenderString() + ';';
            importFix = fix().replace()
                    //最后的一条import语句
                    .text(lastImport)
                    //替换为最后一条import语句,加上LogUtils类
                    .with(lastImport + "\nimport com.sollian.customlintrules.utils.LogUtils;")
                    //替换位置
                    .range(context.getLocation(statement))
                    .autoFix()
                    .build();
        }

        /*
         最终的修复方案
         */
        LintFix.GroupBuilder builder = fix().name("使用LogUtils.printStackTrace替换").composite();
        builder.add(fix);
        if (importFix != null) {
            builder.add(importFix);
        }

        return builder.build();
    }
}

检测的代码很简单。
getLintFix方法是自定义的lint修复方式。我们的目的是将e.printStackTrace()这种语句替换为LogUtils.printStackTrace(e)这种形式。
第一个修复是字符串替换。
第二个修复是检查当前类是否import了LogUtils类,如果没有,则加上。
最后将两个修复打包成一个,然后返回。


app模块中的MainActivity给出了自定义检查的case,可以参考。

注:第一次使用时可能碰到自定义lint不起作用的情况,可以clean再重新build,还不行的话可以重启Android Studio

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

推荐阅读更多精彩内容