【问题标题】:Android app crashes if installed as apk, works if runned trough Android Studio如果作为应用程序安装,Android 应用程序会崩溃,如果通过 Android Studio 运行则可以工作
【发布时间】:2016-09-04 11:23:47
【问题描述】:

我在 Android Studio 2.1.3 中制作了一个应用程序,当然我已经在(不同的)模拟器和我自己的手机(小米红米 Note 3,Android 5.0.2 LRX22G)中进行了测试。当我把应用程序给几个朋友测试时,他们都说了同样的话:应用程序正常启动,他们选择他们感兴趣的活动,他们在文本字段中输入数据以及按下按钮的那一刻(假设做一些数学)为了接收数学答案,应用程序停止并返回到“主要”活动。

compileSdkVersion 24 
buildToolsVersion '23.0.3' 
dexOptions { javaMaxHeapSize "4g" } 
defaultConfig { applicationId "com.example.android.rechner" minSdkVersion 17 
targetSdkVersion 24 
versionCode 4 
versionName 'v.1.6.0 - Entwickler Stufe V.16.9.03' 
multiDexEnabled true

我买了一台平板电脑,希望将同样的崩溃连接到 android studio 并有机会进行调试。我通过BT将apk(调试apk,不是发布)发送到平板电脑(Teclast Tpad X80 Pro(E3E7),Android 5.1),安装了应用程序,并按预期工作! 现在我要生气了! :) 我去找另一个朋友做了和平板电脑一样的事情,但是把文件发送到了三星 Galaxy Grand Prime(没看它有什么安卓版本),当我尝试做数学时崩溃了!他给了我一个三星 Galaxy S3 Mini(GT-I8200N,Android 4.2.2),我通过 BT 发送 APK,同样的事情发生了!

好吧,我开始高兴了,因为我带着手机,启用开发者选项,通过 USB 进行调试,将手机连接到我的电脑,使用 Android Studio 加载应用程序,你猜怎么着?!它可以正常工作,没有任何错误、折痕或任何东西! 我将手机重置为出厂设置,希望能强制出错……我通过 BT 发送了文件,然后……它可以工作了!没有任何错误或崩溃,但重置后手机没有连接到我的电脑(和 Android Studio)。我看不到问题,也无法在调试日志中寻找答案,因为连接到 PC 没有失败:(

我需要提及的是,我在测试过该应用的其他手机上启用了开发者选项,但唯一的区别是它们都没有连接到我的电脑和 Android Studio!

你们中有人遇到过这个问题吗?我在我们的谷歌朋友上搜索了几天类似的东西,但没有找到任何东西。

感谢您的耐心和解答!

最好的问候,罗伯特 P。

L.E.我重新刷了手机,删除了缓存和数据,重新安装了apk,最后崩溃了。我成功阅读了 adb logcat :

W/dalvikvm( 4700): threadid=1: thread exiting with uncaught exception (group=0x40d7d960)
E/AndroidRuntime( 4700): FATAL EXCEPTION: main
E/AndroidRuntime( 4700): java.lang.StringIndexOutOfBoundsException: length=4; regionStart=0; regionLength=-1
E/AndroidRuntime( 4700):        at java.lang.String.startEndAndLength(String.java:583)
E/AndroidRuntime( 4700):        at java.lang.String.substring(String.java:1464)
E/AndroidRuntime( 4700):        at com.example.android.rechner.MainActivityRustZ.displayTimeReal(MainActivityRustZ.java:109)
E/AndroidRuntime( 4700):        at com.example.android.rechner.MainActivityRustZ.access$200(MainActivityRustZ.java:12)
E/AndroidRuntime( 4700):        at com.example.android.rechner.MainActivityRustZ$1.onClick(MainActivityRustZ.java:29)
E/AndroidRuntime( 4700):        at android.view.View.performClick(View.java:4452)
E/AndroidRuntime( 4700):        at android.widget.Button.performClick(Button.java:148)
E/AndroidRuntime( 4700):        at android.view.View$PerformClick.run(View.java:18428)
E/AndroidRuntime( 4700):        at android.os.Handler.handleCallback(Handler.java:725)
E/AndroidRuntime( 4700):        at android.os.Handler.dispatchMessage(Handler.java:92)
E/AndroidRuntime( 4700):        at android.os.Looper.loop(Looper.java:176)
E/AndroidRuntime( 4700):        at android.app.ActivityThread.main(ActivityThread.java:5365)
E/AndroidRuntime( 4700):        at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime( 4700):        at java.lang.reflect.Method.invoke(Method.java:511)
E/AndroidRuntime( 4700):        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1102)
E/AndroidRuntime( 4700):        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:869)
E/AndroidRuntime( 4700):        at dalvik.system.NativeStart.main(Native Method)

