Logger 日志管理

article/2025/9/17 4:25:52

转载请注明出处: http://blog.csdn.net/like_program/article/details/52986553

1.Logger 是什么

在我们日常的开发中,肯定是少不了要和 Log 打交道,回想一下我们是怎么使用 Log 的:先定义一个静态常量 TAG,TAG 的值通常是当前类的类名,然后在需要打印 Log 的地方,调用 Log.d(TAG, "要打印的内容")。每次新写一个类,都要写一个 TAG,这也就算了,最苦逼的是,项目一上线,还要手动去把每个 Log 注释掉。。。

当然,有的同学可能要说,这又没什么,自己封装一个 Log 不就行了。但是对新手来说,自己封装有一定的难度,如果有现成的开源库可以直接拿来用就好了。

Github 上的大牛当然也注意到了这个情况,于是开源日志库 Logger 诞生了。

Logger 的 Github 主页:https://github.com/orhanobut/logger

Logger 提供以下功能:

  • 线程的信息
  • 类的信息
  • 方法的信息
  • 将 JSON 文本人性化输出
  • 将换行符人性化输出
  • 简洁的输出
  • 从日志跳转到源码

Logger 与 原生 Log 最大的不同就是:Logger 打印出来的日志一目了然,引用下官方 Github 上的图片,看下原生 Log 和 Logger 打印日志的区别:

原生 Log 打印的日志:

原生 log

Logger 打印的日志:

Logger

可以看到,Logger 打印的日志把多余的日志全部忽略了,只显示对我们有用的日志,而且还把日志框了起来,可以让我们看得更舒服。不得不说,Logger 做的实在是太人性化了。

Logger 使用方法也不难,Logger 开源库的 Github 主页 写了很详细的使用方法,英文还过得去的同学可以直接去 Logger 主页查看使用方法,有英文恐惧症的同学呢,可以通过这篇博客来学习下。

2.Logger 的使用

打开 Android Studio,新建 LoggerTest 项目。

1.导入依赖

在 LoggerTest / app/ build.gradle 中的 dependencies 中,导入依赖,代码如下:

dependencies {compile 'com.orhanobut:logger:1.15'
}

然后 Android Studio 应该会弹出 Sync Now,也就是箭头 1 指向的位置,如图:

导入依赖

如果弹出了 Sync Now,就点击它,如果没有弹出,就点击箭头 2 指向的选项。

等 Grade 构建完,我们就可以使用 Logger了。

2.简单使用

Logger 的使用方法和原生 Log 差不多,我们先来尝试下打印一个简单文本。

修改 MainActivity.java 中的 onCreate() 方法,代码如下:

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Logger.d("执行了 onCreate");
}

咦,怎么没写 TAG?其实,Logger 是有它自己默认的 TAG 的,默认的 TAG 是 PRETTYLOGGER,我们运行一下程序,看一下输出日志:

没有过滤

我们可以看到,已经成功的将日志输出了,而且比我们之前用的原生 Log 输出的美观多了,日志信息一目了然。箭头指向的就是 Logger 默认的 TAG,所以如果你想让 Logcat 只显示你自己打印的日志,可以在日志过滤器中,将默认 TAG 添加进去。

添加默认 TAG 1

添加默认 TAG 2

或者直接在过滤框中添加:

在过滤框中直接添加

3.打印更多类型数据

Logger 还支持打印 JSON,XML 等格式的数据。

1.JSON 类型数据

private String JSON_CONTENT = "{\"weatherinfo\":{\"city\":\"北京\",\"cityid\":\"101010100\"," +"\"temp\":\"18\",\"WD\":\"东南风\",\"WS\":\"1级\",\"SD\":\"17%\",\"WSE\":\"1\"," +"\"time\":\"17:05\",\"isRadar\":\"1\",\"Radar\":\"JC_RADAR_AZ9010_JB\"," +"\"njd\":\"暂无实况\",\"qy\":\"1011\",\"rain\":\"0\"}}";@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Logger.json(JSON_CONTENT);
}

打印 JSON

2.XML 类型数据

