Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱
MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina.com

Log 日志工具类 保存到文件 MD


目录

日志工具类

几个比较流行的开源库

logger 8K

hugo 6K

timber 5.5K

XLog 1.5K

KLog 1.5K

简介

public final class android.util.Log extends Object

API for sending log output.

Generally, use the Log.v() Log.d() Log.i() Log.w() and Log.e() methods.

The order in terms of verbosity(冗长,赘言), from least to most is ERROR, WARN, INFO, DEBUG, VERBOSE.

Verbose should never be compiled into an application except during development.

Debug logs are compiled in but stripped at runtime. Error, warning and info logs are always kept.

Tip: A good convention约定 is to declare a TAG constant in your class:

private static final String TAG = "MyActivity";

and use that in subsequent随后的 calls to the log methods.

Tip: Don't forget that when you make a call like

Log.v(TAG, "index=" + i);

that when you're building the string to pass into Log.d, the compiler uses a StringBuilder and at least three allocations分配 occur: the StringBuilder itself, the buffer, and the String object. Realistically, there is also another buffer allocation and copy, and even more pressure on the gc. That means that if your log message is filtered out, you might be doing significant work and incurring significant overhead.

日志工具类

超强工具类

一个可打印并可跳转到"日志所在类、所在行"的工具类

效果如下:



public class L {
private static boolean OPEN_LOG = BuildConfig.DEBUG; public static void isOpenLog(boolean isOpenLog) {
OPEN_LOG = isOpenLog;
} public static void v(Object... message) {
if (OPEN_LOG) log(Log.VERBOSE, formatMessage(message));
} public static void d(Object... message) {
if (OPEN_LOG) log(Log.DEBUG, formatMessage(message));
} public static void i(Object... message) {
if (OPEN_LOG) log(Log.INFO, formatMessage(message));
} public static void w(Object... message) {
log(Log.WARN, formatMessage(message));
} public static void e(Object... message) {
log(Log.ERROR, formatMessage(message));
} public static void json(String json) {
json = new GsonBuilder()
.setPrettyPrinting()
.create()
.toJson(new JsonParser().parse(json));
log(Log.DEBUG, formatMessage(json));
} /**
* 根据type输出日志消息,包括方法名,方法行数,Message
*
* @param type 日志类型,如Log.INFO
* @param message 日志内容
*/
private static void log(int type, String message) {
StackTraceElement stackTrace = Thread.currentThread().getStackTrace()[4]; //核心一,获取到执行log方法的代码位置
String className = stackTrace.getClassName();
String tag = className.substring(className.lastIndexOf('.') + 1); //不必须的tag
StringBuilder sb = new StringBuilder(); sb.append("日志")
.append("(") //核心二,通过(fileName:lineNumber)格式,即可在点击日志后跳转到执行log方法的代码位置
.append(stackTrace.getFileName())//文件名
.append(":")
.append(stackTrace.getLineNumber())//行号
.append(")")
.append("_") //后面的属于不必须的信息
.append(stackTrace.getMethodName())//方法名
.append("【")
.append(message)//消息
.append("】"); switch (type) {
case Log.DEBUG:
Log.d(tag, sb.toString());
break;
case Log.INFO:
Log.i(tag, sb.toString());
break;
case Log.WARN:
Log.w(tag, sb.toString());
break;
case Log.ERROR:
Log.e(tag, sb.toString());
break;
case Log.VERBOSE:
Log.v(tag, sb.toString());
break;
}
} private static String formatMessage(Object... messages) {
StringBuilder sb = new StringBuilder();
for (Object mobj : messages) {
sb.append(objToString(mobj, -1)).append("\n");
}
return sb.substring(0, sb.length() - 1).trim();
} //****************************************************************************************** private static String objToString(Object obj, int currentLvel) {
if (obj == null) return "null";
currentLvel++;
StringBuilder sb = new StringBuilder();
if (obj instanceof String) {//字符串
sb.append(getSpace(currentLvel)).append(obj);
} else if (obj instanceof Object[]) {//数组
sb.append("数组:").append("\n");
for (Object mobj : (Object[]) obj) {
sb.append(objToString(mobj, currentLvel)).append("\n");//递归
}
} else if (obj instanceof Collection) {//集合
sb.append("集合:").append("\n");
for (Object mobj : (Collection) obj) {
sb.append(objToString(mobj, currentLvel)).append("\n");//递归
}
} else {//其他对象
sb.append(getSpace(currentLvel)).append(obj.toString());
}
return sb.toString();
} /***
* 格式化目录
*
* @param level 目录层次,也即"| _ _"的个数
*/
private static String getSpace(int level) {
StringBuilder sb = new StringBuilder();
for (int i = 0; i < level; i++) {
sb.append("|__");
}
return sb.toString();
}
}

