07_使用隨機流RandomAccessFile刪除文件指定內(nèi)容

@Author Jacky Wang

轉(zhuǎn)載請注明出處,http://www.reibang.com/p/1c1b35d9be9b

碰到一個需求,在某個服務(wù)的日志文件達到指定大小之后,刪除該日志文件前面多少行,在后面追加新的日志用僧。以下,為此次使用隨機文件流操作文件內(nèi)容所記科阎。

/**   
 * @Title: removeFileLine   
 * @Description: TODO(該方法為從文件開頭刪除前n行)   
 * @param: @param file 文件
 * @param: @param lineNum 刪除的行行數(shù)
 * @param: @throws IOException      
 * @return: void      
 * @throws   
 */ 
public void removeFileLine(File file, int lineNum) throws IOException {
    RandomAccessFile raf = null;
    try {
        raf = new RandomAccessFile(file, "rw");
        // Initial write position.
        // 寫文件的位置標記,從文件開頭開始,后續(xù)讀取文件內(nèi)容從該標記開始
        long writePosition = raf.getFilePointer();
        for (int i = 0; i < lineNum; i++) {
            String line = raf.readLine();
            if (line == null) {
                break;
            }
        }
        // Shift the next lines upwards.
        // 讀文件的位置標記,寫完之后回到該標記繼續(xù)讀該行
        long readPosition = raf.getFilePointer();

        // 利用兩個標記,
        byte[] buff = new byte[1024];
        int n;
        while (-1 != (n = raf.read(buff))) {
            raf.seek(writePosition);
            raf.write(buff, 0, n);
            readPosition += n;
            writePosition += n;
            raf.seek(readPosition);
        }
        raf.setLength(writePosition);
    } catch (IOException e) {
        logger.error("readAndRemoveFirstLines error", e);
        throw e;
    } finally {
        try {
            if (raf != null) {
                raf.close();
            }
        } catch (IOException e) {
            logger.error("close RandomAccessFile error", e);
            throw e;
        }
    }
}

/**   
 * @Title: appendContentToFile   
 * @Description: TODO(在文件末尾追加內(nèi)容)   
 * @param: @param file
 * @param: @param content
 * @param: @throws IOException      
 * @return: void      
 * @throws   
 */ 
public static void appendContentToFile(File file, String content) throws IOException {
    RandomAccessFile randomFile = null;
    try {
        // 打開一個隨機訪問文件流,按讀寫方式
        randomFile = new RandomAccessFile(file, "rw");
        // 文件長度,字節(jié)數(shù)
        long fileLength = randomFile.length();
        // 將寫文件指針移到文件尾根盒。
        randomFile.seek(fileLength);
        randomFile.writeBytes(content);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        if (randomFile != null) {
            randomFile.close();
            randomFile = null;
        }
    }
}

/**   
 * @Title: checkFileInSize   
 * @Description: TODO(檢查文件字節(jié)數(shù)是否超出限制)   
 * @param: @param file
 * @param: @param limitSize
 * @param: @return      
 * @return: boolean      
 * @throws   
 */ 
private boolean checkFileInSize(File file, Long limitSize) {
    return file.length() <= limitSize;
}

以上就是上述需求用到的方法了,下面將上面的方法整合完成需求纠拔。

@Component
@SuppressWarnings("restriction")
@PropertySource("classpath:/config/ivg.properties")
public class AppStartupListener implements ApplicationRunner {

    @Value("${log.startup.path}")
    private String logPath;// 日志保存路徑
    @Value("${log.startup.output.statement}")
    private String outputStatement;// 追加日志內(nèi)容模板
    @Value("${log.startup.limit.size}")
    private Long limitSize;// 文件大小限制,eg:50M:50*1024*1024 = 52428800
    @Value("${log.startup.remove.line}")
    private int removeLineNum;// 超過限制之后一次刪除多少行

    private void initLog() {
        logger.info("***程序啟動日志記錄開始***");

        // 檢查文件是否存在
        File file = null;
        try {
            file = new File(logPath);
            if (!file.exists() || !file.isFile()) {
                logger.info("file is not exist,creating " + logPath + " now...");
                file.createNewFile();
            }
            StringBuilder sb = new StringBuilder(outputStatement);
            SimpleDateFormat sdf = new SimpleDateFormat(" yyyy-MM-dd HH:mm:ss");
            String date = sdf.format(new Date());
            String outputLog = sb.append(date).append("\r\n").toString();

            // 檢查文件大小
            while (!checkFileInSize(file, limitSize)) {
                // 先刪除前三行
                removeFileLine(file, removeLineNum);
            }
            // 追加項目啓動log日志
            appendContentToFile(file, outputLog);
        } catch (FileNotFoundException e) {
            logger.error("StartupLog Listener error,{}", e);
        } catch (IOException e) {
            logger.error("StartupLog Listener error,{}", e);
        }
    }
}

至此需求已完成。通過此次需求,學習了RandomAccessFile隨機文件流的一些簡單使用方式,記錄在此蜜氨。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末械筛,一起剝皮案震驚了整個濱河市伞梯,隨后出現(xiàn)的幾起案子骂倘,更是在濱河造成了極大的恐慌,老刑警劉巖种柑,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件厌丑,死亡現(xiàn)場離奇詭異定欧,居然都是意外死亡,警方通過查閱死者的電腦和手機怒竿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進店門砍鸠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人耕驰,你說我怎么就攤上這事爷辱。” “怎么了朦肘?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵饭弓,是天一觀的道長。 經(jīng)常有香客問我媒抠,道長弟断,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任趴生,我火速辦了婚禮阀趴,結(jié)果婚禮上昏翰,老公的妹妹穿的比我還像新娘。我一直安慰自己刘急,他們只是感情好棚菊,可當我...
    茶點故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著叔汁,像睡著了一般统求。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上攻柠,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天球订,我揣著相機與錄音,去河邊找鬼瑰钮。 笑死冒滩,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的浪谴。 我是一名探鬼主播开睡,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼苟耻!你這毒婦竟也來了篇恒?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤凶杖,失蹤者是張志新(化名)和其女友劉穎胁艰,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體智蝠,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡腾么,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了杈湾。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片解虱。...
    茶點故事閱讀 40,096評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖漆撞,靈堂內(nèi)的尸體忽然破棺而出殴泰,到底是詐尸還是另有隱情,我是刑警寧澤浮驳,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布悍汛,位于F島的核電站,受9級特大地震影響至会,放射性物質(zhì)發(fā)生泄漏员凝。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一奋献、第九天 我趴在偏房一處隱蔽的房頂上張望健霹。 院中可真熱鬧,春花似錦瓶蚂、人聲如沸糖埋。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽瞳别。三九已至,卻和暖如春杭攻,著一層夾襖步出監(jiān)牢的瞬間祟敛,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工兆解, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留馆铁,地道東北人。 一個月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓锅睛,卻偏偏與公主長得像埠巨,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子现拒,可洞房花燭夜當晚...
    茶點故事閱讀 45,037評論 2 355