现在,第 109 行: String unitatiTimpReal = formattedTimpR.substring(0, formattedTimpR.indexOf(".")); //单位

属于:

private void displayTimeReal(double timpreal) {
    Double vZeitD = Double.parseDouble(((EditText) findViewById(R.id.vZeit)).getText().toString());
    Double aufStZahl = Double.parseDouble(((EditText) findViewById(R.id.aufGesSt)).getText().toString());
    timpreal = aufStZahl / ((procente / vZeitD) * mitarbeiter);
    DecimalFormat formattr = new DecimalFormat();
    formattr.setMaximumFractionDigits(2);
    formattr.setMinimumFractionDigits(2);
    String formattedTimpR = formattr.format(timpreal);
    String zecimaleTimpReal = formattedTimpR.substring(formattedTimpR.indexOf(".") + 1); //decimals
    String unitatiTimpReal = formattedTimpR.substring(0, formattedTimpR.indexOf(".")); //unit
    double minutereal = Double.parseDouble(zecimaleTimpReal);
    minutereal = minutereal * 0.6;
    DecimalFormat format = new DecimalFormat();
    format.setMaximumFractionDigits(2);
    String formattedMinuteReal = format.format(minutereal);
    TextView timeTextView = (TextView) findViewById(R.id.rezultat_ore_real);
    timeTextView.setText(getString(R.string.slaufz) + unitatiTimpReal + getString(R.string.shours) + formattedMinuteReal + getString(R.string.sminutes));
}

现在,相同的行,相同的操作,相同的 apk ......但如果我只通过 Android Studio 运行应用程序一次,直到完成固件重写和缓存删除,应用程序将运行没有崩溃!

按照建议更新后: 这一行:

String unitatiTimpReal = formattedTimpR.substring(0, formattedTimpR.indexOf(".")); //unit

进入:

String unitatiTimpReal;
int index = formattedTimpR.indexOf(".");
if(index != -1) {
unitatiTimpReal = formattedTimpR.substring(0, index);
} else {
// formattedTimpR does not contain ".", handle this somehow
unitatiTimpReal = "";
}

我明白了:

W/dalvikvm( 3518): threadid=1: thread exiting with uncaught exception (group=0x40cf4960)
E/AndroidRuntime( 3518): FATAL EXCEPTION: main
E/AndroidRuntime( 3518): java.lang.NumberFormatException: Invalid double: "3,32"
E/AndroidRuntime( 3518):        at java.lang.StringToReal.invalidReal(StringToReal.java:63)
E/AndroidRuntime( 3518):        at java.lang.StringToReal.parseDouble(StringToReal.java:269)
E/AndroidRuntime( 3518):        at java.lang.Double.parseDouble(Double.java:295)
E/AndroidRuntime( 3518):        at com.example.android.rechner.MainActivityRustZ.displayTimeReal(MainActivityRustZ.java:118)
E/AndroidRuntime( 3518):        at com.example.android.rechner.MainActivityRustZ.access$200(MainActivityRustZ.java:12)
E/AndroidRuntime( 3518):        at com.example.android.rechner.MainActivityRustZ$1.onClick(MainActivityRustZ.java:29)
E/AndroidRuntime( 3518):        at android.view.View.performClick(View.java:4452)
E/AndroidRuntime( 3518):        at android.widget.Button.performClick(Button.java:148)
E/AndroidRuntime( 3518):        at android.view.View$PerformClick.run(View.java:18428)
E/AndroidRuntime( 3518):        at android.os.Handler.handleCallback(Handler.java:725)
E/AndroidRuntime( 3518):        at android.os.Handler.dispatchMessage(Handler.java:92)
E/AndroidRuntime( 3518):        at android.os.Looper.loop(Looper.java:176)
E/AndroidRuntime( 3518):        at android.app.ActivityThread.main(ActivityThread.java:5365)
E/AndroidRuntime( 3518):        at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime( 3518):        at java.lang.reflect.Method.invoke(Method.java:511)
E/AndroidRuntime( 3518):        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1102)
E/AndroidRuntime( 3518):        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:869)
E/AndroidRuntime( 3518):        at dalvik.system.NativeStart.main(Native Method)

第 118 行是:

        double minutereal = Double.parseDouble(zecimaleTimpReal);

