使用ADB工具分析Android应用崩溃原因:以闪动校园为例

news2025/4/15 14:25:31

使用adb工具分析模拟器或手机里app出错原因以闪动校园为例

使用ADB工具分析Android应用崩溃原因:以闪动校园为例

前言

应用崩溃是移动开发中常见的问题,尤其在复杂的Android生态系统中,找出崩溃原因可能十分棘手。本文将以流行的校园应用"闪动校园"为例,详细介绍如何利用Android Debug Bridge (ADB)工具分析应用崩溃原因,从而快速定位和解决问题。

1. ADB工具简介

Android Debug Bridge (ADB)是Android SDK中的一个强大命令行工具,它允许开发者与连接的Android设备或模拟器进行通信。通过ADB,我们可以:

  • 安装/卸载应用
  • 传输文件
  • 运行shell命令
  • 收集日志信息
  • 调试应用

ADB的基本架构包括三个组件:

客户端 ⟷ 服务器 ⟷ 守护进程(adbd)
(电脑)    (电脑)     (设备)

2. 准备工作

2.1 安装ADB工具

如果你已经安装了Android Studio,ADB工具已包含在SDK中。你也可以单独下载平台工具:

# Windows用户
# 下载platform-tools后,添加到环境变量Path中
echo %PATH%
setx PATH "%PATH%;C:\path\to\platform-tools"

# Linux/Mac用户
echo $PATH
export PATH=$PATH:/path/to/platform-tools

2.2 验证设备连接

首先,我们需要确认设备已经正确连接并被识别:

adb devices

正常输出应该如下所示:

List of devices attached
emulator-5554    device    # 模拟器
HSKW7N8012345    device    # 物理设备

如果设备显示为unauthorized,需要在设备上确认USB调试授权。

2.3 开启开发者选项

在Android设备上:

  1. 进入设置 > 关于手机
  2. 连续点击版本号7次,开启开发者选项
  3. 返回设置页面,进入开发者选项
  4. 开启USB调试

3. 收集应用崩溃日志

3.1 清除现有日志

在开始分析前,最好先清除现有的日志,以避免干扰:

adb logcat -c

3.2 查找应用包名

要针对特定应用过滤日志,我们需要知道其包名。有几种方法可以找到包名:

方法1:通过应用名称查找

adb shell pm list packages | findstr "闪动"

可能的输出:

package:com.huachenjie.shandong_school

方法2:获取当前运行的应用包名

adb shell dumpsys window | findstr "mCurrentFocus"

方法3:编程方式获取包名

// 在应用代码中
String packageName = getApplicationContext().getPackageName();
Log.d("AppInfo", "Package name: " + packageName);

3.3 运行应用并收集崩溃日志

现在我们知道闪动校园的包名是com.huachenjie.shandong_school,可以针对性地收集日志:

# 过滤特定应用的日志
adb logcat | findstr "com.huachenjie.shandong_school"

# 或者只查看错误级别的日志
adb logcat *:E

让应用崩溃,然后查看日志输出。为了方便分析,我们可以将日志保存到文件:

# 在Windows下保存日志到桌面
adb logcat > C:\Users\用户名\Desktop\crash_log.txt

4. 日志分析与错误定位

4.1 理解Logcat输出格式

Logcat的基本输出格式如下:

日期 时间 PID-TID/包名 优先级/标签: 消息

例如:

05-15 14:30:22.123 1234-5678/com.huachenjie.shandong_school E/AndroidRuntime: FATAL EXCEPTION: main

其中:

  • 05-15 14:30:22.123:日期和时间
  • 1234-5678:进程ID和线程ID
  • com.huachenjie.shandong_school:包名
  • E:错误级别(Error)
  • AndroidRuntime:日志标签
  • FATAL EXCEPTION: main:错误消息

4.2 常见的崩溃类型及分析

空指针异常 (NullPointerException)
E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.huachenjie.shandong_school, PID: 12345
    java.lang.NullPointerException: Attempt to invoke virtual method 'java.lang.String com.huachenjie.shandong_school.model.User.getName()' on a null object reference
    at com.huachenjie.shandong_school.ui.ProfileActivity.updateUI(ProfileActivity.java:120)
    at com.huachenjie.shandong_school.ui.ProfileActivity.onCreate(ProfileActivity.java:65)

