【碼歌】JAVA8實用——快速寫入和讀取文件

Java8快速讀取和寫入文件.png

話不多說,先看題:

隨機生成 Salary {name, baseSalary, bonus }的記錄,如“wxxx,10,1”,每行一條記錄,總共1000萬記錄,寫入文本文件(UFT-8編碼), 然后讀取文件,name的前兩個字符相同的,其年薪累加,比如wx,100萬,3個人,最后做排序和分組,輸出年薪總額最高的10組:

wx 200萬 10人
lt 180萬 8人
... ... ...

name 4位a-z隨機,baseSalary [0,100]隨機 bonus[0-5]隨機 ,年薪總額 = baseSalary*13 + bonus。

思路:

第一步:先編寫一個Salary對象

里面包含有name,baseSalary, bounus屬性,然后編寫一個構(gòu)造器,重寫toString()方法方便序列化數(shù)據(jù),同時編寫構(gòu)建Salary對象的方法build();

/**
 *  name 4位a-z隨機
 *  baseSalary 0-100隨機
 *  bonus 0-5隨機
 *  年薪總額 = baseSalary * 13 + bonus
 */
class Salary {
    // name 4位a-z隨機,baseSalary 0-100隨機,bonus 0-5隨機 年薪總額 = baseSalary * 13 + bonus
    private String name;
    private int baseSalary;
    private int bonus;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getBaseSalary() {
        return baseSalary;
    }

    public void setBaseSalary(int baseSalary) {
        this.baseSalary = baseSalary;
    }

    public int getBonus() {
        return bonus;
    }

    public void setBonus(int bonus) {
        this.bonus = bonus;
    }


    public Salary() {

    }

    public Salary(String name, int baseSalary, int bonus) {
        this.name = name;
        this.baseSalary = baseSalary;
        this.bonus = bonus;
    }

    public Salary build() {
        this.name = getRandomName(4);
        // 0-100隨機數(shù)
        this.baseSalary = (int)(100 * Math.random());
        // 0-5隨機數(shù)
        this.bonus = (int)(5 * Math.random());
        return this;
    }

    @Override
    public String toString() {
        return name + " " + baseSalary + " " + bonus;
    }

    /**
     * 生產(chǎn)Name隨機函數(shù) 4位a-z隨機
     * @param length
     * @return
     */
    private static String getRandomName(int length ){
        String base = "abcdefghijklmnopqrstuvwxyz";
        Random random = new Random();
        StringBuffer sb = new StringBuffer();
        for ( int i = 0; i < length; i++ ){
            int number = random.nextInt(base.length());
            sb.append(base.charAt(number));
        }
        return sb.toString();
    }
}

第二步:編寫寫入方法

? a) 第一種方式,使用Java的IO中BufferedWriter寫入文件

/**
 * 寫入文件
 * @return
 * @throws IOException
 */
public static File writeBuffer() throws IOException {
    File file = new File(FILE_NAME);
    FileOutputStream fos = new FileOutputStream(file);
    BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(fos));
    int i = AMOUNT;
    while(i > 0) {
        Salary salary = new Salary().build();
        writer.write(salary.toString() + "\r\n");
        i --;
    }
    writer.close();
    fos.close();
    return file;
}

? b) 第二種方式,使用JAVA的NIO中的FileChannel進行寫入

/**
 * NIO進行寫入
 * @throws IOException
 */
private static void writeNIO() throws IOException {
    FileOutputStream fos = new FileOutputStream(FILE_NAME, true);
    FileChannel channel = fos.getChannel();
    int i = AMOUNT;
    StringBuffer content = new StringBuffer();
    while(i > 0) {
        Salary salary = new Salary().build();
        content.append(salary.toString()).append("\r\n");
        i --;
    }
    ByteBuffer buf = ByteBuffer.wrap(content.toString().getBytes());
    buf.put(content.toString().getBytes());
    buf.flip();
    channel.write(buf);
    channel.close();
    fos.close();

}

比較下來,單純從這個代碼的性能上講差不太多!

第三步:讀取并解析文件在進行排序

? a) 通過JAVA的IO進行讀?。?/p>

/**
 * Java IO讀取文件的方式
 * @return
 * @throws Exception
 */
