How do I override Class<?>.getName() for certain classes?

那年仲夏 提交于 2019-12-01 23:41:10

Fred's answer is okay, but his aspect could be somewhat more elegant with less code and especially fewer reflection calls. Sorry, I prefer AspectJ native code style, but @AspectJ annotation style would not be much longer:

String around(Class clazz) : call(public String Class.getName()) && target(clazz) {
    NameOverride nameOverride = (NameOverride) clazz.getAnnotation(NameOverride.class);
    return nameOverride == null ? proceed(clazz) : nameOverride.value();
}

Here is the full source code. I added a class without annotation to show the different behaviour and also a restriction to class definitions - @Target(ElementType.TYPE) - to the annotation class. I am also showing package names and imports:

package test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface NameOverride {
    String value();
}
package test;

public class NormalClass {}
package test;

@NameOverride("Custom.fully.qualified.class.name")
public class CustomClass {}
package test;

public class Main {
    public static void main(String[] args) {
        System.out.println(NormalClass.class.getName());
        System.out.println(CustomClass.class.getName());
        System.out.println(new NormalClass().getClass().getName());
        System.out.println(new CustomClass().getClass().getName());
    }
}
package aspectj;

import test.NameOverride;

public aspect GetNameOverrider {
    @SuppressWarnings({ "unchecked", "rawtypes" })
    String around(Class clazz) : call(public String Class.getName()) && target(clazz) {
        NameOverride nameOverride = (NameOverride) clazz.getAnnotation(NameOverride.class);
        return nameOverride == null ? proceed(clazz) : nameOverride.value();
    }
}

The output:

test.NormalClass
Custom.fully.qualified.class.name
test.NormalClass
Custom.fully.qualified.class.name
Fred

This is for sure not the best/fastest solution but maybe a POC...

First of all the file structure:

./src/aspectj:
GetNameOverrider.aj

./src/test:
CustomClass.java    Main.java       NameOverride.java

NameOverride.java:

@Retention(RetentionPolicy.RUNTIME)
public @interface NameOverride {
    String value();
}

GetNameOverrider.aj:

@Aspect
public class GetNameOverrider {
    @Around("call(String getName()) && !within(aspectj..*)")
    public String advice(ProceedingJoinPoint pjp) throws Throwable {

        String ret = (String) pjp.proceed();

        String className = "" + pjp.getTarget();
        className = className.replace("class ", "");
        try {
            test.NameOverride anno = Class.forName(className).getAnnotation(
                    test.NameOverride.class);

            if (anno != null) {
                return anno.value();
            }
        } catch (ClassNotFoundException e) {
            return ret;
        }

        return ret;
    }
}

gives me for Main.java:

public class Main {
    public static void main(String[] args) {
        System.out.println(CustomClass.class.getName());
        System.out.println(new CustomClass().getClass().getName());
    }
}

the output:

Custom.fully.qualified.class.name
Custom.fully.qualified.class.name
标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!