分析:在ProfileActivity.java的第120行,尝试调用user.getName(),但user对象为null。

解决方案:在调用前添加空值检查:

if (user != null) {
    String name = user.getName();
    // 处理name
} else {
    // 处理user为null的情况
    Log.e("ProfileActivity", "User object is null");
    // 可能需要重新获取用户数据或显示错误信息
}
数组索引越界 (ArrayIndexOutOfBoundsException)
E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.huachenjie.shandong_school, PID: 12345
    java.lang.ArrayIndexOutOfBoundsException: length=5; index=5
    at com.huachenjie.shandong_school.util.DataProcessor.processData(DataProcessor.java:78)

分析:在DataProcessor.java的第78行,尝试访问数组索引5,但数组长度只有5(索引应为0-4)。

解决方案:确保索引在有效范围内:

if (index < array.length) {
    // 安全地访问array[index]
} else {
    Log.e("DataProcessor", "Index out of bounds: " + index + " for array length: " + array.length);
}
类型转换异常 (ClassCastException)
E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.huachenjie.shandong_school, PID: 12345
    java.lang.ClassCastException: com.huachenjie.shandong_school.model.Staff cannot be cast to com.huachenjie.shandong_school.model.Student
    at com.huachenjie.shandong_school.ui.CourseActivity.displayStudentInfo(CourseActivity.java:156)

分析:在CourseActivity.java的第156行,尝试将Staff对象强制转换为Student对象。

解决方案:使用instanceof检查类型再转换:

if (user instanceof Student) {
    Student student = (Student) user;
    // 处理学生对象
} else if (user instanceof Staff) {
    Staff staff = (Staff) user;
    // 处理职工对象
} else {
    Log.e("CourseActivity", "Unknown user type: " + user.getClass().getName());
}

4.3 ANR (Application Not Responding) 分析

除了崩溃,应用还可能出现ANR问题。我们可以通过以下命令收集ANR信息:

adb pull /data/anr/traces.txt ./anr_analysis.txt

典型的ANR日志示例:

----- pid 12345 at 2023-05-15 14:30:22 -----
Cmd line: com.huachenjie.shandong_school

DALVIK THREADS (40):
"main" prio=5 tid=1 Blocked
  | group="main" sCount=1 dsCount=0 obj=0x73467890 self=0x7f98765430
  | sysTid=12345 nice=0 cgrp=default sched=0/0 handle=0x7f87654320
  | state=S schedstat=( 0 0 0 ) utm=1234 stm=567 core=0 HZ=100
  | stack=0x7ff1234000-0x7ff1256000 stackSize=8MB
  | held mutexes=
  at com.huachenjie.shandong_school.database.DatabaseHelper.getStudentData(DatabaseHelper.java:230)
  - waiting to lock <0x87654320> (a java.lang.Object) held by thread 23
  at com.huachenjie.shandong_school.ui.MainActivity$loadData(MainActivity.java:178)
  at com.huachenjie.shandong_school.ui.MainActivity.onCreate(MainActivity.java:75)

分析:主线程在等待一个被线程23持有的锁,这导致了UI线程阻塞,引发ANR。

解决方案:避免在主线程进行数据库操作,应该使用异步处理:

// 不要在主线程中直接操作数据库
new Thread(new Runnable() {
    @Override
    public void run() {
        final List<Student> students = databaseHelper.getStudentData();
        
        // 使用Handler或runOnUiThread更新UI
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                updateUI(students);
            }
        });
    }
}).start();

// 或使用AsyncTask(已弃用但仍常见)
private class LoadDataTask extends AsyncTask<Void, Void, List<Student>> {
    @Override
    protected List<Student> doInBackground(Void... params) {
        return databaseHelper.getStudentData();
    }
    
    @Override
    protected void onPostExecute(List<Student> students) {
        updateUI(students);
    }
}

// 较新的选择是使用协程(Kotlin)
lifecycleScope.launch(Dispatchers.IO) {
    val students = databaseHelper.getStudentData()
    withContext(Dispatchers.Main) {
        updateUI(students)
    }
}

