android 收集错误日志 上传服务器

import android.app.Application;
public class MyApp extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        CrashHandler.getInstance().init(this);
    }
}

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.ycfy.carsh">

    <application
        android:name=".MyApp"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.GET_ACCOUNTS"/>
</manifest>
import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Bundle;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button;

//http://m.blog.csdn.net/u013110195/article/details/76974022
public class MainActivity extends AppCompatActivity {

    private static final String TAG = "MainActivity ";
    private Button btn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        btn = (Button) findViewById(R.id.btn);

        checkInitPermissions();

        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                int i = Integer.parseInt("aa");
//                throw new RuntimeException("这是自己定义的异常");
            }
        });

    }

    public static final int REQUEST_PERMISSION_CODE = 100;
    private void checkInitPermissions(){

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            String[] permissions = new String[]{
                    //放入需要授予的权限,例如需要写入的权限
                    Manifest.permission.WRITE_EXTERNAL_STORAGE
            };
            if (ContextCompat.checkSelfPermission(this, permissions[0]) != PackageManager.PERMISSION_GRANTED) {
                if (ActivityCompat.shouldShowRequestPermissionRationale(this, permissions[0])){
                    ActivityCompat.requestPermissions(this, permissions, REQUEST_PERMISSION_CODE);
                }
            }
        } else {
            handleAfterPermissions();
        }
    }

    private void  handleAfterPermissions(){

    }


    @Override
    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == REQUEST_PERMISSION_CODE){
            handleAfterPermissions();
        }
    }
}

import android.content.Context;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Environment;
import android.os.Process;
import android.util.Log;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;

/**
 * function: 截获崩溃日志:当程序产生未捕获异常
 * Created by lzq on 2017/8/10.
 */

public class CrashHandler implements Thread.UncaughtExceptionHandler{

    //手机存储路径
    private final String logPath = Environment.getExternalStorageDirectory().getAbsolutePath() + File.separator + "Crash";

    private static final String TAG = CrashHandler.class.getSimpleName();

    /** 系统默认的异常处理器(默认情况下,系统会终止当前的异常程序) */
    private Thread.UncaughtExceptionHandler mDefaultCrashHandler;
    private Context mContext;
    private String mProcessName;

    /** CrashHandler 实例 */
    private static CrashHandler instance = new CrashHandler();

    /** 私有构造 保证只有一个CrashHandler实例*/
    private CrashHandler(){
    }

    /** 单例模式 */
    public static CrashHandler getInstance(){
        return instance;
    }


    /** 初始化 */
    public void init(Context context){
        mDefaultCrashHandler = Thread.getDefaultUncaughtExceptionHandler();

        //将当前实例设置为系统默认的异常处理器
        Thread.setDefaultUncaughtExceptionHandler(this);

        mContext = context.getApplicationContext();
        mProcessName = context.getPackageName();
    }

    /** 用于格式化日期,作为日志文件名的一部分. */
    private final DateFormat formatter = new SimpleDateFormat("MMdd-HH:mm:ss", Locale.getDefault());


    @Override
    public void uncaughtException(Thread thread, Throwable throwable) {
        Log.e(TAG, "---------------uncaughtException start---------------\r\n");
        try {
            dumpExceptionToLocalFile(thread, throwable); //把异常日志在手机本地文件输出
            uploadExceptionToServer();  //上传crash日志到服务器
        } catch (IOException e) {
            Log.e(TAG, "uncaughtException,ex:" + e.getMessage());
            e.printStackTrace();
        }

        if (mDefaultCrashHandler != null){
            mDefaultCrashHandler.uncaughtException(thread, throwable);
        }else{
            Process.killProcess(Process.myPid());
        }

        Log.e(TAG, "---------------uncaughtException end---------------\r\n");
    }

    /** dump UncaughtException into local file */
    private void dumpExceptionToLocalFile(Thread thread, Throwable throwable) throws IOException{
        //记录数量达到10个就清理数据
        File logDir = new File(logPath);
        if (logDir.exists()){
            clearExLogWhenMax(logDir);
        }else{
            logDir.mkdirs();
        }

        //错误信息文件名
        Date date = new Date();
        String logFileName = formatter.format(date) + String.format("[%s]", thread.getName())+ ".txt";
        File logExFile = new File(logDir, logFileName);
        logExFile.createNewFile();

        //写入信息到文件中
        try {
            PrintWriter pw = new PrintWriter(new BufferedWriter(new FileWriter(logExFile)));

            //时间戳
            pw.println("Time stamp:" + date);

            pw.println("Process[" + mProcessName + "," + Process.myPid() + "]");

            pw.println();

            //手机信息
            dumpPhoneInfo(pw);

            pw.println();

            //导出异常调用栈信息
            throwable.printStackTrace(pw);
            pw.close();

        }catch (IOException ex){
            Log.e(TAG, "dump info failed");
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }

    }


    /**
     * 设置最大日志数量 10
     * @param logDir 日志目录
     */
    private void clearExLogWhenMax(File logDir){

        File[] logFileList = logDir.listFiles();
        if (logFileList == null || logFileList.length == 0){
            return;
        }

        int length = logFileList.length;
        if (length > 10){
            for (File aFile : logFileList){
                try {
                    if (aFile.delete()){
                        Log.d(TAG, "clearExLogWhenMax:" + aFile.getName());
                    }
                }catch (Exception ex){
                    Log.d(TAG, "clearExLogWhenMax:" + ex);
                }
            }
        }

    }


    private void dumpPhoneInfo(PrintWriter pw) throws IOException, PackageManager.NameNotFoundException {
        PackageManager pm = mContext.getPackageManager();
        PackageInfo pi = pm.getPackageInfo(mContext.getPackageName(), PackageManager.GET_ACTIVITIES);
        pw.println("App Version:" + pi.packageName + "_" + pi.versionCode);
        //Android版本号
        pw.println("OS Version:" + Build.VERSION.RELEASE + "_" + Build.VERSION.SDK_INT);

        //手机制造商
        pw.println("Vendor:" + Build.MANUFACTURER);

        //手机型号
        pw.println("Model:" + Build.MODEL);

        //CPU架构
        pw.println("CPU ABI:" + Build.CPU_ABI);

    }


    private void uploadExceptionToServer() {
        //TODO Upload Exception Message To Your Web Server
    }

}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值