private String XML_CONTENT = "<china dn=\"nay\"><city quName=\"黑龙江\" pyName=\"heilongjiang\" " +"cityname=\"哈尔滨\" state1=\"1\" state2=\"1\" stateDetailed=\"多云\"/><city quName=\"吉林\"" +" pyName=\"jilin\" " +"cityname=\"长春\" state1=\"0\" state2=\"0\" stateDetailed=\"\"/><city quName=\"辽宁\" " +"pyName=\"liaoning\" " +"cityname=\"沈阳\" state1=\"1\" state2=\"0\" stateDetailed=\"多云转晴\"/><city " +"quName=\"海南\" pyName=\"hainan\" " +"cityname=\"海口\" state1=\"22\" state2=\"21\" stateDetailed=\"中到大雨转小到中雨\"/></china>";@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Logger.xml(XML_CONTENT);
}

XML 类型数据

3.List 类型数据

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// List 类型数据List<String> list = new ArrayList<>();list.add("hello");list.add("world");Logger.d(list);
}

List 类型数据

4.Map 类型数据

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// Map 类型数据Map<String, String> map = new HashMap<>();map.put("key_hello", "hello");map.put("key_world", "world");Logger.d(map);
}

Map 类型数据

5.Set 类型数据

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// Set 类型数据Set<String> set = new HashSet<>();set.add(new String("hello"));set.add(new String("world"));Logger.d(set);
}

Set 类型数据

4.字符串格式化

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Logger.d("hello %s %d", "world", 5);
}

字符串格式化

5.自定义 TAG

如果你不满意默认的 TAG,也可以自己定义一个 TAG,只需要调用一次 Logger.init() 就行了。因为只需要调用一次,所以可以在 Application 中来完成 Logger 的初始化。

新建 MyApplication 继承 Application,代码如下:

public class MyApplication extends Application {private static String TAG = "LoggerTest";@Overridepublic void onCreate() {super.onCreate();Logger.init(TAG);}
}

修改 AndroidManifest.xml 中 application 的属性,添加 android:name=".MyApplication" ,AndroidManifest.xml 代码如下:

<application
    android:name=".MyApplication"android:allowBackup="true"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:supportsRtl="true"android:theme="@style/AppTheme"><activity android:name=".MainActivity"><intent-filter><action android:name="android.intent.action.MAIN"/><category android:name="android.intent.category.LAUNCHER"/></intent-filter></activity>
</application>

这样 App 启动时,初始化的就是我们自定义的 MyApplication 了。然后我们在 MainActivity 的 onCreate() 中打印一下日志,看下自定义的 TAG 生效了没。

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Logger.d("hello world");
}

自定义 TAG

可以看到,我们自定义的 TAG 已经生效了。那么有的同学可能会有疑问了,我要是不想一直用这个 TAG 呢,要是临时想换个 TAG 用呢,当然也是有办法的,调用 Logger.t("临时TAG名").d() ,就能使用临时的 TAG 打印日志了。

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Logger.t("MyTag").d("hello world");
}

临时TAG

可以看到,临时的 TAG 名被追加到自定义 TAG 的后面了。

6.设置不打印日志

之前说了,日志只是开发的时候需要用到,等到项目上线就不能用了。Logger 当然也考虑到了这一点,通过设置Logger.init(TAG).logLevel(LogLevel.NONE) 可以设置为不打印日志。logLevel() 方法默认的参数是 LogLevel.FULL,也就是打印全部日志。修改 MyApplication 的 onCreate() 方法:

@Override
public void onCreate() {super.onCreate();Logger.init(TAG).logLevel(LogLevel.NONE);
}

运行一下,可以发现 Logger 没有打印日志。

7.打印 Exception

Logger 打印 Exception,可以很清晰的看到 Exception 信息,这里我们写一个数组越界异常:

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);int[] a = new int[3];try {a[4] = 3;} catch (Exception e) {Logger.e(e, "message");}
}

数组越界异常

8.更多个性化设置

Logger 还有一些个性化设置,比如

Logger.init(TAG).methodCount(3) // 方法栈打印的个数,默认是 2.hideThreadInfo() // // 隐藏线程信息,默认显示.methodOffset(2) // 设置调用堆栈的函数偏移值,默认是 0.logAdapter(new AndroidLogAdapter()); // 自定义一个打印适配器

