AndroidUtil - 极简日志工具类 - 异步缓存

import android.os.Environment;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Looper;
import android.os.Message;
import android.util.Log;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.text.SimpleDateFormat;
import java.util.ConcurrentModificationException;
import java.util.Locale;

public class XLogUtil {

    private static final String TAG = "XLogUtil";

    public static final String APP_VERSION = "";

    private static final String LOGPRE_SPLIT = "|";
    private static final String LOGPRE_APPVER = APP_VERSION + LOGPRE_SPLIT;
    private static final String LOGPRE_TIME_FORMAT = "yyyyMMdd-HH:mm:ss:SSS" + LOGPRE_SPLIT;

    /* 每次打印字符串的最大长度 */
    private static final int LOGCAT_MESSAGE_MAX_LENTH = 4000;

    /* 限制每个log文件最大3M */
    private static final int MAX_LOGFILE_LENGTH = 1 * 1024 * 1024;

    /* 限制log文件个数 */
    private static final int MAX_LOGFILE_NUMBER = 5;

    public static final String APP_NAME = "ahisense";
    private static String LOGFILE_PATH = null;
    private static final String FILECONTENT_ENCODE = "UTF-8";

    /* logfile缓存大小 128k */
    private static final long LOGFILE_BUFFER_MAX = 128 * 1024;

    /* 6sec */
    private static final long LOGFILE_DELAY = 6 * 1000;
    /* 5mins */
    private static final long LOGFILE_THREAD_EXIT_DELAY = 5 * 60 * 1000;

    /* 处理消息 */
    private static final int MSG_SAVE_LOG_FILE = 1;
    private static final int MSG_LOG_FILE_THREAD_EXIT = 2;

    /* 内存泄露检测开关 */
    public static final boolean DEBUG_MEMORYWATCHER = true;
    /* STRICT_MODE开关 */
    public static final boolean DEBUG_STRICT_MODE = false;

    private static StringBuilder logFileBuffer;
    private static Handler logFileHandler;

    private LogUtil() {
    }

    /**
     * 打印debug级别的log
     *
     * @param tag
     * @param msg
     */
    public static void d(String tag, Object... msg) {
        String text = buildMessage(msg, LOGPRE_APPVER);
        if (text != null) {
            int len = text.length();
            if (len <= LOGCAT_MESSAGE_MAX_LENTH) {
                Log.d(tag, text);
            } else {
                do {
                    Log.d(tag, text.substring(0, LOGCAT_MESSAGE_MAX_LENTH));
                    text = text.substring(LOGCAT_MESSAGE_MAX_LENTH);
                    len = text.length();
                } while (len > LOGCAT_MESSAGE_MAX_LENTH);
                if (len > 0) {
                    Log.d(tag, text);
                }
            }
        }

        f(tag, msg);
    }

    /**
     * 打印debug级别的log
     *
     * @param tag
     * @param msg
     */
    public static void d(boolean isSaveLog, String tag, Object... msg) {
        String text = buildMessage(msg, LOGPRE_APPVER);
        if (text != null) {
            int len = text.length();
            if (len <= LOGCAT_MESSAGE_MAX_LENTH) {
                Log.d(tag, text);
            } else {
                do {
                    Log.d(tag, text.substring(0, LOGCAT_MESSAGE_MAX_LENTH));
                    text = text.substring(LOGCAT_MESSAGE_MAX_LENTH);
                    len = text.length();
                } while (len > LOGCAT_MESSAGE_MAX_LENTH);
                if (len > 0) {
                    Log.d(tag, text);
                }
            }
        }

        if (isSaveLog) {
            f(tag, msg);
        }
    }

    /**
     * 打印warn级别的log
     *
     * @param tag
     * @param msg
     */
    public static void w(String tag, Object... msg) {
        String text = buildMessage(msg, LOGPRE_APPVER);
        if (text != null) {
            int len = text.length();
            if (len <= LOGCAT_MESSAGE_MAX_LENTH) {
                Log.w(tag, text);
            } else {
                do {
                    Log.w(tag, text.substring(0, LOGCAT_MESSAGE_MAX_LENTH));
                    text = text.substring(LOGCAT_MESSAGE_MAX_LENTH);
                    len = text.length();
                } while (len > LOGCAT_MESSAGE_MAX_LENTH);
                if (len > 0) {
                    Log.w(tag, text);
                }
            }
        }
    }