5. 高级调试技巧

5.1 使用自定义过滤器

可以同时使用多个过滤条件优化日志查看体验:

# 查看特定应用的特定标签
adb logcat -v threadtime ActivityManager:I MyApp:D *:S

# 使用正则表达式过滤
adb logcat | findstr -r "Exception|Error|FATAL"

5.2 利用Android Studio分析崩溃

除了命令行工具,Android Studio也提供了强大的日志分析功能:

  1. 在Android Studio中,打开Logcat窗口
  2. 连接设备并选择应用进程
  3. 使用过滤器和搜索功能定位错误

5.3 使用自定义日志记录器

在闪动校园应用中实现一个自定义的日志记录器,可以大大提高调试效率:

public class Logger {
    private static final String TAG = "闪动校园";
    private static final boolean DEBUG = BuildConfig.DEBUG;
    
    public static void d(String message) {
        if (DEBUG) {
            Log.d(TAG, buildLogMsg(message));
        }
    }
    
    public static void e(String message, Throwable e) {
        Log.e(TAG, buildLogMsg(message), e);
        
        // 在开发版本中,可以将错误信息保存到文件
        if (DEBUG) {
            saveErrorToFile(message, e);
        }
    }
    
    private static String buildLogMsg(String message) {
        StackTraceElement element = Thread.currentThread().getStackTrace()[4];
        String className = element.getClassName();
        className = className.substring(className.lastIndexOf('.') + 1);
        
        return String.format("%s.%s(%s:%d): %s",
                className, 
                element.getMethodName(),
                element.getFileName(),
                element.getLineNumber(),
                message);
    }
    
    private static void saveErrorToFile(String message, Throwable e) {
        try {
            File logDir = new File(Environment.getExternalStorageDirectory(), "闪动校园/logs");
            if (!logDir.exists()) {
                logDir.mkdirs();
            }
            
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss", Locale.CHINA);
            String fileName = "error-" + sdf.format(new Date()) + ".txt";
            File logFile = new File(logDir, fileName);
            
            PrintWriter writer = new PrintWriter(new FileWriter(logFile));
            writer.println("错误信息: " + message);
            writer.println("时间: " + new Date().toString());
            writer.println("\n堆栈跟踪:");
            e.printStackTrace(writer);
            writer.close();
        } catch (Exception ex) {
            Log.e(TAG, "保存错误日志失败", ex);
        }
    }
}

在应用代码中使用:

try {
    // 可能会抛出异常的代码
    User user = getUserFromServer();
    processUserData(user);
} catch (Exception e) {
    Logger.e("获取用户数据失败", e);
    // 处理错误,例如显示友好的错误消息
    showErrorDialog("无法连接到服务器,请稍后再试");
}

5.4 监控应用性能

除了崩溃分析,我们还可以使用ADB监控应用性能:

# 监控内存使用
adb shell dumpsys meminfo com.huachenjie.shandong_school

# 监控CPU使用
adb shell top -n 1 | findstr "com.huachenjie.shandong_school"

# 监控电池使用
adb shell dumpsys batterystats com.huachenjie.shandong_school

6. 自动化崩溃分析

6.1 使用脚本自动收集和分析日志

可以创建批处理脚本或Shell脚本自动化日志收集过程:

Windows批处理脚本 (collect_logs.bat):

@echo off
echo 清除现有日志...
adb logcat -c

echo 开始收集日志,按Ctrl+C停止...
adb logcat -v threadtime > "%USERPROFILE%\Desktop\app_log_%date:~0,4%%date:~5,2%%date:~8,2%_%time:~0,2%%time:~3,2%%time:~6,2%.txt"

echo 日志已保存到桌面。

Linux/Mac Shell脚本 (collect_logs.sh):

#!/bin/bash
echo "清除现有日志..."
adb logcat -c

echo "开始收集日志,按Ctrl+C停止..."
adb logcat -v threadtime > ~/Desktop/app_log_$(date +%Y%m%d_%H%M%S).txt

echo "日志已保存到桌面。"

6.2 集成Crash报告工具

对于生产环境,建议集成专业的崩溃报告工具,如Firebase Crashlytics:

// 在app/build.gradle中添加依赖
dependencies {
    // Firebase Crashlytics
    implementation 'com.google.firebase:firebase-crashlytics:18.3.7'
    implementation 'com.google.firebase:firebase-analytics:21.3.0'
}

初始化代码:

public class MyApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        
        // 初始化Firebase
        FirebaseApp.initializeApp(this);
        
        // 开发环境下禁用崩溃报告
        if (BuildConfig.DEBUG) {
            FirebaseCrashlytics.getInstance().setCrashlyticsCollectionEnabled(false);
        }
        
        // 设置用户信息以便更好地分析崩溃
        try {
            User currentUser = UserManager.getInstance().getCurrentUser();
            if (currentUser != null) {
                FirebaseCrashlytics.getInstance().setUserId(currentUser.getId());
                
                // 添加自定义键值对
                FirebaseCrashlytics.getInstance().setCustomKey("user_type", currentUser.getType());
                FirebaseCrashlytics.getInstance().setCustomKey("school_id", currentUser.getSchoolId());
            }
        } catch (Exception e) {
            Log.e("Application", "设置用户信息失败", e);
        }
    }
}

7. 闪动校园案例分析

以下是一个实际案例,说明如何使用ADB分析闪动校园应用的一个具体崩溃问题:

问题描述

用户报告在打开"课程表"页面时应用崩溃。

分析过程

步骤1: 收集崩溃日志

adb logcat -c
adb logcat | findstr "com.huachenjie.shandong_school" > crash_log.txt

步骤2: 日志分析

崩溃日志示例:

05-15 10:23:45.678 12345-12345/com.huachenjie.shandong_school E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.huachenjie.shandong_school, PID: 12345
    java.lang.IllegalStateException: Could not execute method for android:onClick
    at androidx.appcompat.app.AppCompatViewInflater$DeclaredOnClickListener.onClick(AppCompatViewInflater.java:414)
    ...
    Caused by: java.lang.reflect.InvocationTargetException
    at java.lang.reflect.Method.invoke(Native Method)
    ...
    Caused by: java.lang.NullPointerException: Attempt to invoke interface method 'java.util.List<com.huachenjie.shandong_school.model.Course> com.huachenjie.shandong_school.api.CourseService.getCourses(java.lang.String)' on a null object reference
    at com.huachenjie.shandong_school.ui.CourseActivity.loadCourses(CourseActivity.java:145)
    at com.huachenjie.shandong_school.ui.CourseActivity.onButtonClick(CourseActivity.java:98)

步骤3: 定位根本原因

分析表明在CourseActivity.java第145行,应用尝试调用courseService.getCourses(semesterId),但courseService对象为null。

步骤4: 创建修复方案

// 原始代码(有问题)
private void loadCourses(String semesterId) {
    List<Course> courses = courseService.getCourses(semesterId);
    displayCourses(courses);
}

// 修复后的代码
private void loadCourses(String semesterId) {
    if (courseService == null) {
        // 懒加载初始化服务
        courseService = ServiceLocator.getCourseService();
        
        // 如果仍然为null,优雅处理
        if (courseService == null) {
            Log.e("CourseActivity", "CourseService初始化失败");
            Toast.makeText(this, "无法加载课程数据,请重启应用", Toast.LENGTH_LONG).show();
            return;
        }
    }
    
    try {
        List<Course> courses = courseService.getCourses(semesterId);
        if (courses != null) {
            displayCourses(courses);
        } else {
            showEmptyCourseView();
        }
    } catch (Exception e) {
        Logger.e("加载课程数据失败", e);
        Toast.makeText(this, "加载课程数据失败: " + e.getMessage(), Toast.LENGTH_LONG).show();
        showErrorView();
    }
}

8. 总结

通过本文,我们详细介绍了如何使用ADB工具分析Android应用崩溃问题,以闪动校园应用为例。主要内容包括:

  1. ADB基础知识与设置
  2. 收集应用崩溃日志的方法
  3. 常见崩溃类型分析与解决
  4. ANR问题的识别与处理
  5. 高级调试技巧
  6. 自动化崩溃分析工具
  7. 实际案例分析与解决

掌握这些技能将帮助开发者更快地定位和解决应用问题,提高应用稳定性和用户体验。

