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

简介: 使用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;
            }
        }
    };
}


相关实践学习
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
相关文章
|
7月前
|
SQL 人工智能 监控
SLS Copilot 实践:基于 SLS 灵活构建 LLM 应用的数据基础设施
本文将分享我们在构建 SLS SQL Copilot 过程中的工程实践,展示如何基于阿里云 SLS 打造一套完整的 LLM 应用数据基础设施。
2258 102
|
存储 Android开发
如何查看Flutter应用在Android设备上已被撤销的权限?
如何查看Flutter应用在Android设备上已被撤销的权限?
648 64
|
11月前
|
Android开发 数据安全/隐私保护 开发者
Android自定义view之模仿登录界面文本输入框(华为云APP)
本文介绍了一款自定义输入框的实现,包含静态效果、hint值浮动动画及功能扩展。通过组合多个控件完成界面布局,使用TranslateAnimation与AlphaAnimation实现hint文字上下浮动效果,支持密码加密解密显示、去除键盘回车空格输入、光标定位等功能。代码基于Android平台,提供完整源码与attrs配置,方便复用与定制。希望对开发者有所帮助。
232 0
|
11月前
|
XML Java Android开发
Android自定义view之网易云推荐歌单界面
本文详细介绍了如何通过自定义View实现网易云音乐推荐歌单界面的效果。首先,作者自定义了一个圆角图片控件`MellowImageView`,用于绘制圆角矩形图片。接着,通过将布局放入`HorizontalScrollView`中,实现了左右滑动功能,并使用`ViewFlipper`添加图片切换动画效果。文章提供了完整的代码示例,包括XML布局、动画文件和Java代码,最终展示了实现效果。此教程适合想了解自定义View和动画效果的开发者。
475 65
Android自定义view之网易云推荐歌单界面
|
11月前
|
Android开发 开发者
Android企业级实战-界面篇-3
本文是《Android企业级实战-界面篇》系列的第三篇,主要介绍分割线和条形跳转框的实现方法,二者常用于设置和个人中心界面。文章通过具体代码示例展示了如何实现这两种UI组件,并提供了效果图。实现前需准备`dimens.xml`、`ids.xml`、`colors.xml`等文件,部分资源可参考系列第一、二篇文章。代码中详细说明了布局文件的配置,如分割线的样式定义和条形跳转框的组件组合,帮助开发者快速上手并应用于实际项目中。
142 1
|
10月前
|
监控 安全 Linux
AWK在网络安全中的高效应用:从日志分析到威胁狩猎
本文深入探讨AWK在网络安全中的高效应用,涵盖日志分析、威胁狩猎及应急响应等场景。通过实战技巧,助力安全工程师将日志分析效率提升3倍以上,构建轻量级监控方案。文章详解AWK核心语法与网络安全专用技巧,如时间范围分析、多条件过滤和数据脱敏,并提供性能优化与工具集成方案。掌握AWK,让安全工作事半功倍!
378 0
|
11月前
|
XML Android开发 数据格式
Android企业级实战-界面篇-2
本文为《Android企业级实战-界面篇》系列第二篇,主要介绍三个UI模块的实现:用户资料模块、关注与粉丝统计模块以及喜欢和收藏功能模块。通过详细的XML代码展示布局设计,包括dimens、ids、colors配置文件的使用,帮助开发者快速构建美观且功能齐全的界面。文章结合实际效果图,便于理解和应用。建议配合第一篇文章内容学习,以获取完整工具类支持。
173 0
|
11月前
|
算法 Java Android开发
Android企业级实战-界面篇-1
本文详细介绍了Android企业级开发中界面实现的过程,涵盖效果展示、实现前准备及代码实现。作者通过自身经历分享了Android开发经验,并提供了`dimens.xml`、`ids.xml`、`colors.xml`和`strings.xml`等配置文件内容,帮助开发者快速构建规范化的UI布局。文章以一个具体的用户消息界面为例,展示了如何使用线性布局(LinearLayout)和相对布局(RelativeLayout)实现功能模块排列,并附带注意事项及使用方法,适合初学者和进阶开发者参考学习。
246 0
|
存储 监控 算法
基于 PHP 语言的滑动窗口频率统计算法在公司局域网监控电脑日志分析中的应用研究
在当代企业网络架构中,公司局域网监控电脑系统需实时处理海量终端设备产生的连接日志。每台设备平均每分钟生成 3 至 5 条网络请求记录,这对监控系统的数据处理能力提出了极高要求。传统关系型数据库在应对这种高频写入场景时,性能往往难以令人满意。故而,引入特定的内存数据结构与优化算法成为必然选择。
335 3