    /**
     * 打印error级别的log
     *
     * @param tag
     * @param msg
     */
    public static void e(String tag, Object... msg) {
        String text = buildMessage(msg, LOGPRE_APPVER);
        if (text != null) {
            int len = text.length();
            if (len <= LOGCAT_MESSAGE_MAX_LENTH) {
                Log.e(tag, text);
            } else {
                do {
                    Log.e(tag, text.substring(0, LOGCAT_MESSAGE_MAX_LENTH));
                    text = text.substring(LOGCAT_MESSAGE_MAX_LENTH);
                    len = text.length();
                } while (len > LOGCAT_MESSAGE_MAX_LENTH);
                if (len > 0) {
                    Log.e(tag, text);
                }
            }
        }

        f(tag, msg);
    }

    /**
     * 打印error级别的log
     *
     * @param tag
     * @param msg
     */
    public static void e(boolean isSaveLog, String tag, Object... msg) {
        String text = buildMessage(msg, LOGPRE_APPVER);
        if (text != null) {
            int len = text.length();
            if (len <= LOGCAT_MESSAGE_MAX_LENTH) {
                Log.e(tag, text);
            } else {
                do {
                    Log.e(tag, text.substring(0, LOGCAT_MESSAGE_MAX_LENTH));
                    text = text.substring(LOGCAT_MESSAGE_MAX_LENTH);
                    len = text.length();
                } while (len > LOGCAT_MESSAGE_MAX_LENTH);
                if (len > 0) {
                    Log.e(tag, text);
                }
            }
        }

        if (isSaveLog) {
            f(tag, msg);
        }
    }

    /**
     * 打印敏感信息,用户信息级别的log
     *
     * @param tag 标签
     * @param msg 日志内容
     */
    public static void s(boolean isSaveLog, String tag, Object... msg) {
        String text = buildMessage(msg, LOGPRE_APPVER);
        if (text != null) {
            int len = text.length();
            if (len <= LOGCAT_MESSAGE_MAX_LENTH) {
                Log.d(tag, text);
            } else {
                do {
                    Log.d(tag, text.substring(0, LOGCAT_MESSAGE_MAX_LENTH));
                    text = text.substring(LOGCAT_MESSAGE_MAX_LENTH);
                    len = text.length();
                } while (len > LOGCAT_MESSAGE_MAX_LENTH);
                if (len > 0) {
                    Log.d(tag, text);
                }
            }
        }

        if (isSaveLog) {
            f(tag, msg);
        }
    }

    /**
     * 保存log到文件
     *
     * @param tag
     * @param msg
     */
    public static void f(String tag, Object... msg) {
        String text = buildMessage(msg, getTimePrefix(), tag + LOGPRE_SPLIT, LOGPRE_APPVER);
        if (text != null) {
            putLogFileBuffer(text);
        }
    }


    private static String getTimePrefix() {
        SimpleDateFormat df = new SimpleDateFormat(LOGPRE_TIME_FORMAT, Locale.ENGLISH);
        return df.format(System.currentTimeMillis());
    }

    private static String buildMessage(Object[] msg, String... prefix) {
        if (msg == null || msg.length == 0) {
            return null;
        }

        StringBuilder sb = new StringBuilder();
        try {
            if (prefix != null) {
                for (String p : prefix) {
                    sb.append(p);
                }
            }
            for (Object m : msg) {
                sb.append(m != null ? m : "");
            }
        } catch (ConcurrentModificationException e) {
            Log.e(TAG, e.getMessage());
            return "";
        }
        return sb.toString();
    }


    /**
     * 向文件中写文本内容
     *
     * @param file
     * @param content
     * @param append
     */
    public static void writeFile(File file, String content, boolean append) {
        if (null == file || null == content) {
            return;
        }
        byte[] bytes = null;
        try {
            bytes = content.getBytes(FILECONTENT_ENCODE);
        } catch (UnsupportedEncodingException e) {
            w(TAG, "write file failed: ", e.getMessage());
        }
        if (bytes != null) {
            writeFile(file, bytes, append);
        }
    }

    /**
     * 向文件中写二进制内容
     *
     * @param file
     * @param content
     * @param append
     */
    public static void writeFile(File file, byte[] content, boolean append) {
        FileOutputStream out = null;
        try {
            out = new FileOutputStream(file, append);
            out.write(content);
            out.flush();
        } catch (IOException e) {
            w(TAG, "write file failed: ", e.getMessage());
        } finally {
            if (out != null) {
                try {
                    out.close();
                } catch (IOException e) {
                }
            }
        }
    }