public static List<Salary> readFileIO() throws Exception {
    File file = new File(FILE_NAME);
    List<Salary> list = new ArrayList<>();
    InputStreamReader reader = new InputStreamReader(new FileInputStream(file)); // 建立一個輸入流對象reader
    BufferedReader br = new BufferedReader(reader); // 建立一個對象,它把文件內(nèi)容轉(zhuǎn)成計算機能讀懂的語言
    String line = ""; // 每一行的內(nèi)容
    int i = 1;
    while ((line = br.readLine()) != null) {
        String[] split = line.trim().split(" ");// .trim()可以去掉首尾多余的空格
        list.add(new Salary(split[0], Integer.valueOf(split[1]), Integer.valueOf(split[2]))); // 添加一個Salary實體
        i++;
    }
    reader.close();
    br.close();
    return list;
}

? b) 通過JAVA的NIO讀?。?/p>

/**
 * JDK8 NIO讀取文件
 * @return
 * @throws Exception
 */
public static List<Salary> readFileNIO() throws Exception {
    List<Salary> list = new ArrayList<>();
    Files.lines(Paths.get(FILE_NAME)).forEach(line -> {
        String[] split = line.trim().split(" ");// .trim()可以去掉首尾多余的空格
        list.add(new Salary(split[0], Integer.valueOf(split[1]), Integer.valueOf(split[2]))); // 添加一個Salary實體
    });
    return list;
}

二者比較下來:使用jdk1.8讀取更簡單,效率性能更高!

第四步:使用JDK8的Stream進行排序和截取

/**
 * 排序并獲取前十?dāng)?shù)據(jù)
 * @param salaries
 */
public static void sort(List<Salary> salaries) {
    Map<String, GroupSalary> result = new HashMap<>();
    salaries.forEach(salary -> {
        String shortName = salary.getName().substring(0, 2);
        GroupSalary groupSalary = null;
        List<Salary> salaryList = null;
        if (result.containsKey(shortName)) {
            groupSalary = result.get(shortName);
            salaryList = groupSalary.getSalaries();
        } else {
            groupSalary = new GroupSalary();
            salaryList = new ArrayList<>();
            groupSalary.setSalaries(salaryList);
        }
        salaryList.add(salary);
        groupSalary.setShortName(shortName);
        groupSalary.setTotal(groupSalary.getTotal() + salary.getBaseSalary() * 13 + salary.getBonus());
        result.put(shortName, groupSalary);
    });

    List<GroupSalary> r = result.entrySet().stream()
            .sorted((Map.Entry<String, GroupSalary> o1, Map.Entry<String, GroupSalary> o2) -> o2.getValue().getTotal() - o1.getValue().getTotal())
            .map(entry -> entry.getValue()).collect(Collectors.toList()).subList(0,10);

    r.forEach(groupSalary -> {
        System.out.println(groupSalary.getShortName() + " " + groupSalary.getTotal() + " " + groupSalary.getSalaries().size());
    });
}

同時將數(shù)據(jù)封裝在GroupSalary中

class GroupSalary {
    private List<Salary> salaries;
    private String shortName;
    private int total;

    public List<Salary> getSalaries() {
        return salaries;
    }

    public void setSalaries(List<Salary> salaries) {
        this.salaries = salaries;
    }

    public String getShortName() {
        return shortName;
    }

    public void setShortName(String shortName) {
        this.shortName = shortName;
    }

    public int getTotal() {
        return total;
    }

    public void setTotal(int total) {
        this.total = total;
    }


}

到此如果寫入100W數(shù)據(jù)基本上1分鐘不到就可以完成寫入和讀取,歡迎大家進行改寫和優(yōu)化。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,711評論 19 139
  • 2018-03-1 (稻盛哲學(xué)學(xué)習(xí)會)打卡第25天 姓名:王燕君 部門:分水碶 組別:待定 【知~學(xué)習(xí)】 誦讀《活...
    王燕君分水碶閱讀 218評論 0 0
  • 字符串函數(shù)char_length(字符串)字符數(shù)length(字符串)字節(jié)數(shù)left(字符串,length)獲得左...
    ssttIsme閱讀 157評論 0 1
  • 學(xué)習(xí)感悟:愛上自己的工作,從工作中找樂趣,這樣就不會感覺到自己很累。 1、付出不亞于任何人的努力7分 2、要謙虛不...
    管宏洋閱讀 177評論 0 0

友情鏈接更多精彩內(nèi)容