@user13 我想要达到的目标: 在我为用户进行的其他一些计算中,根据它输入的数据,我实时转换工业时间(以分钟为单位的时间,小时(更准确地说是 60 分钟正常时间中的 100 分钟工业时间)

private void displayTimeReal(double timpreal) {

将 var vZeitD 初始化为 double,该用户需要输入以进行数学运算

Double vZeitD = Double.parseDouble(((EditText) findViewById(R.id.vZeit)).getText().toString());

将 var aufStZahl 初始化为 double,用户需要输入数学运算 - 在这里我可以修改整数,因为只允许输入整数

Double aufStZahl = Double.parseDouble(((EditText) findViewById(R.id.aufGesSt)).getText().toString());

计算所需时间(但以工业时间计算,即 100 分钟)

timpreal = aufStZahl / ((procente / vZeitD) * mitarbeiter);

我将答案分开(例如 12.75,其中 12 是小时,75 是分钟)并设置最大值和最小值显示小数 (2):

DecimalFormat formattr = new DecimalFormat();
formattr.setMaximumFractionDigits(2);
formattr.setMinimumFractionDigits(2);
String formattedTimpR = formattr.format(timpreal);

取工业小数换算

    String zecimaleTimpReal = formattedTimpR.substring(formattedTimpR.indexOf(".") + 1); //decimals

在完成数学运算后从结果数字中获取单位(单位保持原样显示

String unitatiTimpReal = formattedTimpR.substring(0, formattedTimpR.indexOf(".")); //unit

因为我必须再次进行数学运算,所以我不能使用字符串,我将小数转换为双精度并进行实时转换(60 分钟)

double minutereal = Double.parseDouble(zecimaleTimpReal);
minutereal = minutereal * 0.6;

我再次设置了分钟的最大小数位数(这样我就不会显示像 39.326562365 这样的响应)(例如:39.33 分钟而不是 39.326562365)

DecimalFormat format = new DecimalFormat();
format.setMaximumFractionDigits(2);
String formattedMinuteReal = format.format(minutereal);

向用户显示结果:

TextView timeTextView = (TextView) findViewById(R.id.rezultat_ore_real);
timeTextView.setText(getString(R.string.slaufz) + unitatiTimpReal + getString(R.string.shours) + formattedMinuteReal + getString(R.string.sminutes));
}

【问题讨论】:

  • 这是一个供你做的测试。将手机的语言设置为英语并运行您的应用程序。它会崩溃吗?现在设置为德语并运行它。它现在崩溃了吗?
  • 我的朋友,你闻到了一些东西:如果电话语言是德语,那么当需要做数学运算时,应用程序就会崩溃。如果电话语言是英语,它可以工作!我的电脑、我的手机和我的平板电脑都是英文的 :) 所以这可能是我的一个理由,我认为它可以工作......你也知道我能做些什么来解决这个问题吗?!

标签: java android android-logcat crash-dumps


【解决方案1】:

问题似乎是由于设备的Locale 的数字格式不同造成的。

当您使用DecimalFormat 对象时,小数点分隔符会因设备的语言设置而异。例如在英语中它是一个点,例如3.14159 但在德语中是逗号,例如3,14159.

这会导致问题,因为当您调用 indexOf() 时,您的代码总是在寻找点。

您可以通过多种方式解决此问题。一种方法是在创建 DecimalFormat 对象时指定语言环境,例如:

NumberFormat nf = NumberFormat.getNumberInstance(Locale.ENGLISH);
DecimalFormat formattr = (DecimalFormat)nf;

或者你可以只设置小数点分隔符:

DecimalFormat formattr = new DecimalFormat();
formattr.setDecimalSeparator('.');

但是,也许您希望它为德国用户以不同的方式显示。在这种情况下,您可以更改检查点的代码以检查正确的字符:

String zecimaleTimpReal = formattedTimpR.substring(formattedTimpR.indexOf("" + formattr.getDecimalFormatSymbols().getDecimalSeparator()) + 1);

添加到空字符串只是将字符转换为字符串的技巧。

请注意,还有一个“分组符号”,它也可以是逗号或点,例如123,456.78。您可能想要禁用分组,如下所示:

formattr.setGroupingUsed(false);

【讨论】:

  • 谢谢!!!今天,你是我的英雄! :D。我使用了最后一个建议,它会搜索正确的字符,因为我会将应用程序翻译成更多语言。再次感谢!!!!!!!
【解决方案2】:

问题出在109行中formattedTimpR不包含字符串"."

正因为如此,indexOf() 方法返回-1,所以基本上你调用的是substring(0, -1),因此出现错误。

我真的不知道你想达到什么目标,但你至少可以在致电substring() 之前进行检查,如下所示:

String unitatiTimpReal;
int index = formattedTimpR.indexOf(".");
if(index != -1) {
    unitatiTimpReal = formattedTimpR.substring(0, index);
} else {
    // formattedTimpR does not contain ".", handle this somehow
    unitatiTimpReal = "";
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-03-12
    • 2015-12-29
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多