Java - 使用特定名称创建匿名Exception子类

时间:2017-08-17 15:44:38

标签: java android crashlytics byte-buddy crashlytics-android

我们正在使用Fabric / Crashlytics将非崩溃问题发送到他们的服务器,然后检查日志和堆栈跟踪。

不幸的是,Crashlytics API要求为不同的问题类型使用不同的Exception子类实例。 (在iOS SDK中,您可以简单地将字符串用于不同的问题类型,但在Android版本中,您不能)。

public static void craslyticsRecordError(String issueType, String errorMsg)
{
    Exception e = new Exception(issueType + "-" + errorMsg);
    Crashlytics.logException(e);
}

现在我们有很多不同的问题类型,我们也在运行时动态定义(从字符串部分一起构建)。所以问题是,是否可以使用某些Java Vodoo / Hacks(反射?)在运行时创建一个具有特定名称(我们用作问题类型的字符串的名称)的子类,并从中创建和实例。 / p>

这有可能吗?

我现在正在摆弄ByteBuddy,试图让它发挥作用 - 但我坚持让我的动态类正确地调用超类构造函数。此外,我不太确定我是否可以在以后加入Exception。

感谢拉斐尔·温特哈尔特(Rafael Winterhalter),我更进一步了解这件事;)我已经尝试过你所提供的:

public static void craslyticsRecordError(String issueType, String errorMsg)
{
    Class<? extends Exception> dynamicType = new ByteBuddy()
            .subclass(Exception.class)
            .name(issueType)
            .make()
            .load(Fabric.class.getClassLoader(), ClassLoadingStrategy.Default.WRAPPER)
            .getLoaded();
    try
    {
        Exception e = dynamicType.getConstructor(String.class).newInstance(errorMsg);

        GameLog.d("[Fabric] Inner Exception: " + e);
        Crashlytics.logException(e);
    }
    catch (Exception e1)
    {
        Exception e = new Exception(issueType + "-" + errorMsg);
        GameLog.d("[Fabric] ERROR ONE");
        Crashlytics.logException(e);
        e1.printStackTrace();
    }
}

现在我正面临这个运行时异常:

java.lang.NoClassDefFoundError: net.bytebuddy.dynamic.loading.NoOpClassFileTransformer
     at net.bytebuddy.dynamic.loading.ByteArrayClassLoader.<init>(ByteArrayClassLoader.java:136)
     at net.bytebuddy.dynamic.loading.ByteArrayClassLoader.of(ByteArrayClassLoader.java:187)
     at net.bytebuddy.dynamic.loading.ByteArrayClassLoader.load(ByteArrayClassLoader.java:212)
     at net.bytebuddy.dynamic.loading.ClassLoadingStrategy$Default$WrappingDispatcher.load(ClassLoadingStrategy.java:285)
     at net.bytebuddy.dynamic.loading.ClassLoadingStrategy$Default.load(ClassLoadingStrategy.java:120)
     at net.bytebuddy.dynamic.TypeResolutionStrategy$Passive.initialize(TypeResolutionStrategy.java:79)
     at net.bytebuddy.dynamic.DynamicType$Default$Unloaded.load(DynamicType.java:4376)
     at org.utils.Fabric.craslyticsRecordError(Fabric.java:47)
     at android.os.MessageQueue.nativePollOnce(Native Method)
     at android.os.MessageQueue.next(MessageQueue.java:323)
     at android.os.Looper.loop(Looper.java:143)
     at android.app.ActivityThread.main(ActivityThread.java:7224)
     at java.lang.reflect.Method.invoke(Native Method)
     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1230)
     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1120)

现在使用此特定于Android的代码:

Class<? extends Exception> dynamicType = new ByteBuddy()
        .subclass(Exception.class)
        .name(issueType)
        .make()
        .load(Fabric.class.getClassLoader(), AndroidClassLoadingStrategy.Default.WRAPPER)
        .getLoaded();

并使用Android特定的gradle编译行:

compile 'net.bytebuddy:byte-buddy-android:1.7.3'

我仍然面临与以前相同的运行时异常。

1 个答案:

答案 0 :(得分:1)

默认情况下,Byte Buddy复制其超类的构造函数。在您的情况下,它复制所有Exception构造函数,您可以生成如下异常:

Class<? extends Exception> dynamicType = new ByteBuddy()
  .subclass(Exception.class)
  .name(issueType)
  .make()
  .load(Fabric.class.getClassLoader(), ClassLoadingStrategy.Default.WRAPPER)
  .getLoaded();

 Exception e = dynamicType.getConstructor(String.class).newInstance(errorMsg);