redis日志分析

x33g5p2x  于2021-12-05 转载在 Redis  
字(10.1k)|赞(0)|评价(0)|浏览(298)

首先复习一下IO流:

关于读取文件:

  • BufferedReader

从字符输入流中读取文本,缓冲各个字符,从而提供字符、数组和行的高效读取

  • InputStreamReader

字节流通向字符流的桥梁 以UTF-8编码读取

  • FileInputStream

从文件系统中的某个文件中获取输入字节

这三个通常一起使用:

BufferedReader reader=new BufferedReader
        (new InputStreamReader(new FileInputStream(new File(path))));

终的意思是:从path指定的路径中的文件以UFT-8解码,最终生成BufferedReader 对象,读取文件内容

之前在万年历的时候讲解了Calendar类,这里补充一下Java Date类:

在 Java 中获取当前时间,可以使用 java.util.Date 类和 java.util.Calendar 类完成。其中,Date 类主要封装了系统的日期和时间的信息,Calendar 类则会根据系统的日历来解释 Date 对象

Date 类

Date 类表示系统特定的时间戳,可以精确到毫秒。Date 对象表示时间的默认顺序是星期、月、日、小时、分、秒、年。

1. 构造方法

Date 类有如下两个构造方法。

  • Date():此种形式表示分配 Date 对象并初始化此对象,以表示分配它的时间(精确到毫秒),使用该构造方法创建的对象可以获取本地的当前时间。

  • Date(long date):此种形式表示从 GMT 时间(格林尼治时间)1970 年 1 月 1 日 0 时 0 分 0 秒开始经过参数 date 指定的毫秒数。

Date date1 = new Date();    // 调用无参数构造函数
System.out.println(date1.toString());    // 输出:Wed May 18 21:24:40 CST 2016
Date date2 = new Date(60000);    // 调用含有一个long类型参数的构造函数
System.out.println(date2);    // 输出:Thu Jan 0108:01:00 CST 1970

2. 常用方法:

举个例子:假设,某一天特定时间要去做一件事,而且那个时间已经过去一分钟之后才想起来这件事还没有办,这时系统将会提示已经过去了多 长时间。具体的代码如下:

import java.util.Date;
import java.util.Scanner;
public class Test11 {
    public static void main(String[] args) {
        Scanner input = new Scanner(System.in);
        System.out.println("请输入要做的事情:");
        String title = input.next();
        Date date1 = new Date(); // 获取当前日期
        System.out.println("[" + title + "] 这件事发生时间为:" + date1);
        try {
            Thread.sleep(60000);// 暂停 1 分钟
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        Date date2 = new Date();
        System.out.println("现在时间为:" + date2);
        if (date2.before(date1)) {
            System.out.println("你还有 " + (date2.getTime() - date1.getTime()) / 1000 + " 秒需要去完成【" + title + "】这件事!");
        } else {
            System.out.println("【" + title + "】事情已经过去了 " + (date2.getTime() - date1.getTime()) / 1000 + " 秒");
        }
    }
}

在该程序中,分别使用 Date 类的无参数构造方法创建了两个 Date 对象。在创建完第一个 Date 对象后,使用 Thread.sleep() 方法让程序休眠 60 秒,然后再创建第二个 Date 对象,这样第二个 Date 对象所表示的时间将会在第一个 Date 对象所表示时间之后,因此“date2.before(date1)”条件表达式不成立,从而执行 else 块中的代码,表示事情已经发生过。

请输入要做的事情:
收快递
【收快递】这件事发生时间为:Fri Oct 12 11:11:07 CST 2018
现在时间为:Fri Oct 12 11:12:07 CST 2018
【收快递】事情已经过去了 60 秒

顺带再次回顾一下Calendar 类:

Calendar 类是一个抽象类,它为特定瞬间与 YEAR、MONTH、DAY_OF—MONTH、HOUR 等日历字段之间的转换提供了一些方法,并为操作日历字段(如获得下星期的日期) 提供了一些方法。

创建 Calendar 对象不能使用 new 关键字,因为 Calendar 类是一个抽象类,但是它提供了一个 getInstance() 方法来获得 Calendar类的对象。getInstance() 方法返回一个 Calendar 对象,其日历字段已由当前日期和时间初始化。

Calendar c = Calendar.getInstance();

Calendar类的常用方法:

Calendar 对象可以调用 set() 方法将日历翻到任何一个时间,当参数 year 取负数时表示公元前。Calendar 对象调用 get() 方法可以获取有关年、月、日等时间信息,参数 field 的有效值由 Calendar 静态常量指定。

Calendar 类中定义了许多常量,分别表示不同的意义。

