使用logcat让Android应用支持查看实时日志并输出至界面显示功能

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 使用logcat让Android应用支持查看实时日志并输出至界面显示功能

不使用USB线接Android设备连接电脑,也不用电脑上装Android studio和logcat工具,


如何查看应用的实时日志呢?方法还是有的。


先附图:看这功能是不是很赞?


机器强大了就是好,有好多创新可以派上用场了。后续继续探索新鲜的新功能。


运维的兄弟们可以松口气了,给你们减减压。


日志排查获取从此如此简单。甚至可以给手机互通,日志显示到你手机上也能。


这有什么用?方便现场运维人员快速的协助研发定位和找到问题。


当然了,没问题也不用看日志了。看日志就是为了分析和定位问题的一种有效途径。


可以选择通过蓝牙发送到手机,或一键发送到后台。更智能点儿的,后台可触发某个终端自动上送日志。



且日志还是实时输出的,这样从应用的后门调起查看日志的窗口,就很方便的查看实时的日志输出啦


这功能是不是很赞?且可以清空窗口,保存日志,发送日志给后台等功能。


/**
Author:yangyongzhen
qq:534117529
Date:20200221
*/
package com.xxxx.xxx.activity;
import android.annotation.SuppressLint;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.text.method.ScrollingMovementMethod;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.RandomAccessFile;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
public class LogActivity extends BaseActivity implements View.OnClickListener {
    String cmds = "";
    StringBuilder Sb = new StringBuilder("this is log");
    private BufferedReader mReader = null;
    private Process exec;
    private int mPId;
    private String mPID;
    private boolean mRunning = true;
    EditText etlog;
    Button btnStop,btnSave,btnClear;
    RandomAccessFile randomFile = null;
    private final String DIR = CommonStatus.PATH_DIR + "/log";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        toDispPageInfo("日志查看");
        etlog = findViewById(R.id.et_logs);
        btnStop =  findViewById(R.id.btn_stop);
        btnStop.setOnClickListener(this);
        btnSave =  findViewById(R.id.btn_save);
        btnSave.setOnClickListener(this);
        btnClear =  findViewById(R.id.btn_clear);
        btnClear.setOnClickListener(this);
        mPId = android.os.Process.myPid();
        mPID = String.valueOf(mPId);
        cmds = "logcat  *:e *:d | grep \"(" + mPID + ")\"";;
    }
    @Override
    protected void onResume() {
        super.onResume();
        try {
            exec = Runtime.getRuntime().exec(cmds);
            mReader = new BufferedReader(new InputStreamReader(exec.getInputStream()), 1024);
            new Thread(new Runnable() {
                @Override
                public void run() {
                    String line="";
                    try {
                        while (mRunning && (line = mReader.readLine())!=null) {
                            Sb.append(line);
                            if (!mRunning) {
                                break;
                            }
                            if (line.length()==0) {
                                continue;
                            }
                            final Message msg = Message.obtain();
                            msg.what = 2;
                            msg.obj = line+ "\n";
                            mhandler.sendMessage(msg);
                            Thread.sleep(100);
                        }
                    }catch (Exception e){
                        Log.e(TAG, e.toString());
                    }
                    finally {
                        Log.e(TAG, "finally");
                        Log.e(TAG, "" + mRunning);
                        if (line == null) {
                            Log.e(TAG, "line is null");
                        }
                        if (exec != null) {
                            exec.destroy();
                        }
                        if (mReader != null) {
                            try {
                                mReader.close();
                                mReader = null;
                            } catch (IOException e) {
                            }
                        }
                    }
                }
            }).start();
        } catch (IOException e) {
            e.printStackTrace();
        }
        //startShowLog();
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mRunning = false;
        if (exec != null) {
            exec.destroy();
        }
        if (mReader != null) {
            try {
                mReader.close();
                mReader = null;
            } catch (IOException e) {
            }
        }
    }
    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.btn_stop:
                mRunning = false;
                break;
            case R.id.btn_save:
                SimpleDateFormat format1 = new SimpleDateFormat("yyyy-MM-ddHHmmss", Locale.CHINA);
                String date = format1.format(new Date(System.currentTimeMillis()));
                try {
                    String path = DIR + "/" + date + ".log";
                    randomFile = new RandomAccessFile(path, "rw");
                    randomFile.write(etlog.getText().toString().getBytes());
                    Toast t = Toast.makeText(mContext, "日志保存成功,path="+path, Toast.LENGTH_SHORT);
                    t.show();
                } catch (IOException e) {
                    //e.printStackTrace();
                    LogUtil.d(e.toString());
                }
                break;
            case R.id.btn_clear:
               etlog.setText("");
                break;
        }
    }
    @Override
    public int initLayout() {
        return R.layout.activity_log;
    }
    @Override
    public void initView() {
        toBack();
    }
    @Override
    public void initData() {
    }
    @SuppressLint("HandlerLeak")
    Handler mhandler = new Handler() {
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case 1:
                    break;
                case 2:
                    //显示日志
                    etlog.setMovementMethod(ScrollingMovementMethod.getInstance());
                    etlog.setSelection(etlog.getText().length(), etlog.getText().length());
                    etlog.setText(etlog.getText().append(msg.obj.toString()));
                    break;
            }
        }
    };
}


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
2天前
|
算法 Java 数据库
Android 应用的主线程在什么情况下会被阻塞?
【10月更文挑战第20天】为了避免主线程阻塞,我们需要合理地设计和优化应用的代码。将耗时操作移到后台线程执行,使用异步任务、线程池等技术来提高应用的并发处理能力。同时,要注意避免出现死循环、不合理的锁使用等问题。通过这些措施,可以确保主线程能够高效地运行,提供流畅的用户体验。
8 2
|
5天前
|
Java API Android开发
安卓应用程序开发的新手指南:从零开始构建你的第一个应用
【10月更文挑战第20天】在这个数字技术不断进步的时代,掌握移动应用开发技能无疑打开了一扇通往创新世界的大门。对于初学者来说,了解并学习如何从无到有构建一个安卓应用是至关重要的第一步。本文将为你提供一份详尽的入门指南,帮助你理解安卓开发的基础知识,并通过实际示例引导你完成第一个简单的应用项目。无论你是编程新手还是希望扩展你的技能集,这份指南都将是你宝贵的资源。
22 5
|
5天前
|
移动开发 Dart 搜索推荐
打造个性化安卓应用:从零开始的Flutter之旅
【10月更文挑战第20天】本文将引导你开启Flutter开发之旅,通过简单易懂的语言和步骤,让你了解如何从零开始构建一个安卓应用。我们将一起探索Flutter的魅力,实现快速开发,并见证代码示例如何生动地转化为用户界面。无论你是编程新手还是希望扩展技能的开发者,这篇文章都将为你提供价值。
|
15天前
|
调度 Android开发 开发者
构建高效Android应用:探究Kotlin多线程优化策略
【10月更文挑战第11天】本文探讨了如何在Kotlin中实现高效的多线程方案,特别是在Android应用开发中。通过介绍Kotlin协程的基础知识、异步数据加载的实际案例,以及合理使用不同调度器的方法,帮助开发者提升应用性能和用户体验。
36 4
|
XML 数据可视化 Java
Android常见界面布局(详细介绍)
Android常见界面布局(详细介绍)
460 0
Android常见界面布局(详细介绍)
|
Android开发
Android笔记:软键盘弹出遮盖原来界面的布局控件
Android笔记:软键盘弹出遮盖原来界面的布局控件
173 0
|
Web App开发 Java Android开发
android界面开发小结——android笔记---控件和布局
控件简介 ============================================================== 控件的设置主要依靠layout文件夹中的activity_main.
735 0
|
XML 编解码 Android开发
Android软件开发之盘点界面五大布局
雨松MOMO原创文章如转载,请注明:转载自雨松MOMO的博客原文地址:http://blog.csdn.net/xys289187120/article/details/6655494 1.线性布局(LinearLayout)         线性布局的形式可以分为两种,第一种横向线性布局 第二种纵向线性布局,总而言之都是以线性的形式 一个个排列出来的,纯线性布局的缺点是很不方便修改控件的显示位置,所以开发中经常会 以 线性布局与相对布局嵌套的形式设置布局。
685 0
|
22天前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的海洋中,自定义控件是那片璀璨的星辰。它不仅让应用界面设计变得丰富多彩,还提升了用户体验。本文将带你探索自定义控件的核心概念、实现过程以及优化技巧,让你的应用在众多竞争者中脱颖而出。
|
22天前
|
Java Android开发 Swift
安卓与iOS开发对比:平台选择对项目成功的影响
【10月更文挑战第4天】在移动应用开发的世界中,选择合适的平台是至关重要的。本文将深入探讨安卓和iOS两大主流平台的开发环境、用户基础、市场份额和开发成本等方面的差异,并分析这些差异如何影响项目的最终成果。通过比较这两个平台的优势与挑战,开发者可以更好地决定哪个平台更适合他们的项目需求。
83 1