有兴趣的同学可以去 Logger 的 Github 主页 查看Logger 的更多用法。

源码下载


http://chatgpt.dhexx.cn/article/vcDMUW6B.shtml

相关文章

【转】最详细的Log4J使用教程一、入门实例二、Log4J基本使用方法三、Spring中使用Log4J四、实战经验总结

原文地址&#xff1a;http://www.codeceo.com/article/log4j-usage.html 日志是应用软件中不可缺少的部分&#xff0c;Apache的开源项目log4j是一个功能强大的日志组件,提供方便的日志记录。在apache网站&#xff1a;jakarta.apache.org/log4j可以免费下载到Log4j最新版本的软…

Logger打印日志

1. 一个最基本的例子 使用Logging框架写Log基本上就三个步骤 引入loggerg类和logger工厂类声明logger记录日志 下面看一个例子 // 1. 引入slf4j接口的Logger和LoggerFactory import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class UserService { // 2. 声明…

rename 批量修改文件名

1.批量修改文件后缀&#xff1a; rename s/// *.nii rename s/\.nii/\.txt/ * #把.nii后缀改为.txt rename s/\.txt// * #把.txt后缀去掉 rename s/$/\.nii/ * #加上后缀.nii rename s//\.nii/ * #这样会把.nii放到文件名前面&#xff0c;导致文件被隐藏。 rename s/^/dm/ * #…

Linux基本功系列之rename命令实战

文章目录 一. rename 命令介绍二. 语法格式及常用选项三. 参考案例3.1 将当前目录下所有.cfg的文件&#xff0c;替换为.txt结尾3.2 将所有出现mufeng的部分都替换为mufeng13.3 将mufeng0开头都变成mufeng00开头3.4 rename支持正则表示式 总结 前言&#x1f680;&#x1f680;&a…

VS项目rename

Visual Studio c项目更改相关文件名字 项目里面文件夹 Rename 右键 .sln文件, 打开方式–> txt方式打开编辑找到里面project的文件夹名字, Rename. 然后到资源管理器里, 找到对应的folder, Rename. .vcxproj相关文件 Rename 相关文件Rename .sln文件, txt打开–>Ren…

rename 命令 – 批量修改文件名称

rename 命令的功能是用于批量修改文件名称。与 mv 命令一次只能修改一个文件名不同&#xff0c;rename命令能够基于正则表达式对文件名进行批量修改&#xff0c;但要求是把匹配规则准确的描述给系统。 rename 命令的参数有三项&#xff1a;其一是当前文件名中要被修改的字符&am…

linux之rename命令

用字符串替换的方式批量改变文件名 rename 命令存在两个版本用法上有所区别 C语言版本, 支持通配符 [常用通配符说明] ? 表示一个任意字符 * 表示一个或一串任意字符 [charset] 可替代charset集中的任意单个字符Perl版本, 支持正则表达式 [常用正则表达式符号说明] ^…

pandas:案例详解 rename函数 修改列名和行名

pandas&#xff1a;案例详解rename函数 修改列名和索引 rename函数简介0 构建学习数据1 修改索引两种方式2 修改列名两种方式3 是否替换原列表3 pandas 字母转换大小写3 使用axis参数常见问题问题&#xff1a;参数位置错误修改办法&#xff1a; rename函数简介 df.rename(inde…

关于Unity 2020找不到PBR graph的问题,shader graph 10 版本

2020.4以后采用了船新的Shader Graph&#xff0c;变动挺大的。 建议还是换版本或者去官网学习。 PBR是没有了&#xff0c;但是有两个空的Shader 一个是Blank Shader&#xff0c;即空shader&#xff0c;自己进去里面创建PBR(前排提醒&#xff0c;不是简单拖一个PBR nodes&#x…

Substance与PBR工作流总结

关于PBR PBR即基于物理的渲染&#xff0c;是一套尝试基于真实世界光照物理模型的渲染技术合集&#xff0c;使用了一种更符合物理学规律的方式来模拟光线&#xff0c;达到更真实的渲染效果&#xff0c;而且可以直接通过物理参数来直观地达到想要的结果&#xff0c;不用通过拙劣的…

