收集崩溃信息上传服务器

直接贴完整代码,详细介绍,里面有做介绍

public class ExceptionCrashHanlder implements UncaughtExceptionHandler {
    private static final String TAG = ExceptionCrashHanlder.class.getSimpleName();
    private static ExceptionCrashHanlder instance;
    private Context mContext;
    private UncaughtExceptionHandler mDefaultException;

    /**
     * 双重校验锁,减少并发
     */
    public static ExceptionCrashHanlder getInstance() {
        if (instance == null) {
            synchronized (ExceptionCrashHanlder.class) {
                if (instance == null) {
                    instance = new ExceptionCrashHanlder();
                }
            }
        }
        return instance;
    }

    public void init(Context context) {
        mContext = context;
        //设置全局变量是本类
        Thread.currentThread().setUncaughtExceptionHandler(this);
        //获取系统默认的
        mDefaultException = Thread.currentThread().getDefaultUncaughtExceptionHandler();
    }

    @Override
    public void uncaughtException(Thread t, Throwable e) {
        //全局异常捕获处理
        Log.e(TAG, "捕获异常了");
        //1.奔溃的详细信息
        //2.应用信息 包名 版本号
        //3.手机信息
        //4.上传问题,不在这里

        //写入到本地文件 ex 当前版本 手机信息
        String crashFileName  = saveInfoToSD(e);
        //缓存崩溃日志文件
        cacheCrashFile(crashFileName);
        //让默认系统处理
        mDefaultException.uncaughtException(t, e);
    }

    /**
     * 缓存文件
     */
    private void cacheCrashFile(String fileName) {
        SharedPreferences sp = mContext.getSharedPreferences("crash", Context.MODE_PRIVATE);
        sp.edit().putString("CRASH_FILE_NAME", fileName).commit();
    }
    /**
     * 获取本地文件
     */
    public File getCrashFile() {
        String crashFileName = mContext.getSharedPreferences("crash",
                Context.MODE_PRIVATE).getString("CRASH_FILE_NAME", "");
        return new File(crashFileName);
    }