    private static File getLogFile() {
        if (LOGFILE_PATH == null) {
            try {
                String sdcard = Environment.getExternalStorageDirectory().getCanonicalPath();
                LOGFILE_PATH = sdcard + "/" + APP_NAME + "/";
            } catch (IOException e) {
                Log.e(TAG, "getExternalStorageDirectory failed!");
            }
        }

        File path = new File(LOGFILE_PATH);
        if (!path.exists()) {
            if (!path.mkdirs()) {
                Log.e(TAG, "create log directory failed!");
                return null;
            }
        }

        File file = new File(LOGFILE_PATH + "log.0");
        if (file.exists() && file.length() > MAX_LOGFILE_LENGTH) {
            File tmp = new File(LOGFILE_PATH + "log." + (MAX_LOGFILE_NUMBER - 1));
            if (tmp.exists()) {
                if (!tmp.delete()) {
                    Log.e(TAG, "delete log file failed");
                    return null;
                }
            }
            for (int i = MAX_LOGFILE_NUMBER - 2; i >= 0; i--) {
                tmp = new File(LOGFILE_PATH + "log." + i);
                if (tmp.exists()) {
                    if (!tmp.renameTo(new File(LOGFILE_PATH + "log." + (i + 1)))) {
                        Log.e(TAG, "rename log file failed");
                        return null;
                    }
                }
            }
        }
        return file;
    }


    private static void putLogFileBuffer(String text) {
        synchronized (LogUtil.class) {
            if (logFileBuffer == null) {
                logFileBuffer = new StringBuilder(text);
            } else {
                logFileBuffer.append(text);
            }
            logFileBuffer.append("\n");

            boolean saveNow = logFileBuffer.length() >= LOGFILE_BUFFER_MAX;
            // schedule log file buffer saving
            if (logFileHandler == null) {
                HandlerThread thread = new HandlerThread("logfile_thread");
                thread.start();
                logFileHandler = new LogFileHandler(thread.getLooper());
                logFileHandler.sendEmptyMessageDelayed(MSG_SAVE_LOG_FILE, saveNow ? 0 : LOGFILE_DELAY);
            } else {
                logFileHandler.removeMessages(MSG_LOG_FILE_THREAD_EXIT);
                if (saveNow) {
                    logFileHandler.removeMessages(MSG_SAVE_LOG_FILE);
                    logFileHandler.sendEmptyMessage(MSG_SAVE_LOG_FILE);
                } else if (!logFileHandler.hasMessages(MSG_SAVE_LOG_FILE)) {
                    logFileHandler.sendEmptyMessageDelayed(MSG_SAVE_LOG_FILE, LOGFILE_DELAY);
                }
            }
        }
    }

    private static void flushLogFileBufferLocked() {
        synchronized (LogUtil.class) {
            if (logFileBuffer == null) {
                return;
            }

            String text = logFileBuffer.toString();
            logFileBuffer = null;
            File file = getLogFile();
            if (file != null) {
                writeFile(file, text, true);
            } else {
                w(TAG, "get log file failed.");
            }
        }
    }

    private static class LogFileHandler extends Handler {

        public LogFileHandler(Looper looper) {
            super(looper);
        }