BPR算法

目录 什么是BPR算法 BPR算法简介 显示反馈与隐式反馈 矩阵分解的不足 BPR算法 符号定义 BPR算法解决方式 BPR算法两个基本假设 BPR算法推导 贝叶斯定理 BPR推导 BPR算法流程 BPR算法代码与结果 数据 BPR算法代码 BPR结果展示 什么是BPR算法 BPR算法简介 BPR&…

基于物理的渲染技术(PBR)系列一

笔者介绍&#xff1a;姜雪伟&#xff0c;IT公司技术合伙人&#xff0c;IT高级讲师&#xff0c;CSDN社区专家&#xff0c;特邀编辑&#xff0c;畅销书作者&#xff0c;国家专利发明人;已出版书籍&#xff1a;《手把手教你架构3D游戏引擎》电子工业出版社和《Unity3D实战核心技术…

[引擎开发] PBR材质的原理

[本文大纲] 基础概念篇 引入 光线与介质的作用 光线的传播路径 体积散射和表面光照 光线和介质外观 微平面理论 概念介绍 中间向量 能量守恒定律 微平面理论的不足 光照计算 半球积分 …

PBR材质基础概念,限制及未来发展

最近几年图形学社区对PBR的关注非常高&#xff0c;也许是由于Disney以及一些游戏引擎大厂的助推&#xff0c;也许是因为它可以被轻松集成进实时渲染的游戏引擎当中&#xff0c;也许是因为许多人发现现在只需要调几个参数就能实现具有非常精细细节的表面着色了。反正现在网络上随…

PBR渲染(一)——PBR理论基础

PBR基础框架 PBR理论基础PBR基础框架 1.PBR理论基础 BSDF&#xff08;双向散射表面分布函数&#xff09; 对于一般的基于物理的表面材质来说&#xff0c;使用第一项BSSRDF双向散射表面反射分布函数&#xff0c;其简化后的重点就是BRDF双向反射率分布函数&#xff0c;忽略了复…

Unity ShaderGraph没有PBR Graph的解决方法

1.创建光照Shader Graph 2.打开Shader Graph界面 Fragment选项卡上右击——Add Block Note——添加Alpha和Alpha Clip Threshold两个属性 可以看到此时这两个属性是灰色的&#xff0c;并不生效&#xff0c;是因为未启用Alpha Clipping 3.Graph Inspector选项卡中&#xff0c;勾…

在3ds max中,什么是PBR材质?

PBR是Physically-Based Rendering&#xff08;基于物理的渲染)的简称。 PBR材质是一种基于物理的渲染材质&#xff0c;可提供灯光与曲面交互方式的精确表示。注&#xff1a; 仅当活动渲染器支持时&#xff0c;该材质才会显示在“材质/贴图浏览器”中。 在3ds max中PBR材质分为…

理解PBR:从原理到实现(上)

PBR 从离线渲染进入游戏引擎 CSDN 博客的公式排版有问题&#xff0c;可以看我的 GitHub Page&#xff1a;https://neil3d.github.io/unreal/pbr-theory.html 基于物理的渲染&#xff0c;即 PBR&#xff0c;Physically Based Rendering&#xff0c;这个技术名词最早是有马特法尔…

什么是PBR

PBR就是基于物理的渲染。 首先得介绍光照模型。光照模型是一种数学模型&#xff0c;也就是一种算法。这些算法用于计算机模拟物体在光照下的表现&#xff0c;具体就是 计算物体某一点处的光强。 这些模型从理论而言分为两类。 基于物理理论的和 基于感知经验的。 PBR就是基于…

BPR

BPB模型概念 BPR(Bayesian Personalized Ranking)推荐模型是基于贝叶斯后验优化的个性化排序算法。从user-iem矩阵训练出多个矩阵&#xff0c;而且一个矩阵表示一个用户的item偏好情况来获得用对多个item的偏好关系的推荐系统。本身不优化用户对物品的评分&#xff0c;只是借由…