    /**
     * 保存获取的 软件信息,设备信息和出错信息保存在SDcard中
     */
    private String saveInfoToSD(Throwable e) {
        String fileName = null;
        StringBuffer sb = new StringBuffer();
        for (Map.Entry<String, String> entry : obtainSimpleInfo(mContext).entrySet()) {
            String key = entry.getKey();
            String value = entry.getValue();
            sb.append(key).append(" = ").append(value).append("\n");
        }
        sb.append(obtainExceptionInfo(e));

        //将sb拼起来的保存一个文件
        if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
            /**
             * 比如说要在temp目录下建立一个test.txt文件,在Windows下应该这么写:
             File file1 = new File ("C:\tmp\test.txt");
             在Linux下则是这样的:
             File file2 = new File ("/tmp/test.txt");
             如果要考虑跨平台,则最好是这么写:
             File myFile = new File("C:" + File.separator + "tmp" + File.separator, "test.txt");
             */
            File file = new File(mContext.getFilesDir() + File.separator + "crash"
                    + File.separator);
            //先删除之前的异常信息
            if (file.exists()) {
                deleteDir(file);
            }
            //再重新创建新的文件夹
            if (!file.exists()) {
                file.mkdir();
            }
            try {
                fileName = file.toString() + File.separator + getAssignTime("yyyy_MM_dd_HH_mm") + ".txt";
                FileOutputStream fos = new FileOutputStream(fileName);
                fos.write(sb.toString().getBytes());
                fos.flush();//缓冲区的数据强行输出
                fos.close();
            } catch (Exception e1) {
                e1.printStackTrace();
            }
        }
        return fileName;

    }

    /**
     * 获取时间
     */
    private String getAssignTime(String dateFormat) {
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
            DateFormat format = new SimpleDateFormat(dateFormat);
            long currentTime = System.currentTimeMillis();
            return format.format(currentTime);
        }
        return null;
    }

    /**
     * 递归删除目录下的所有文件及子目录下所有文件
     */


    private boolean deleteDir(File file) {
        if (file.isDirectory()) {
            String[] list = file.list();
            // 递归删除目录中的子目录下
            for (String s : list) {
                boolean success = deleteDir(new File(file, s));
                if (!success) {
                    return false;
                }
            }
        }
        // 目录此时为空,可以删除
        return file.delete();
    }

    /**
     * 获取系统未捕捉的错误信息
     */
    private String obtainExceptionInfo(Throwable e) {
        StringWriter stringWriter = new StringWriter();
        PrintWriter printWriter = new PrintWriter(stringWriter);
        e.printStackTrace(printWriter);
        printWriter.close();
        return stringWriter.toString();
    }

    /**
     * 获取一些简单的信息,软件版本,手机版本,型号等信息存放在HashMap中
     */
    private HashMap<String, String> obtainSimpleInfo(Context context) {
        HashMap<String, String> map = new HashMap<>();
        PackageManager manager = context.getPackageManager();
        PackageInfo packageInfo = null;
        try {
            packageInfo = manager.getPackageInfo(context.getPackageName()
                    , PackageManager.GET_ACTIVITIES);

        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }
        map.put("versionName", packageInfo.versionName);
        map.put("versionCode", "" + packageInfo.versionCode);
        map.put("MODEL", Build.MODEL);//版本
        map.put("SDK_INT", Build.VERSION.SDK_INT + "");//版本号
        map.put("PRODUCT", Build.PRODUCT); //手机制造商
        map.put("MOBLE_INFO", getMobileInfo()); //手机信息
        return map;
    }

    //获取手机信息
    public String getMobileInfo() {
        StringBuffer sb = new StringBuffer();
        try {
            Field[] fields = Build.class.getDeclaredFields();
            for (Field field : fields) {
                field.setAccessible(true);//可以访问所有的信息
                String name = field.getName();
                String value = field.get(null).toString();
                sb.append(name + "=" + value);
                sb.append("\n");
            }
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return sb.toString();
    }
}