  • Calendar.YEAR:年份。

  • Calendar.MONTH:月份。

  • Calendar.DATE:日期。

  • Calendar.DAY_OF_MONTH:日期,和上面的字段意义完全相同。

  • Calendar.HOUR:12小时制的小时。

  • Calendar.HOUR_OF_DAY:24 小时制的小时。

  • Calendar.MINUTE:分钟。

  • Calendar.SECOND:秒。

  • Calendar.DAY_OF_WEEK:星期几。

例如,要获取当前月份可用如下代码:

int month = Calendar.getInstance().get(Calendar.MONTH);

注意:如果整型变量 month 的值是 0,表示当前日历是在 1 月份;如果值是 11,则表示当前日历在 12 月份。

使用 Calendar 类处理日期时间的实例如下:

Calendar calendar = Calendar.getInstance(); // 如果不设置时间,则默认为当前时间
calendar.setTime(new Date()); // 将系统当前时间赋值给 Calendar 对象
System.out.println("现在时刻:" + calendar.getTime()); // 获取当前时间
int year = calendar.get(Calendar.YEAR); // 获取当前年份
System.out.println("现在是" + year + "年");
int month = calendar.get(Calendar.MONTH) + 1; // 获取当前月份(月份从 0 开始,所以加 1)
System.out.print(month + "月");
int day = calendar.get(Calendar.DATE); // 获取日
System.out.print(day + "日");
int week = calendar.get(Calendar.DAY_OF_WEEK) - 1; // 获取今天星期几(以星期日为第一天)
System.out.print("星期" + week);
int hour = calendar.get(Calendar.HOUR_OF_DAY); // 获取当前小时数(24 小时制)
System.out.print(hour + "时");
int minute = calendar.get(Calendar.MINUTE); // 获取当前分钟
System.out.print(minute + "分");
int second = calendar.get(Calendar.SECOND); // 获取当前秒数
System.out.print(second + "秒");
int millisecond = calendar.get(Calendar.MILLISECOND); // 获取毫秒数
System.out.print(millisecond + "毫秒");
int dayOfMonth = calendar.get(Calendar.DAY_OF_MONTH); // 获取今天是本月第几天
System.out.println("今天是本月的第 " + dayOfMonth + " 天");
int dayOfWeekInMonth = calendar.get(Calendar.DAY_OF_WEEK_IN_MONTH); // 获取今天是本月第几周
System.out.println("今天是本月第 " + dayOfWeekInMonth + " 周");
int many = calendar.get(Calendar.DAY_OF_YEAR); // 获取今天是今年第几天
System.out.println("今天是今年第 " + many + " 天");
Calendar c = Calendar.getInstance();
c.set(2012, 8, 8); // 设置年月日,时分秒将默认采用当前值
System.out.println("设置日期为 2012-8-8 后的时间:" + c.getTime()); // 输出时间

打印一个日历:

import java.util.Calendar;
public class CalendarDemo {
    public static void main(String[] args) {
        Calendar calendar = Calendar.getInstance();
        calendar.set(2016, 5, 1); // 实际的calendar对象所表示的日期为2016年6月1日
        // 判断2016年6月1日为一周中的第几天
        int index = calendar.get(Calendar.DAY_OF_WEEK) - 1;
        char[] title = { '日', '一', '二', '三', '四', '五', '六' }; // 存放曰历的头部
        int daysArray[][] = new int[6][7];// 存放日历的数据
        int daysInMonth = 31; // 该月的天数
        int day = 1; // 自动增长
        for (int i = index; i < 7; i++) {
            // 填充第一周的日期数据,即日历中的第一行
            daysArray[0][i] = day++;
        }
        for (int i = 1; i < 6; i++) {
            // 填充其他周的日历数据,控制行
            for (int j = 0; j < 7; j++) {
                // 如果当前day表示的是本月最后一天,则停止向数组中继续赋值
                if (day > daysInMonth) {
                    i = 6;
                    break;
                }
                daysArray[i][j] = day++;
            }
        }
        System.out.println("------------------2016 年 6 月--------------------\n");
        for (int i = 0; i < title.length; i++) {
            System.out.print(title[i] + "\t");
        }
        System.out.print("\n");
        // 输出二元数组daysArray中的元素
        for (int i = 0; i < 6; i++) {
            for (int j = 0; j < 7; j++) {
                if (daysArray[i][j] == 0) {
                    if (i != 0) {
                        // 如果到月末,则完成显示日历的任务,停止该方法的执行
                        return;
                    }
                    System.out.print("\t");
                    continue;
                }
                System.out.print(daysArray[i][j] + "\t");
            }
            System.out.print("\n");
        }
    }
}

该程序看似复杂其实很简单。因为 Calendar 类所表示的时间月份是 set() 方法中表示月份的参数值 +1,因此 Calendar 类的实际时间为 2016 年 6 月 1 日。在下面的代码中分别获取 6 月 1 日为本周中的第几天,以便在相应的星期下开始输出 6 月份的日历。程序中的 daysArray 是一个二元数组,该二元数组控制了日历的格式输出,第一个子数组控制日历的行,第二个子数组控制曰历的列,即可输出二元数组中的每一个元素。

------------------2016 年 6 月--------------------

日 一 二 三 四 五 六
   1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31

关于Hashmap的参数讲解:

习题讲解:

package 任务C__日志分析;

import java.io.*;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.*;
import java.util.stream.Collectors;

/** * @author ${范涛之} * @Description * @create 2021-12-03 10:08 */
public class Test {
    public static void main(String[] args) throws IOException {

        List<String> list = new ArrayList<>();                                           //定义集合
        int savenum = 0;                                                                 //定义存盘次数
        File file = new File("D:\\C桌面\\CSDN学习计划\\技术训练营\\redis.log");    //定义文件路径

        /** * 分开写的: */
// InputStreamReader inp = new InputStreamReader(new FileInputStream(file),"utf-8");
// BufferedReader reader = new BufferedReader(inp);

        /** * 合起来写: */
        BufferedReader reader = new BufferedReader
                (new InputStreamReader(new FileInputStream(new File(String.valueOf(file)))));

        /** * 存取文件到list集合中: */
        while (reader.read() != -1) {
            list.add(reader.readLine());
        }

        /** * 统计存盘次数,以数组中出现关键字“DB saved on disk”为基准 */
        List<String> savedCountList = list.stream()
                .filter(n -> n.contains("DB saved on disk"))
                .collect(Collectors.toList());
        for (String x : savedCountList) {
            savenum++;
        }
        // 输出存储数据的次数
        System.out.println("存盘次数为:" + savenum);

        /** * 统计出最小存盘时间和最大存盘时间 */
        //开始存储时间的数组。
        List<String> saveStartTimeList = timeList(list, "Background saving started by pid");
        //调用 toDate 方法,把字符串列表转化为 Date 列表
        List<Date> saveStartTimeDateList = toDate(saveStartTimeList);

        //完成存储时间的数组。
        List<String> saveEndTimeList = timeList(list, "DB saved on disk");

        //调用 toDate 方法,把字符串列表转化为时间列表
        List<Date> saveEndTimeDateList = toDate(saveEndTimeList);

        //存储消耗时间的数组
        List<Long> saveSpentTimeList = new ArrayList<>();
        /** * 通过结束时间减去开始时间算出时间差 */
        for (int i =0;i<saveStartTimeList.size();i++){
            saveSpentTimeList.add(saveEndTimeDateList.get(i).getTime()-saveStartTimeDateList.get(i).getTime());
        }
        long maxSaveTime = saveSpentTimeList.get(0);
        long minSaveTime = saveSpentTimeList.get(0);
        /** * 在获取的时间查里面筛选处最大和最小的 */
        for (int i = 0; i < saveSpentTimeList.size(); i++) {
            if (saveSpentTimeList.get(i) < minSaveTime) {
                minSaveTime = saveSpentTimeList.get(i);
            } else if (saveSpentTimeList.get(i) > maxSaveTime) {
                maxSaveTime = saveSpentTimeList.get(i);
            }
        }
        //输出最大、最小存储时间
        System.out.println("最大存储时间为:" + maxSaveTime + "毫秒");
        System.out.println("最小存储时间为:" + minSaveTime + "毫秒");

        //统计 * 号后出现的单词
        HashMap<String,Integer> wordMap = new HashMap<>(100);
        for (String str:list){
            String[] arr = str.split(" ");
            for (int i = 4; i <arr.length; i++) {
                // 首先判断是否是数字:
                if (!isNumeric(arr[i])){
                    // 判断集合中有没有这个单词,没有的话就加上,次数加一
                    if (!wordMap.containsKey(arr[i])){
                        wordMap.put(arr[i],1);
                    }
                    else  if (wordMap.containsKey(arr[i])){
                        Integer count = wordMap.get(arr[i]);
                        wordMap.replace(arr[i],count+1);
                    }
                }
            }
        }
        //对 wordMap 中的数据排序
        List<Map.Entry<String, Integer>> sortedWordMap = new ArrayList<Map.Entry<String, Integer>>(wordMap.entrySet());
        //输出 * 后出现的单词,及其次数
        System.out.println("单词出现的次数为(降序排列):");
        for (
                Map.Entry<String, Integer> entry : sortedWordMap) {
            System.out.println(entry.getKey() + ":" + entry.getValue() + " 次");
        }

        //统计 pid
        HashMap<String, Integer> pidMap = new HashMap<>(100);
        List<String> pidList = list.stream()
                .filter(s -> s.contains("pid"))
                .collect(Collectors.toList());

        for (String str : pidList) {
            String[] arr = str.split(" ");
            //因为 pid 在日志中是第 10 个,因此索引为 9
            //如果集合中没有该 pid,则添加,并且次数记为 1。
            if (!pidMap.containsKey(arr[9])) {
                pidMap.put(arr[9], 1);
            }
            //如果集合中有该 pid,则将其次数 +1。
            else if (pidMap.containsKey(arr[9])) {
                Integer count = pidMap.get(arr[9]);
                pidMap.replace(arr[9], count + 1);
            }
        }
        //筛选 pidMap 中 value 大于 1 的数据
        Collection<Integer> values = pidMap.values();
        while (values.contains(1)) {
            values.remove(1);
        }

        List<Map.Entry<String, Integer>> repeatedPidMap = new ArrayList<Map.Entry<String, Integer>>(pidMap.entrySet());
        //输出重复出现的pid
        System.out.println("重复出现的pid有:");
        for (
                Map.Entry<String, Integer> entry : repeatedPidMap) {
            System.out.println(entry.getKey() + ":" + entry.getValue() + " 次");
        }

    }

    /** * 会用到的方法: */
    //获取时间数组的方法,传入参数为list列表和关键字
    public static List<String> timeList(List<String> list, String keyword) {
        List<String> TimeList = new ArrayList<>();
        list.stream()
                .filter(s -> s.contains(keyword))
                .forEach(s -> {
                    s = (s.split(" ")[1] + " " + s.split(" ")[2]);
                    TimeList.add(s);
                });
        return TimeList;
    }

    // toDate方法:
    public static List<Date> toDate(List<String> stringList) {
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");
        List<Date> dateList = new ArrayList<>();
        for (String s : stringList) {
            try {
                Date d = sdf.parse(s);
                dateList.add(d);
            } catch (ParseException e) {
                e.printStackTrace();
            }
        }
        return dateList;
    }

    //判断字符串是否为数字的方法
    public static boolean isNumeric(String str) {
        return str.chars().allMatch(Character::isDigit);
    }

}

结果截图:

相关文章

微信公众号

最新文章

更多

目录