java - 如果作为应用程序安装,则Android应用程序崩溃;如果通过Android Studio运行,则可以运行

标签 java android android-logcat crash-dumps

我在Android Studio 2.1.3中制作了一个应用程序,因为我已经在(不同的)仿真器和我自己的手机(小米Redmi Note 3,Android 5.0.2 LRX22G)中进行了测试。当我将应用程序交给几个 friend 进行测试时,他们都说了同样的事情:应用程序正常启动,他们选择了他们感兴趣的 Activity ,他们在文本字段中输入了数据,并在按下按钮的那一刻(假设(进行一些数学运算)以接收数学答案,该应用程序将停止并返回“主要” Activity 。

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),安装了该应用程序,并按预期工作!
现在我生气了! :)
我去找另一个 friend 做的事情与平板电脑相同,但是将文件发送给了三星Galaxy Grand Prime(看起来没有什么android版本),当我尝试做数学运算时崩溃了!他给了我一个三星Galaxy S3 Mini(GT-I8200N,Android 4.2.2),我通过BT发送APK,发生了同样的事情!

好吧,我开始高兴了,因为我随身带了手机,启用了开发人员选项,通过USB调试,将手机连接到计算机,使用Android Studio加载应用程序,然后猜猜是什么?!它的工作没有任何错误,折痕或其他任何东西!
我将手机重置为出厂设置,希望强制执行此错误...我通过BT发送了文件,并且...工作正常!尽管重置后手机没有连接到我的计算机(和Android Studio),但没有任何错误或崩溃。因为连接到PC不会失败,所以我看不到问题,并且无法在调试日志中寻找答案:(

我需要提及的是,在我测试过该应用程序的其他手机上启用了开发人员选项,但唯一的区别是它们均未连接到我的计算机和Android Studio!

你们有没有遇到过这个问题?我已经在Google friend 上搜索了几天类似的内容,但未找到任何内容。

感谢您的耐心等待和解答!

最好的问候,罗伯特·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行:
字符串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));
}

最佳答案

问题似乎是由于数字根据设备的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);

添加到空字符串只是将字符从char转换为String的一个技巧。

请注意,还有一个“分组符号”,它也可以是逗号或点。 123,456.78。您可能要禁用分组,如下所示:
formattr.setGroupingUsed(false);

关于java - 如果作为应用程序安装,则Android应用程序崩溃;如果通过Android Studio运行,则可以运行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39316379/

相关文章:

android - 在 Android 应用中获取 logcat 时需要获取应用名称

android - 如何增加 Logcat 中的行数(日志级别)

java - (Eclipse) 无法显示标记/nullpointerException googlemaps api android

android - 强制关闭不会退出 android 应用程序,而是打开一个处于错误状态的先前 Activity

android - Activity 和 Fragment 生命周期和方向变化

java - 如何在 Eclipse 中自动重命名 Java 方法?

java - 带有 libgdx 的 Android 应用程序在一段时间后被推送到后台

java - Twitter4J 抛出异常 java.lang.NoSuchMethodError :

Java EE 将 Controller 类与 DAO 类连接

java - onItemClick 中的 Android Studio 警报对话框