Bugly - Bugly 基本使用( App 质量追踪平台)

Bugly

1、基本介绍
  1. Bugly 是腾讯推出的 App 质量追踪平台,专门帮开发者发现、定位、解决线上 App 的崩溃、卡顿等问题

    1. 官方网址:https://bugly.qq.com/v2/

    2. 开发文档:https://bugly.qq.com/docs/user-guide/instruction-manual-android/?v=1.0.0

  2. Bugly 优势在于免费、接入简单,适合中小团队快速搭建质量监控体系

2、场景实例
  1. 以前:用户反馈"点了一下就闪退了",开发者复现不了,无从下手

  2. 现在:Bugly 后台直接看到崩溃堆栈、机型、系统版本、操作路径,精准定位到某行代码

3、演示
(1)Setting
  1. 项目级 build.gradle
groovy 复制代码
android {
    defaultConfig {
        ndk {
            abiFilters 'armeabi'
        }
    }
}

...

dependencies {
    implementation 'com.tencent.bugly:crashreport:latest.release'
}
  1. AndroidManifest.xml
xml 复制代码
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
  1. proguard-rules.pro,避免混淆 Bugly

    -dontwarn com.tencent.bugly.**
    -keep public class com.tencent.bugly.**{*;}

(2)Activity
xml 复制代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="MainActivity"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <Button
        android:id="@+id/btn_simulate"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="模拟一个异常"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toBottomOf="@+id/textView" />
</androidx.constraintlayout.widget.ConstraintLayout>
java 复制代码
public class MainActivity extends AppCompatActivity {

    public static final String TAG = MainActivity.class.getSimpleName();

    private final String[] permissions = {
            Manifest.permission.INTERNET,
            Manifest.permission.ACCESS_NETWORK_STATE,
            Manifest.permission.ACCESS_WIFI_STATE
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        EdgeToEdge.enable(this);
        setContentView(R.layout.activity_main);
        ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
            Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
            v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
            return insets;
        });

        boolean needRequest = false;
        for (String permission : permissions) {
            int re = checkSelfPermission(permission);
            Log.i(TAG, "check ---------- 权限:" + permission + ",授权结果:" + re);
            if (re != PackageManager.PERMISSION_GRANTED) {
                needRequest = true;
                break;
            }
        }

        if (!needRequest) {
            next();
            return;
        }

        ActivityResultLauncher<String[]> activityResultLauncher = registerForActivityResult(
                new ActivityResultContracts.RequestMultiplePermissions(),
                o -> {
                    for (Map.Entry<String, Boolean> stringBooleanEntry : o.entrySet()) {
                        Log.i(TAG, "request ---------- 权限:" + stringBooleanEntry.getKey() + ",授权结果:" + stringBooleanEntry.getValue());
                    }

                    for (boolean granted : o.values()) {
                        if (!granted) {
                            Toast.makeText(this, "未授权权限", Toast.LENGTH_SHORT).show();
                            finish();
                            return;
                        }
                    }

                    next();
                }
        );
        activityResultLauncher.launch(permissions);
    }

    private void next() {
        CrashReport.initCrashReport(getApplicationContext(), "【App ID】", false);

        Button btnSimulate = findViewById(R.id.btn_simulate);

        btnSimulate.setOnClickListener(v -> {
            int i = 10 / 0;
        });
    }
}
相关推荐
唐青枫1 天前
Java JDBC 实战指南:从 Connection 到事务和连接池
java
一个做软件开发的牛马1 天前
MyBatis-Plus 从零实战:完整搭建可运行 Demo,BaseMapper 零 SQL、Wrapper 条件构造、分页插件与代码生成器详解
java·后端
用户3721574261351 天前
Java 处理 PDF 图片:提取 PDF 中的图片,并压缩 PDF 图片体积
java
逐光老顽童1 天前
Java 与 Kotlin 混合开发避坑指南:30 个真实案例实录
android·kotlin
用户3721574261351 天前
Java 打印 Word 文档:从基础打印到高级设置
java
爱勇宝1 天前
鸿蒙生态的下半场:开发者不只要能开发,还要能赚钱
android·前端·程序员
Yeyu2 天前
刷新一帧的艺术:invalidate / postInvalidate / postInvalidateOnAnimation全解析
android
用户3521802454752 天前
当 Prompt 学会"热更新":Spring Boot × Nacos3 AI 实战
java·spring boot·ai编程
潘潘潘2 天前
Android OTA 升级原理和流程介绍
android
东坡白菜2 天前
破局全栈:一个前端开发的Java入门实战记录(1)
java·全栈