android 异步写入日志到文件中


  1. 有些android设备关闭了logcat日志输出,想要查看程序的日志信息就得将log写到文件中.
  2. 写文件是耗时操作,为不影响性能,需要异步线程处理.
  3. log记录是很频繁的操作,不该每次记录日志都创建线程,否则CPU开销太大,反倒影响了性能,需要保持线程常在,有需要时,再工作.

外部调用:

AsyncLogger.Logging("TAG", "my log info...");

AsyncLogger.LoggingE("TAG", "description",exception);


代码实现:

package com.example.mydemo;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.ConcurrentLinkedQueue;

import android.os.Environment;
import android.util.Log;

public class AsyncLogger {
	private static volatile AsyncLogger instance = null;
	private SimpleDateFormat mFormat = null;
	private WriteThread mThread = null;

	private AsyncLogger() {
		mThread = new WriteThread();
		mFormat = new SimpleDateFormat("MM-dd HH:mm:ss:SS");
		mThread.start();
	}

	//使用安全的单例模式
	public static AsyncLogger getInstance() {
		if (instance == null) {
			synchronized (AsyncLogger.class) {
				if (instance == null) {
					instance = new AsyncLogger();
				}
			}
		}
		return instance;
	}

	//外部直接调用该方法打印Exception堆栈
	public static void LoggingE(String tag, String str, Exception e) {
		Log.e(tag, str,e);
		//将Exception的错误信息转换成String
		String log = "";
		try {
			StringWriter sw = new StringWriter();
			PrintWriter pw = new PrintWriter(sw);
			e.printStackTrace(pw);
			log = str + "\r\n" + sw.toString() + "\r\n";
		} catch (Exception e2) {
			log = str + " fail to print Exception";
		}
		AsyncLogger.getInstance().Log(tag, str);
	}
	//外部直接调用该方法打印Log
	public static void Logging(String tag, String str) {
		Log.e(tag, str);
		AsyncLogger.getInstance().Log(tag, str);
	}

	public synchronized void Log(String tag, String str) {
		String time = mFormat.format(new Date());
		mThread.enqueue(time + " " + tag + " " + str);
	}

	//线程保持常在,不工作时休眠,需要工作时再唤醒就可.
	public class WriteThread extends Thread {
		private boolean isRunning = false;
		private String filePath = null;
		private Object lock = new Object();
		private ConcurrentLinkedQueue<String> mQueue = new ConcurrentLinkedQueue<String>();

		public WriteThread() {
			String sdcard = getPath();
			if (sdcard != null)
				filePath = sdcard + "/wusn_log.txt";
			isRunning = true;
		}
	
		public String getPath() {
			return exist() ? Environment.getExternalStorageDirectory().toString() : null;
		}

		public boolean exist() {
			return Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED);
		}

		//将需要写入文本的字符串添加到队列中.线程休眠时,再唤醒线程写入文件
		public void enqueue(String str) {
			mQueue.add(str);
			if (isRunning() == false) {
				awake();
			}
		}

		public boolean isRunning() {
			return isRunning;
		}

		public void awake() {
			synchronized (lock) {
				lock.notify();
			}
		}
		@Override
		public void run() {
			while (true) {
				synchronized (lock) {
					isRunning = true;
					while (!mQueue.isEmpty()) {
						try {
							//pop出队列的头字符串写入到文件中
							recordStringLog(mQueue.poll());
						} catch (Exception e) {
							e.printStackTrace();
						}
					}
					isRunning = false;
					try {
						lock.wait();
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}

			}
		}

		public void recordStringLog(String text) {
			File file = new File(filePath);
			if (!file.exists()) {
				file.getParentFile().mkdirs();
				try {
					file.createNewFile();
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
			try {
				//append data to file
				FileWriter filerWriter = new FileWriter(file, true);
				BufferedWriter bufWriter = new BufferedWriter(filerWriter);
				bufWriter.write(text);
				bufWriter.newLine();
				bufWriter.close();
				filerWriter.close();
			} catch (IOException e) {
				e.printStackTrace();
			}
		}

		/**
		 * 判断日志文件是否存在
		 * 
		 * @return
		 */
		public boolean isExitLogFile() {
			File file = new File(filePath);
			if (file.exists() && file.length() > 3) {
				return true;
			} else {
				return false;
			}
		}

		/**
		 * 删除日志文件
		 */
		public void deleteLogFile() {
			File file = new File(filePath);
			if (file.exists()) {
				file.delete();
			}
		}
	}
}
PS:记得在代码中注册SD卡访问权限

参考了网上的一些资料,向资料原创者表示感谢

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值