参考资料

风车模拟器相关

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2334562.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言中while的相关题目

一、题目引入 以下程序中,while循环的循环次数是多少次? 二、代码分析 首先要明确的一点 while循环是当循环条件为真 就会一直循环 不会停止 while中i是小于10的 说明i可以取到0 1 2 3 4 5 6 7 8 9 进入第一个if判断i小于1为真时执行continue i0是为真的 执行continue 后…

「Unity3D」图片导入选项取消Read/Write,就无法正确显示导入大小,以及Addressable打包无法正确显示的问题

如果在Edit -> Project Settings -> Editor中的“Load texture data on demand”勾选&#xff0c;就会让图片导入设置中&#xff0c;不勾选Read/Write&#xff0c;就无法正确显示纹理的大小数字。 更进一步的问题是&#xff0c;使用Addressable打包的时候&#xff0c; 如…

Xcode为不同环境配置不同的环境变量

一般有三种方式&#xff1a; 一、通过多Target 二、通过scheme,也就是多configurations 三、通过.xcconfig文件 先来看第二种方式&#xff1a;通过scheme,也就是多configurations,包括自定义User-settings 第一步&#xff1a;增加configurations,Xcode默认为我们生成了…

阿里通义实验室发布图片数字人项目LAM,实现高保真重建

简介 LAM项目结合了3D Gaussian Splatting&#xff08;高斯点云渲染&#xff09;和大规模预训练模型的优势&#xff0c;解决了传统头部重建方法效率低、依赖多数据的痛点。其背景源于AI生成内容&#xff08;AIGC&#xff09;领域对实时、高保真3D头像生成的需求&#xff0c;尤其…

镜像端口及观察端口的配置

配好路由器的各个接口的IP PC1ping PC3的IP&#xff0c;在路由器中抓2/0/0端口的包&#xff0c;可观察到无结果 输入observe-port interface g 2/0/0 命令配置观察端口 输入mirror to observe-port both命令 &#xff08;其中both表示接收来去的数据包&#xff0c;inboun…

STM32——I2C通讯(软件模拟)

I2C概念 I2C:Inter-Integrated Circuit&#xff08;内部集成电路&#xff09; Philps公司80年代初期开发的&#xff0c;引脚少&#xff0c;硬件实现简单&#xff0c;可扩展性广泛地使用在系统内多个集成电路&#xff08;IC&#xff09;间的低速通讯 简单的双向两线制总线协议…

JetBrains Terminal 又发布新架构,Android Studio 将再次迎来新终端

不到一年的时间&#xff0c;JetBrains 又要对 Terminal 「大刀阔斧」&#xff0c;本次发布的新终端是重构后的全新的架构&#xff0c;而上一次终端大调整还是去年 8 月的 v2024.2 版本&#xff0c;并且在「Android Studio Ladybug | 2024.2.1」也被引入。 不知道你们用不用内置…

论文:Generalized Category Discovery with Large Language Models in the Loop

论文下载地址&#xff1a;Generalized Category Discovery with Large Language Models in the Loop - ACL Anthology 1、研究背景 尽管现代机器学习系统在许多任务上取得了优异的性能&#xff0c;绝大多数都遵循封闭世界的设置&#xff0c;假设训练和测试数据来自同一组预定义…

第十六届蓝桥杯 省赛C/C++ 大学B组

编程题目现在在洛谷上都可以提交了。 未完待续&#xff0c;写不动了。 C11 编译命令 g A.cpp -o A -Wall -lm -stdc11A. 移动距离 本题总分&#xff1a;5 分 问题描述 小明初始在二维平面的原点&#xff0c;他想前往坐标 ( 233 , 666 ) (233, 666) (233,666)。在移动过程…

【计网】网络交换技术之分组交换(复习自用,重要1)

复习自用的&#xff0c;处理得比较草率&#xff0c;复习的同学或者想看基础的同学可以看看&#xff0c;大佬的话可以不用浪费时间在我的水文上了 另外两种交换技术可以直接点击链接访问相关笔记&#xff1a; 电路交换 报文交换 一、分组交换的定义 1.定义 分组交换&#x…

解密CHASE-SQL和XiYan-SQL多智能体AI如何最终实现TEXT2SQL的突破

