使用logcat让Android应用支持查看实时日志并输出至界面显示功能
时间:2022-07-22
本文章向大家介绍使用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;
}
}
};
}
- Linux操作系统DNS解析(nameserver)监控脚本
- 不给“爸爸”添麻烦 - iTOP iOS 动态库改造
- 移动SEO分享:php自动提交复合型Sitemap到百度搜索
- 《Android外部存储》
- Android JNI出坑指南
- 《iPhone X ARKit Face Tracking》
- 结合标签广告,定制一个QQ邮箱订阅
- SecureCRT全局发送相同命令,快速抓取服务器信息的方法
- [不定期更新]简单的shell脚本练习实例
- 超简单的MySQL主从复制配置步骤
- 解决Centos下vsftp无法上传文件的问题,附vsftp配置详解
- 为iFrame添加动态载入效果,提高用户体验
- 分享超炫的表白页面和爱的纪念日源码
- 分享WordPress Mobile Pack汉化精简版及隐藏指定插件更新提示的方法
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- PHP实现微信提现功能(微信商城)
- 关于Yii2框架跑脚本时内存泄漏问题的分析与解决
- laravel 框架结合关联查询 when()用法分析
- php 实现简单的登录功能示例【基于thinkPHP框架】
- laravel框架邮箱认证实现方法详解
- 使用Git实现Laravel项目的自动化部署
- php传值和传引用的区别点总结
- php pdo连接数据库操作示例
- 关于Laravel参数验证的一些疑与惑
- PHP cookie与session会话基本用法实例分析
- Laravel 微信小程序后端实现用户登录的示例代码
- 使用PHP开发留言板功能
- PHP的new static和new self的区别与使用
- php时间戳转换代码详解
- thinkPHP和onethink微信支付插件分享