Log日志框架-对原生Log进行简单封装

闲来无事,对以前项目中使用的对Log的封装使用抽取出来,写成一个Demo供博友参考。
Demo是以Gradle构建的项目。其中涉及到打Release包跟Debug包的时候对于日志输出的控制,由gradle脚本进行控制,下文我再细说。
先看下日志的输出效果:
这里写图片描述
效果比较简洁,主要展示日志的具体来源及日志消息,并提供点击快速定位代码的功能。
下面直接看代码:

package com.csf.simplelog.utils;

import android.text.TextUtils;
import android.util.Log;

/**
 * ClassName: LogUtil
 * Description:日志工具类
 * Created by chensf on 2016-7-27 9:53.
 */
public class LogUtil {

    private static final String TAG = LogUtil.class.getSimpleName();
    /**
     * 日志输出等级
     */
    private static int LOG_LEVEL = Log.VERBOSE;

    /**
     * 是否显示日志
     */
    private static boolean isShowLog = true;

    private static final String DOUBLE_DIVIDER = "-----------------------------------------------------------------------------------------------------------";

    public static void init(boolean isShowLog) {
        LogUtil.isShowLog = isShowLog;
    }

    public static void init(boolean isShowLog, int logLevel) {
        LogUtil.isShowLog = isShowLog;
        LogUtil.LOG_LEVEL = logLevel;
    }

    public static int v(Object msg) {
        return v(TAG, msg);
    }

    public static int v(String tag, Object msg) {
        return v(tag, msg, null);
    }

    public static int v(String tag, Object msg, Throwable tr) {
        return printLog(Log.VERBOSE, tag, msg, tr);
    }

    public static int d(Object msg) {
        return d(TAG, msg);
    }

    public static int d(String tag, Object msg) {
        return d(tag, msg, null);
    }

    public static int d(String tag, Object msg, Throwable tr) {
        return printLog(Log.DEBUG, tag, msg, tr);
    }

    public static int i(Object msg) {
        return i(TAG, msg);
    }

    public static int i(String tag, Object msg) {
        return i(tag, msg, null);
    }

    public static int i(String tag, Object msg, Throwable tr) {
        return printLog(Log.INFO, tag, msg, tr);
    }

    public static int w(Object msg) {
        return w(TAG, msg);
    }

    public static int w(String tag, Object msg) {
        return w(tag, msg, null);
    }

    public static int w(String tag, Object msg, Throwable tr) {
        return printLog(Log.WARN, tag, msg, tr);
    }

    public static int e(Object msg) {
        return e(TAG, msg);
    }

    public static int e(String tag, Object msg) {
        return e(tag, msg, null);
    }

    public static int e(String tag, Object msg, Throwable tr) {
        return printLog(Log.ERROR, tag, msg, tr);
    }

    private static int printLog(int type, String tag, Object msgObj, Throwable tr) {
        if (!isShowLog) {
            return 0;
        }
        String msg;
        StringBuilder builder = new StringBuilder(DOUBLE_DIVIDER).append('\n').append(getFunctionName())
                .append(DOUBLE_DIVIDER).append('\n').append("  ");
        if (msgObj == null) {
            msg = "";
        } else {
            msg = msgObj.toString();
        }
        if (!TextUtils.isEmpty(msg)) {
            builder.append(msg);
        }
        if (tr != null) {
            builder.append('\n').append(Log.getStackTraceString(tr));
        }
        builder.append('\n').append(DOUBLE_DIVIDER);
        switch (type) {
            case Log.VERBOSE:
                if (LOG_LEVEL <= Log.VERBOSE) {
                    return Log.v(tag, builder.toString());
                }
                break;
            case Log.DEBUG:
                if (LOG_LEVEL <= Log.DEBUG) {
                    return Log.d(tag, builder.toString());
                }
                break;
            case Log.INFO:
                if (LOG_LEVEL <= Log.INFO) {
                    return Log.i(tag, builder.toString());
                }
                break;
            case Log.WARN:
                if (LOG_LEVEL <= Log.WARN) {
                    return Log.w(tag, builder.toString());
                }
                break;
            case Log.ERROR:
                if (LOG_LEVEL <= Log.ERROR) {
                    return Log.e(tag, builder.toString());
                }
                break;
        }
        return 0;
    }

