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


相关实践学习
通过日志服务实现云资源OSS的安全审计
本实验介绍如何通过日志服务实现云资源OSS的安全审计。
相关文章
|
11月前
|
存储 Android开发
如何查看Flutter应用在Android设备上已被撤销的权限?
如何查看Flutter应用在Android设备上已被撤销的权限?
571 64
|
10月前
|
Android开发 数据安全/隐私保护 开发者
Android自定义view之模仿登录界面文本输入框(华为云APP)
本文介绍了一款自定义输入框的实现,包含静态效果、hint值浮动动画及功能扩展。通过组合多个控件完成界面布局,使用TranslateAnimation与AlphaAnimation实现hint文字上下浮动效果,支持密码加密解密显示、去除键盘回车空格输入、光标定位等功能。代码基于Android平台,提供完整源码与attrs配置,方便复用与定制。希望对开发者有所帮助。
202 0
|
10月前
|
XML Java Android开发
Android自定义view之网易云推荐歌单界面
本文详细介绍了如何通过自定义View实现网易云音乐推荐歌单界面的效果。首先,作者自定义了一个圆角图片控件`MellowImageView`,用于绘制圆角矩形图片。接着,通过将布局放入`HorizontalScrollView`中,实现了左右滑动功能,并使用`ViewFlipper`添加图片切换动画效果。文章提供了完整的代码示例,包括XML布局、动画文件和Java代码,最终展示了实现效果。此教程适合想了解自定义View和动画效果的开发者。
440 65
Android自定义view之网易云推荐歌单界面
|
10月前
|
Android开发 开发者
Android企业级实战-界面篇-3
本文是《Android企业级实战-界面篇》系列的第三篇,主要介绍分割线和条形跳转框的实现方法,二者常用于设置和个人中心界面。文章通过具体代码示例展示了如何实现这两种UI组件,并提供了效果图。实现前需准备`dimens.xml`、`ids.xml`、`colors.xml`等文件,部分资源可参考系列第一、二篇文章。代码中详细说明了布局文件的配置,如分割线的样式定义和条形跳转框的组件组合,帮助开发者快速上手并应用于实际项目中。
128 1
|
10月前
|
XML Android开发 数据格式
Android企业级实战-界面篇-2
本文为《Android企业级实战-界面篇》系列第二篇,主要介绍三个UI模块的实现:用户资料模块、关注与粉丝统计模块以及喜欢和收藏功能模块。通过详细的XML代码展示布局设计,包括dimens、ids、colors配置文件的使用,帮助开发者快速构建美观且功能齐全的界面。文章结合实际效果图,便于理解和应用。建议配合第一篇文章内容学习,以获取完整工具类支持。
160 0
|
10月前
|
算法 Java Android开发
Android企业级实战-界面篇-1
本文详细介绍了Android企业级开发中界面实现的过程,涵盖效果展示、实现前准备及代码实现。作者通过自身经历分享了Android开发经验,并提供了`dimens.xml`、`ids.xml`、`colors.xml`和`strings.xml`等配置文件内容,帮助开发者快速构建规范化的UI布局。文章以一个具体的用户消息界面为例,展示了如何使用线性布局(LinearLayout)和相对布局(RelativeLayout)实现功能模块排列,并附带注意事项及使用方法,适合初学者和进阶开发者参考学习。
228 0
|
前端开发 Java Shell
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
851 20
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
11月前
|
NoSQL 应用服务中间件 PHP
布谷一对一直播源码android版环境配置流程及功能明细
部署需基于 CentOS 7.9 系统,硬盘不低于 40G,使用宝塔面板安装环境,包括 PHP 7.3(含 Redis、Fileinfo 扩展)、Nginx、MySQL 5.6、Redis 和最新 Composer。Swoole 扩展需按步骤配置。2021.08.05 后部署需将站点目录设为 public 并用 ThinkPHP 伪静态。开发环境建议 Windows 操作系统与最新 Android Studio,基础配置涉及 APP 名称修改、接口域名更换、包名调整及第三方登录分享(如 QQ、微信)的配置,同时需完成阿里云与腾讯云相关设置。
|
Dart 前端开发 Android开发
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
421 4
【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex

热门文章

最新文章