测试

 try {
            File file = ExceptionCrashHanlder.getInstance().getCrashFile();
            if (file.exists()) {
                InputStreamReader reader = new InputStreamReader(new FileInputStream(file));
                int len = -1;
                char[] buf = new char[1024];
                while ((len = reader.read(buf)) != -1) {
                    String s = new String(buf);
                    Log.e("TAG", s);
                    //上传服务器这里就不做了
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

//        int i = 5 / 0;
<think>好的,用户想了解如何在Flutter中上传崩溃日志到服务器。首先,我得回忆一下Flutter处理异常的方法。记得Flutter有一个全局的异常捕获机制,可以通过重写ErrorWidget或者使用FlutterError.onError来捕获错误。不过可能还需要结合Zone来处理异步错误。 接下来,需要考虑如何捕获未处理的异常。Dart中的runZonedGuarded可能是个关键点,因为这样可以捕获到Zone内的所有未处理异常。同时,Flutter自身的错误可能需要通过FlutterError.onError来捕获,然后将两者整合起来,确保所有错误都能被处理。 然后,捕获到异常后需要记录日志。日志的内容应该包括错误信息、堆栈跟踪,可能还需要设备信息,比如操作系统版本、设备型号、应用版本等。这时候可能需要用到一些包,比如device_info_plus来获取设备信息,还有package_info_plus来获取应用版本。 接下来是上传服务器的问题。用户需要自己实现一个上传的逻辑,可能用http包发送POST请求到指定的API接口。这里需要注意处理网络请求的异常,避免因为上传失败导致日志丢失,可能需要将日志缓存到本地,等有网络时再重试。 另外,还要考虑日志的存储。可以使用shared_preferences或者文件存储来临时保存日志,特别是当上传失败时,下次启动应用时可以重新尝试上传。 可能还需要考虑错误日志的格式,方便服务器解析。比如将日志转换为JSON格式,包含时间戳、错误信息、堆栈跟踪、设备信息等字段。 然后,用户可能会有不同的需求,比如是否需要实时上传,还是批量上传。如果是实时上传,可能会影响用户体验,特别是当用户网络不好时。所以可能需要采用缓存机制,延迟上传。 还有,测试这部分功能也很重要。开发者需要模拟各种崩溃情况,确保日志能够正确捕获并上传。可能需要手动抛出异常来测试流程是否正常。 另外,需要注意权限问题,比如在Android和iOS上需要网络权限,以及在Android中可能还需要访问设备信息的权限,需要在配置文件中声明。 可能还需要考虑用户隐私,确保上传的日志不包含敏感信息,或者对日志中的数据进行脱敏处理。 最后,总结步骤:设置全局异常捕获,收集日志信息,存储日志,上传服务器,处理上传失败的情况,确保日志最终能成功送达服务器。</think>在Flutter中实现崩溃日志上传服务器,可通过以下步骤完成: --- ### 一、捕获全局异常 #### 1. 初始化全局异常捕获 ```dart void main() { runZonedGuarded(() { WidgetsFlutterBinding.ensureInitialized(); FlutterError.onError = (details) { // 处理Flutter框架层错误 _reportError(details.exception, details.stack); }; runApp(MyApp()); }, (error, stack) { // 处理异步错误 _reportError(error, stack); }); } ``` #### 2. 错误上报方法 ```dart Future<void> _reportError(dynamic error, StackTrace stack) async { // 获取设备信息(需添加依赖包) final deviceInfo = await DeviceInfoPlugin().deviceInfo; final packageInfo = await PackageInfo.fromPlatform(); // 构造日志对象 final logData = { 'timestamp': DateTime.now().toIso8601String(), 'error': error.toString(), 'stack_trace': stack.toString(), 'platform': Platform.operatingSystem, 'device_info': deviceInfo.data, 'app_version': packageInfo.version, }; // 上传服务器 await _uploadToServer(logData); } ``` --- ### 二、实现上传逻辑 #### 1. 添加HTTP依赖 ```yaml dependencies: http: ^0.13.4 ``` #### 2. 上传方法示例 ```dart Future<void> _uploadToServer(Map<String, dynamic> logData) async { final url = Uri.parse('https://your-server.com/api/crash-report'); try { final response = await http.post( url, headers: {'Content-Type': 'application/json'}, body: jsonEncode(logData), ); if (response.statusCode != 200) { _saveLogLocally(logData); // 上传失败时本地存储 } } catch (e) { _saveLogLocally(logData); // 网络异常时存储 } } ``` --- ### 三、本地缓存未发送日志 #### 1. 使用shared_preferences存储 ```dart Future<void> _saveLogLocally(Map<String, dynamic> log) async { final prefs = await SharedPreferences.getInstance(); final cachedLogs = prefs.getStringList('crash_logs') ?? []; cachedLogs.add(jsonEncode(log)); await prefs.setStringList('crash_logs', cachedLogs); } // 应用启动时检查并发送缓存 void _sendCachedLogs() async { final prefs = await SharedPreferences.getInstance(); final cachedLogs = prefs.getStringList('crash_logs') ?? []; for (final log in cachedLogs) { await _uploadToServer(jsonDecode(log)); } await prefs.remove('crash_logs'); } ``` --- ### 四、优化建议 1. **添加重试机制**:对失败的上传尝试设置指数退避重试 2. **数据压缩**:对大型堆栈跟踪进行GZIP压缩 3. **用户标识**:关联用户ID帮助定位问题 4. **敏感信息过滤**:移除可能包含的隐私数据 --- ### 五、完整流程示意图 ``` 捕获异常 → 收集设备信息 → 本地缓存 → 网络上传服务器接收 → 清空缓存 ``` 通过上述步骤,即可实现Flutter应用崩溃日志的自动化收集上传。需根据具体服务器API调整数据格式和字段。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值