    private static String getFunctionName() {
        StackTraceElement[] elements = Thread.currentThread().getStackTrace();
        if (elements == null) {
            return "";
        }
        for (StackTraceElement ste : elements) {
            if (ste.isNativeMethod()) {
                continue;
            }
            if (ste.getClassName().equals(Thread.class.getName())) {
                continue;
            }
            if (ste.getClassName().equals(LogUtil.class.getName())) {
                continue;
            }
            return "  " + ste.getFileName().substring(0, ste.getFileName().indexOf(".")) + "." + ste.getMethodName()
                    + " (" + ste.getFileName() + ":" + ste.getLineNumber() + ")\n";
        }
        return "";


    }

}

167行的代码,其中提供快速定位功能的代码在getFunctionName这个方法中,通过循环获取Stack Trace Element中的element,分析获取到具体的调用类方法及对应行数。代码如下:

private static String getFunctionName() {
        StackTraceElement[] elements = Thread.currentThread().getStackTrace();
        if (elements == null) {
            return "";
        }
        for (StackTraceElement ste : elements) {
            if (ste.isNativeMethod()) {
                continue;
            }
            if (ste.getClassName().equals(Thread.class.getName())) {
                continue;
            }
            if (ste.getClassName().equals(LogUtil.class.getName())) {
                continue;
            }
            return "  " + ste.getFileName().substring(0, ste.getFileName().indexOf(".")) + "." + ste.getMethodName()
                    + " (" + ste.getFileName() + ":" + ste.getLineNumber() + ")\n";
        }
        return "";

下面我们将目光回到对Log的封装类LogUtil上,LogUtil只是对原生Log类进行简单的封装,通过增加属性isShowLog及LOG_LEVEL对我们的log进行控制,方便在发布Release包的时候快速关闭日志输出。
LogUtil提供两个初始函数,

public static void init(boolean isShowLog) {
        LogUtil.isShowLog = isShowLog;
    }

    public static void init(boolean isShowLog, int logLevel) {
        LogUtil.isShowLog = isShowLog;
        LogUtil.LOG_LEVEL = logLevel;
    }

这两个函数二选一,最好是在应用的入口进行初始化,一般我们会自定义一个Application的继承类。在其方法onCreate中就能进行应用的初始工作。
代码如下:

这里写图片描述
然后在AndroidManifest中引用这个MyApplication类

   <application
        android:name=".config.MyApplication"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme" >

在MyApplication的onCreate方法中,我们看到,我这里通过
BuildConfig.LOG_DEBUG这个值来控制日志是否输出。
我们来看这个语句LogUtil.init(true, Log.VERBOSE);其中参一是是否开启日志输出,参二顾名思义,就是日志的级别,这里我将级别控制设置为最低的VERBOSE,意思就是所有的日志信息,VERBOSE,DEBUG,INFO,WARN,ERROR这些级别的信息都会输出。如果传递参数Log.ERROR,那就只会输出错误日志,因为我们的Log.ERROR的级别比VERBOSE,DEBUG,INFO,WARN都要高。具体的大家去看到Demo的效果。
回到上面的BuildConfig.LOG_DEBUG,机智的小伙伴会发现BuildConfig中并没有LOG_DEBUG这个属性,那么我们来讲下开文提到
“涉及到打Release包跟Debug包的时候对于日志输出的控制,由gradle脚本进行控制”
通过Gradle脚本来进行控制日志的输出,这样我们就不用在打包的时候手动去修改代码。
下面贴出部分Gradle脚本:

 buildTypes {
        debug {
            //显示日志
            buildConfigField("boolean","LOG_DEBUG","true");
        }
        release {
            //不显示日志
            buildConfigField("boolean","LOG_DEBUG","false");
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }

在主工程的Gradle脚本的buildTypes下,在debug跟release中增加buildConfigField(“boolean”,”LOG_DEBUG”,”true”);
这个语句的意思是在BuildConfig中增加字段LOG_DEBUG,debug下LOG_DEBUG的值为true,release下则为false.然后再通过该值去初始化LogUtil,达到控制日志输出。
大家如果还有什么疑问,可以留言跟我一起进行交流讨论。
Demo下载地址: http://download.csdn.net/download/q919233914/9587719

转载注明出处: http://blog.csdn.net/q919233914/article/details/52046667

  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
在uni-app中,可以通过原生插件来实现一些uni-app本身不支持的功能或者性能优化。下面是一个简单的uni-app原生插件的编写示例: 1. 创建一个uni-app项目,并选择需要封装原生插件。 2. 在项目根目录下创建一个platforms目录,并在该目录下创建一个android和ios目录,用来存放对应平台的原生代码。 3. 在android目录下创建一个Android Studio项目,并将需要封装原生插件添加到该项目中。 4. 在Android Studio中创建一个类,继承自com.dcloud.bridge.NativeAPI类,示例代码如下: ```java public class MyPlugin extends NativeAPI { public String sayHello(String name) { return "Hello " + name + "!"; } } ``` 5. 在Android Studio中创建一个类,继承自com.dcloud.bridge.Wrapper类,示例代码如下: ```java public class MyPluginWrapper extends Wrapper { public MyPluginWrapper(Context context) { super(context); } @Override public Object onMessage(String id, JSONObject data) { if (TextUtils.equals("sayHello", id)) { String name = data.optString("name"); MyPlugin myPlugin = new MyPlugin(); String result = myPlugin.sayHello(name); JSONObject json = new JSONObject(); try { json.put("result", result); } catch (JSONException e) { e.printStackTrace(); } return json; } return super.onMessage(id, data); } } ``` 6. 在Android Studio中创建一个类,继承自com.dcloud.bridge.Plugin类,示例代码如下: ```java public class MyPluginPackage extends PluginPackage { @Override public Wrapper createWrapper(Context context) { return new MyPluginWrapper(context); } } ``` 7. 在Android Studio中创建一个类,继承自com.dcloud.bridge.PluginManager类,示例代码如下: ```java public class MyPluginManager extends PluginManager { public MyPluginManager(Application application) { super(application); registerPackage("myplugin", new MyPluginPackage()); } } ``` 8. 修改AndroidManifest.xml文件,示例代码如下: ```xml <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.uniappmyplugin"> <application android:name=".MyApplication" android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" 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> <service android:name=".MyService" /> <receiver android:name=".MyReceiver" /> <provider android:name=".MyProvider" /> <meta-data android:name="uniplugin" android:value="true" /> <meta-data android:name="uniplugin_name" android:value="myplugin" /> </application> </manifest> ``` 9. 在uni-app项目的manifest.json文件中添加以下代码: ```json "app-plus": { "android": { "useAndroidX": true, "permission": [ { "name": "android.permission.CAMERA", "desc": "摄像头权限", "def": "android.permission.CAMERA", "protectionLevel": "dangerous" } ], "gradleProperties": { "android.useAndroidX": true }, "dependencies": { "libs": [ { "path": "libs/xxx.aar" // xxx.aar为你的插件文件名 } ] } } } ``` 10. 在uni-app项目中调用插件方法,示例代码如下: ```javascript const plugin = uni.requireNativePlugin('myplugin') plugin.sayHello('world').then(res => { console.log(res.result) }).catch(err => { console.log(err) }) ``` 以上就是一个简单的uni-app原生插件的编写示例,具体实现方式可能会因插件需求而异。需要注意的是,uni-app原生插件开发需要具备一定的Android/iOS开发经验,如果你不熟悉原生开发,可以参考uni-app提供的插件市场,寻找已有的插件。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值