        @Override
        public void handleMessage(Message msg) {
            if (null == msg) {
                return;
            }
            switch (msg.what) {
                case MSG_SAVE_LOG_FILE:
                    synchronized (LogUtil.class) {
                        logFileHandler.removeMessages(MSG_SAVE_LOG_FILE);
                        flushLogFileBufferLocked();
                        logFileHandler.sendEmptyMessageDelayed(MSG_LOG_FILE_THREAD_EXIT, LOGFILE_THREAD_EXIT_DELAY);
                    }
                    break;

                case MSG_LOG_FILE_THREAD_EXIT:
                    synchronized (LogUtil.class) {
                        if (!logFileHandler.hasMessages(MSG_SAVE_LOG_FILE)) {
                            logFileHandler.getLooper().quit();
                            logFileHandler = null;
                        }
                    }
                    break;

                default:
                    break;
            }
        }
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
一、简介当下Java后端的SpringBoot微服务框架大火,原因离不开注解的使用,其简单易配置的注解方式使得更多的社区为其编写适用于SpringBoot的框架,也就是注解逐渐取代了传统的xml配置方式。那么注解在Android中也同样的得到了升华,著名的框架有ButterKnife、 Dagger2、Retrofit等等。今天带来一款Android中比较实用的注解框架AopArms,其用法简单,里面编写了Android开发中常用的一套注解,如日志异步处理、缓存、SP、延迟操作、定时任务、重试机制、try-catch安全机制、过滤频繁点击等,后续还会有更多更强大的注解功能加入。 本篇主要内容讲解在Android中的基本用法,关于AOP在Android中的实践请参考另外一篇Android开发之AOP编程。二、引入方式1、在主工程中添加依赖//引入aspectjx插件 apply plugin: 'android-aspectjx' dependencies {     ...     implementation 'cn.com.superLei:aop-arms:1.0.2' }2、项目跟目录的gradle脚本中加入buildscript {     repositories {         mavenCentral()     }     dependencies {         //该库基于沪江aspect插件库         classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.4'     } }3、在Application中初始化AopArms.init(this);三、基本使用1、缓存篇(可缓存任意类型)1、插入缓存     /**      * key:缓存的键      * expiry:缓存过期时间,单位s      * @return 缓存的值      */     @Cache(key = "userList", expiry = 60 * 60 * 24)     private ArrayList<User> initData() {         ArrayList<User> list = new ArrayList<>();         for (int i=0; i<5; i ){             User user = new User();             user.setName("艾神一不小心:" i);             user.setPassword("密码:" i);             list.add(user);         }         return list;     }      2、获取缓存     private ArrayList<User> getUser() {         return ArmsCache.get(this).getAsList("userList", User.class);     } 3、移除缓存     /**      * key:缓存的键      * beforeInvocation:缓存的清除是否在方法之前执行, 如果出现异常缓存就不会清除   默认false      * allEntries:是否清空所有缓存(与key互斥)  默认false      */     @CacheEvict(key = "userList", beforeInvocation = true, allEntries = false)     public void removeUser() {         Log.e(TAG, "removeUser: >>>>");     }2、SharedPreferences篇(可保存对象)1、保存key到sp     @Prefs(key = "article")     private Article initArticle() {         Article article = new Article();         article.author = "jerry";         article.title = "hello android";         article.createDate = "2019-05-31";         article.content = "this is a test demo";         return article;     }      2、从sp中移除key     /**      * key:sp的键      * allEntries:是否清空所有存储(与key互斥)  默认false      */     @PrefsEvict(key = "article", allEntries = false)     public void removeArticle() {         Log.e(TAG, "removeArticle: >>>>");     } 3、通过key从sp中获取value     public void getArticle() {         Article article = ArmsPreference.get(this, "article", null);         Log.e(TAG, "getArticle: " article);     }3、异步篇@Async     public void asyn() {         Log.e(TAG, "useAync: " Thread.currentThread().getName());     }4、try-catch安全机制篇//自动帮你try-catch   允许你定义回调方法     @Safe(callBack = "throwMethod")     public void safe() {         String str = null;         str.toString();     }          //自定义回调方法(注意要和callBack的值保持一致)     private void throwMethod(Throwable throwable){         Log.e(TAG, "throwMethod: >>>>>" throwable.toString());     }5、重试机制篇/**      * @param count 重试次数      * @param delay 每次重试的间隔      * @param asyn 是否异步执行      * @param retryCallback 自定义重试结果回调      * @return 当前方法是否执行成功      */     @Retry(count = 3, delay = 1000, asyn = true, retryCallback = "retryCallback")     public boolean retry() {         Log.e(TAG, "retryDo: >>>>>>" Thread.currentThread().getName());         return false;     }          private void retryCallback(boolean result){         Log.e(TAG, "retryCallback: >>>>" result);     }6、定时任务篇/**      * @param interval 初始化延迟      * @param interval 时间间隔      * @param timeUnit 时间单位      * @param count 执行次数      * @param taskExpiredCallback 定时任务到期回调      */     @Scheduled(interval = 1000L, count = 10, taskExpiredCallback = "taskExpiredCallback")     public void scheduled() {         Log.e(TAG, "scheduled: >>>>");     }          private void taskExpiredCallback(){         Log.e(TAG, "taskExpiredCallback: >>>>");     }7、延迟任务篇//开启延迟任务(10s后执行该方法)     @Delay(key = "test", delay = 10000L)     public void delay() {         Log.e(TAG, "delay: >>>>>");     }          //移除延迟任务     @DelayAway(key = "test")     public void cancelDelay() {         Log.e(TAG, "cancelDelay: >>>>");     }8、过滤频繁点击//value默认500ms     @SingleClick(value = 2000L)     private void onclick(){         Log.e(TAG, "onclick: >>>>");     }9、拦截篇(如登录)1、在需要进行拦截的方法添加注解     @Intercept("login_intercept")     public void loginIntercept() {         Log.e(TAG, "intercept: 已登陆>>>>");     } 2、(建议,统一处理)在Application中进行进行监听拦截回调 public class MyApplication extends Application {     private static final String TAG = "MyApplication";     private static MyApplication mApplication;     @Override     public void onCreate() {         super.onCreate();         mApplication = this;         AopArms.init(this);         AopArms.setInterceptor(new Interceptor() {             @Override             public boolean intercept(String key, String methodName) throws Throwable {                 Log.e(TAG, "intercept methodName:>>>>>" methodName);                 if ("login_intercept".equals(key)){                     String userId = ArmsPreference.get(mApplication, "userId", "");                     if (TextUtils.isEmpty(userId)){                         Toast.makeText(mApplication, "您还没有登录", Toast.LENGTH_SHORT).show();                         return true;//代表拦截                     }                 }                 return false;//放行             }         });     } }四、参考HujiangTechnology/gradle_plugin_android_aspectjx深入理解Android之AOP

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值