想象一个世界,无论技术背景如何,任何人都能轻松查询海量数据库、挖掘深层洞察。比如:“我想知道安徽地区最畅销电子产品的第三季度销售额?”——只需一句话。“去年营销支出与客户获取成本之间的相关性如何?”——像聊天一样输入问题。这就是Text-to-SQL的承诺:将人类语言…

思考力提升的黄金标准:广度、深度与速度的深度剖析

文章目录 引言一、广度的拓展&#xff1a;构建多元知识网络1.1 定义与重要性1.2 IT技术实例与提升策略小结&#xff1a;构建多元知识网络&#xff0c;提升IT领域思考力广度 二、深度的挖掘&#xff1a;追求知识的精髓2.1 定义与重要性2.2 IT技术实例与提升策略小结&#xff1a;…

web自动化:下拉选择框、弹出框、滚动条的操作

web自动化&#xff1a;下拉选择框、弹出框、滚动条的操作 一、下拉选择框 1、导包 from selenium.webdriver.support.select inport Select 2、实例化对象 Select(element) 3、常用方法 通过option索引来定位&#xff0c;从0开始&#xff1a;select_by_index(index)通过…

数字人:打破次元壁,从娱乐舞台迈向教育新课堂(4/10)

摘要&#xff1a;数字人正从娱乐领域的璀璨明星跨界到教育领域的智慧导师&#xff0c;展现出无限潜力。从虚拟偶像、影视游戏到直播短视频&#xff0c;数字人在娱乐产业中大放异彩&#xff0c;创造巨大商业价值。在教育领域&#xff0c;数字人助力个性化学习、互动课堂和虚拟实…

互联网三高-数据库高并发之分库分表ShardingJDBC

1 ShardingJDBC介绍 1.1 常见概念术语 ① 数据节点Node&#xff1a;数据分片的最小单元&#xff0c;由数据源名称和数据表组成 如&#xff1a;ds0.product_order_0 ② 真实表&#xff1a;再分片的数据库中真实存在的物理表 如&#xff1a;product_order_0 ③ 逻辑表&#xff1a…

Android游戏逆向工程全面指南

文章目录 第一部分&#xff1a;基础概念与环境搭建1.1 游戏逆向工程概述1.2 法律与道德考量1.3 开发环境准备基础工具集&#xff1a;环境配置示例&#xff1a; 第二部分&#xff1a;静态分析技术2.1 APK反编译与资源提取使用Apktool解包&#xff1a;关键文件分析&#xff1a; 2…

antv x6使用(支持节点排序、新增节点、编辑节点、删除节点、选中节点)

项目需要实现如下效果流程图&#xff0c;功能包括节点排序、新增节点、编辑节点、删除节点、选中节点等 html部分如下&#xff1a; <template><div class"MindMapContent"><el-button size"small" click"addNode">新增节点&…

榕壹云在线商城系统:基于THinkPHP+ Mysql+UniApp全端适配、高效部署的电商解决方案

项目背景&#xff1a;解决多端电商开发的痛点 随着移动互联网的普及和用户购物习惯的碎片化&#xff0c;传统电商系统面临以下挑战&#xff1a; 1. 多平台适配成本高&#xff1a;需要同时开发App、小程序、H5等多端应用&#xff0c;重复开发导致资源浪费。 2. 技术依赖第三方…

Android studio打包uniapp插件

一.参考资料与环境准备 原生工程配置需要使用到Android studio和HbuilderX 当前测试的as版本-20240301,下载地址&#xff1a;HbuilderX版本&#xff1a;4.36 二.插件创建流程 1.导入下载的UniPlugin-Hello-AS工程&#xff08;下载地址见参考资料&#xff09; 2.生成jks证书…

App Cleaner Pro for Mac 中 Mac软件卸载工具

App Cleaner Pro for Mac 中 Mac软件卸载工具 一、介绍 App Cleaner & Uninstaller Pro Mac破解&#xff0c;是一款Mac软件卸载工具&#xff0c;残余垃圾清除工具&#xff01;可以卸载应用程序或只删除不需要的服务文件&#xff0c;甚至可以删除以前删除的应用程序中的文…