实验案例

L.i("包青天");
L.i(5);
L.i(true);
L.json("{\"name\":\"bqt\",\"age\":30,\"more\":[true,985,\"河南\"] }");
L.i(new Person("白乾涛"));
L.i("----------------------连续---------------------");
L.i("包青天", 5, true, new Person("白乾涛"));
L.i("----------------------数组---------------------");
Object[] array = {"包青天", 5, true, new Person("白乾涛")};
L.i(array);
L.i("-----------------------集合--------------------");
L.i(Arrays.asList(array));
L.i("---------------------嵌套----------------------");
List<Object> mList = new ArrayList<>();
mList.add("这是一级元素");
mList.add(10086);
mList.add(new Person("包青天"));
mList.add(array);
mList.add(Arrays.asList(array));
L.i(mList);

实验结果

日志(MainActivity.java:53)_test【包青天】
日志(MainActivity.java:54)_test【5】
日志(MainActivity.java:55)_test【true】
日志(MainActivity.java:56)_test【{
"name": "bqt",
"age": 30,
"more": [
true,
985,
"河南"
]
}】
日志(MainActivity.java:57)_test【Person{name='白乾涛'}】
日志(MainActivity.java:58)_test【----------------------连续---------------------】
日志(MainActivity.java:59)_test【包青天
5
true
Person{name='白乾涛'}】
日志(MainActivity.java:60)_test【----------------------数组---------------------】
日志(MainActivity.java:62)_test【包青天
5
true
Person{name='白乾涛'}】
日志(MainActivity.java:63)_test【-----------------------集合--------------------】
日志(MainActivity.java:64)_test【集合:
|__包青天
|__5
|__true
|__Person{name='白乾涛'}】
日志(MainActivity.java:65)_test【---------------------嵌套----------------------】
日志(MainActivity.java:72)_test【集合:
|__这是一级元素
|__10086
|__Person{name='包青天'}
数组:
|__|__包青天
|__|__5
|__|__true
|__|__Person{name='白乾涛'} 集合:
|__|__包青天
|__|__5
|__|__true
|__|__Person{name='白乾涛'}】

一个专门保存日志到文件的工具类

