我们正在使用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'
我仍然面临与以前相同的运行时异常。
答案 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);