public class LaunchLog {
private static long tempTime;
private static FileOutputStream outputStream;
private static final String DIR = Environment.getExternalStorageDirectory().getAbsolutePath() + "/bqtlog"; public static void write(String text) {
try {
File dir = new File(DIR);
if (!dir.exists()) {
dir.mkdirs();
}
String date = new SimpleDateFormat("HH:mm:ss_SSS", Locale.getDefault()).format(new Date());
if (outputStream == null) {
outputStream = new FileOutputStream(new File(dir, "log.txt"), true);
outputStream.write(("\n------------------------- " + date + " ------------------------\n").getBytes());
tempTime = System.currentTimeMillis();
}
outputStream.write(("\n" + date + "\t" + text + "(耗时" + (System.currentTimeMillis() - tempTime) + "毫秒)").getBytes());
outputStream.flush();
tempTime = System.currentTimeMillis();
} catch (Exception e) {
e.printStackTrace();
}
} public static void close() {
if (outputStream != null) {
try {
outputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}

一个简单封装的日志工具类

public class L {
/**
* 是否需要打印bug,可以在application的onCreate函数里面初始化
*/
public static boolean isDebug = true;
private static final String TAG = "bqt";
private L() {
}
//**********************************************下面四个是默认tag的函数 ********************************************
public static void i(String msg) {
if (isDebug) Log.i(TAG, msg);
}
public static void d(String msg) {
if (isDebug) Log.d(TAG, msg);
}
public static void e(String msg) {
if (isDebug) Log.e(TAG, msg);
}
public static void v(String msg) {
if (isDebug) Log.v(TAG, msg);
}
//******************************************下面是传入自定义tag的函数 ************************************************
public static void i(String tag, String msg) {
if (isDebug) Log.i(tag, msg);
}
public static void d(String tag, String msg) {
if (isDebug) Log.i(tag, msg);
}
public static void e(String tag, String msg) {
if (isDebug) Log.i(tag, msg);
}
public static void v(String tag, String msg) {
if (isDebug) Log.i(tag, msg);
}
}

一个保存日志到文件的工具类

产生的文件内容

public class MyLog {
private static boolean MYLOG_SWITCH = true; // 是否打印日志及是否写入到文件的总开关
private static boolean MYLOG_WRITE_TO_FILE = true;// 是否将日志写入到文件的开关
private static char MYLOG_TYPE = 'v';// 输入日志类型,仅输出比自己级别高的日志,日志级别ERROR, WARN, INFO, DEBUG, VERBOSE
private static String MYLOG_PATH_SDCARD_DIR = Environment.getExternalStorageDirectory().getPath() + "/bqt";// 日志文件的路径
private static String MYLOGFILEName = "Log-";// 日志文件名称①
private static SimpleDateFormat logfile = new SimpleDateFormat("yyyy-MM-dd");// 日志文件名称②
//**************************************************************************************************************************
public static void w(String tag, String text) {
log(tag, text, 'w');
}
public static void e(String tag, String text) {
log(tag, text, 'e');
}
public static void d(String tag, String text) {
log(tag, text, 'd');
}
public static void i(String tag, String text) {
log(tag, text, 'i');
}
public static void v(String tag, String text) {
log(tag, text, 'v');
}
//**************************************************************************************************************************
//根据tag, msg和等级输出日志
private static void log(String tag, String msg, char level) {
if (MYLOG_SWITCH) {
if ('e' == level) Log.e(tag, msg);
else if ('w' == level && ('w' == MYLOG_TYPE || 'i' == MYLOG_TYPE || 'd' == MYLOG_TYPE || 'v' == MYLOG_TYPE)) Log.w(tag, msg);
else if ('i' == level && ('i' == MYLOG_TYPE || 'd' == MYLOG_TYPE || 'v' == MYLOG_TYPE)) Log.i(tag, msg);
else if ('d' == level && ('d' == MYLOG_TYPE || 'v' == MYLOG_TYPE)) Log.d(tag, msg);
else if ('v' == level && 'v' == MYLOG_TYPE) Log.v(tag, msg);
if (MYLOG_WRITE_TO_FILE) writeLogtoFile(String.valueOf(level), tag, msg);
}
}
//将日志写入文件,文件名格式为MYLOGFILEName + logfile.format(nowtime) + ".txt",如【Log-2015-12-25.txt】
private static void writeLogtoFile(String mylogtype, String tag, String text) {
Date nowtime = new Date();
SimpleDateFormat myLogSdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String myLogs = myLogSdf.format(nowtime) + " " + mylogtype + " " + tag + " " + text;
if (!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) return;//判断sdcard是否插入
File fileDir = new File(MYLOG_PATH_SDCARD_DIR);
if (!fileDir.exists()) fileDir.mkdirs();//创建文件夹,这一步不能少
File file = new File(fileDir, MYLOGFILEName + logfile.format(nowtime) + ".txt");
try {
FileWriter filerWriter = new FileWriter(file, true);//true代表在文件中原来的数据后面添加,不进行覆盖
BufferedWriter bufWriter = new BufferedWriter(filerWriter);
bufWriter.write(myLogs);
bufWriter.newLine();
bufWriter.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}

2017-7-11

Log 日志工具类 保存到文件 MD的更多相关文章

  1. C# Log日志工具类

    using System; using System.Collections.Generic; using System.Text; using System.IO; public class Log ...

  2. Android开发调试日志工具类[支持保存到SD卡]

    直接上代码: package com.example.callstatus; import java.io.File; import java.io.FileWriter; import java.i ...

  3. Android utils 之 日志工具类

    工具类 在开发的过程中,我们时常会对代码执行特定的处理,而这部分处理在代码中可能多次用到,为了代码的统一性.规范性等,通过建工具类的方式统一处理.接下来我会罗列各种工具类. 日志工具类 在utils文 ...

  4. 封装一个简单好用的打印Log的工具类And快速开发系列 10个常用工具类

    快速开发系列 10个常用工具类 http://blog.csdn.net/lmj623565791/article/details/38965311 ------------------------- ...

  5. 002-基本业务搭建【日志,工具类dbutils,dbcp等使用】

    一.需求分析 1.1.概述 1.用户进入“客户管理”,通过列表方式查看用户: 2.客户名称,模糊查询用户列表 3.客户名称,可查看客户详细信息 4.新增.编辑.删除功能等 二.系统设计 需要对原始需求 ...

  6. 【个人使用.Net类库】(2)Log日志记录类

    开发接口程序时,要保证程序稳定运行就要时刻监控接口程序发送和接收的数据,这就需要一个日志记录的类将需要的信息记录在日志文件中,便于自己维护接口程序.(Web系统也是如此,只是对应的日志实现比这个要复杂 ...

  7. logger日志工具类

    日志工厂类 package cn.itcast.utils; import java.util.logging.FileHandler; import java.util.logging.Handle ...

  8. Java 基于log4j的日志工具类

    对log4j日志类进行了简单封装,使用该封装类的优势在于以下两点: 1.不必在每个类中去创建对象,直接类名 + 方法即可 2.可以很方便的打印出堆栈信息 package com.tradeplatfo ...

  9. JAVA日志工具类

    package com.ming.util; import java.io.File; import java.io.FileWriter; import java.io.IOException; i ...

随机推荐

  1. Python之路【第十篇】: python基础之socket编程

    阅读目录 一 客户端/服务器架构 二 osi七层 三 socket层 四 socket是什么 五 套接字发展史及分类 六 套接字工作流程 七 基于TCP的套接字 八 基于UDP的套接字 九 recv与 ...

  2. Xamarin无法调试Android项目

    Xamarin无法调试Android项目   项目可以正常编译,生成APK,也可以通过右键菜单部署.但是一旦开启调试,就报错.错误信息如下: 没有为此解决方案配置选中要生成的项目   出现这种问题是因 ...

  3. wpf企业应用之主从结构列表

    主从结构在企业级应用中相当常见,这里结合我的例子谈一下wpf中主从结构列表展示的常用做法,具体效果见 wpf企业级开发中的几种常见业务场景. 首先,Model有两种,主表对应model(假设为mode ...

  4. CentOS下重新安装yum的方法

    不小心误删除了VPS下面的yum,大家都知道yum在linux中是很重要的一个功能,软件的下载,系统的更新都要靠他.没有yum,系统基本处于半残废状态. yum的安装操作: 在SSH里面依次输入下面的 ...

  5. hdu 1973 bfs+素数判断

    题意:给出两个四位数,现要改变第一个数中的个,十,百,千位当中的一个数使它最终变成第二个数,要求这过程中形成的数是素数,问最少的步骤题解:素数筛选+bfsSample Input31033 81791 ...

  6. wx小程序碎碎念

    对button组件的例子中,js代码的一点理解 for (var i = 0; i < types.length; ++i) { (function(type) { // 循环构建目标函数 pa ...

  7. centos7安装redis-4.0.1集群

    试验机操作系统:CentOS Linux release 7.2.1511 (Core) 本文的目的是教会大家快速搭建redis集群,完了再深入学习. 试问如果不上手试验,看的资料再多有个毛用? 下载 ...

  8. bash编程之 ~制作Mini Linux系统~

    说明1:在一个Linux宿主机系统上,通过以上步骤,可以制作一个微小的Linux系统(可以放置在U盘中等),然后在其它的主机(虚拟机或者物理机)上运行,以实现最小化定制系统的目的. 说明2:上图中黑色 ...

  9. Ubuntu16.04中安装stlink驱动

    系统环境: Vmware12, Ubuntu16.04 Stlink version:v1.4.0 一.安装依赖包: sudo apt-get install libusb-1.0 sudo apt- ...

  10. 求平方根C++

    求平方根,正根.曾经都不会.昨天看数学,看到了,写了出来.自己又小优化了一下,非常不错. // squareRoot.cpp -- 2011-08-29-01.04 #include "st ...