SpringBoot + FFmpeg實現(xiàn)一個簡單的M3U8切片轉(zhuǎn)碼系統(tǒng)

使用大名鼎鼎的ffmpeg褒颈,把視頻文件切片成m3u8柒巫,并且通過springboot,可以實現(xiàn)在線的點播谷丸。

想法

客戶端上傳視頻到服務(wù)器堡掏,服務(wù)器對視頻進(jìn)行切片后,返回m3u8刨疼,封面等訪問路徑布疼。可以在線的播放币狠。
服務(wù)器可以對視頻做一些簡單的處理游两,例如裁剪,封面的截取時間漩绵。

視頻轉(zhuǎn)碼文件夾的定義

喜羊羊與灰太狼  // 文件夾名稱就是視頻標(biāo)題
  |-index.m3u8  // 主m3u8文件贱案,里面可以配置多個碼率的播放地址
  |-poster.jpg  // 截取的封面圖片
  |-ts      // 切片目錄
    |-index.m3u8  // 切片播放索引
    |-key   // 播放需要解密的AES KEY

實現(xiàn)

需要先在本機安裝FFmpeg,并且添加到PATH環(huán)境變量止吐,如果不會先通過搜索引擎找找資料

工程

image

pom

<project xmlns="http://maven.apache.org/POM/4.0.0"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.demo</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>


    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.4.5</version>
        <relativePath /> <!-- lookup parent from repository -->
    </parent>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.junit.vintage</groupId>
            <artifactId>junit-vintage-engine</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
            <exclusions>
                <exclusion>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-starter-tomcat</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-undertow</artifactId>
        </dependency>
        <dependency>
            <groupId>commons-codec</groupId>
            <artifactId>commons-codec</artifactId>
        </dependency>
        <dependency>
            <groupId>com.google.code.gson</groupId>
            <artifactId>gson</artifactId>
        </dependency>

    </dependencies>

    <build>
        <finalName>${project.artifactId}</finalName>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <executable>true</executable>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

配置文件



server:
  port: 80


app:
  # 存儲轉(zhuǎn)碼視頻的文件夾地址
  video-folder: "C:\\Users\\Administrator\\Desktop\\tmp"

spring:
  servlet:
    multipart:
      enabled: true
      # 不限制文件大小
      max-file-size: -1
      # 不限制請求體大小
      max-request-size: -1
      # 臨時IO目錄
      location: "${java.io.tmpdir}"
      # 不延遲解析
      resolve-lazily: false
      # 超過1Mb宝踪,就IO到臨時目錄
      file-size-threshold: 1MB
  web:
    resources:
      static-locations:
        - "classpath:/static/"
        - "file:${app.video-folder}" # 把視頻文件夾目錄,添加到靜態(tài)資源目錄列表

TranscodeConfig碍扔,用于控制轉(zhuǎn)碼的一些參數(shù)

package com.demo.ffmpeg;

public class TranscodeConfig {
    private String poster;              // 截取封面的時間          HH:mm:ss.[SSS]
    private String tsSeconds;           // ts分片大小瘩燥,單位是秒
    private String cutStart;            // 視頻裁剪,開始時間        HH:mm:ss.[SSS]
    private String cutEnd;              // 視頻裁剪不同,結(jié)束時間        HH:mm:ss.[SSS]
    public String getPoster() {
        return poster;
    }

    public void setPoster(String poster) {
        this.poster = poster;
    }

    public String getTsSeconds() {
        return tsSeconds;
    }

    public void setTsSeconds(String tsSeconds) {
        this.tsSeconds = tsSeconds;
    }

    public String getCutStart() {
        return cutStart;
    }

    public void setCutStart(String cutStart) {
        this.cutStart = cutStart;
    }

    public String getCutEnd() {
        return cutEnd;
    }

    public void setCutEnd(String cutEnd) {
        this.cutEnd = cutEnd;
    }

    @Override
    public String toString() {
        return "TranscodeConfig [poster=" + poster + ", tsSeconds=" + tsSeconds + ", cutStart=" + cutStart + ", cutEnd="
                + cutEnd + "]";
    }
}

MediaInfo厉膀,封裝視頻的一些基礎(chǔ)信息

package com.demo.ffmpeg;

import java.util.List;

import com.google.gson.annotations.SerializedName;

public class MediaInfo {
    public static class Format {
        @SerializedName("bit_rate")
        private String bitRate;
        public String getBitRate() {
            return bitRate;
        }
        public void setBitRate(String bitRate) {
            this.bitRate = bitRate;
        }
    }

    public static class Stream {
        @SerializedName("index")
        private int index;

        @SerializedName("codec_name")
        private String codecName;

        @SerializedName("codec_long_name")
        private String codecLongame;

        @SerializedName("profile")
        private String profile;
    }
    
    // ----------------------------------

    @SerializedName("streams")
    private List<Stream> streams;

    @SerializedName("format")
    private Format format;

    public List<Stream> getStreams() {
        return streams;
    }

    public void setStreams(List<Stream> streams) {
        this.streams = streams;
    }

    public Format getFormat() {
        return format;
    }

    public void setFormat(Format format) {
        this.format = format;
    }
}

FFmpegUtils,工具類封裝FFmpeg的一些操作

package com.demo.ffmpeg;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.List;

import javax.crypto.KeyGenerator;

import org.apache.commons.codec.binary.Hex;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.StringUtils;

import com.google.gson.Gson;


public class FFmpegUtils {
    
    private static final Logger LOGGER = LoggerFactory.getLogger(FFmpegUtils.class);
    
    
    // 跨平臺換行符
    private static final String LINE_SEPARATOR = System.getProperty("line.separator");
    
    /**
     * 生成隨機16個字節(jié)的AESKEY
     * @return
     */
    private static byte[] genAesKey ()  {
        try {
            KeyGenerator keyGenerator = KeyGenerator.getInstance("AES");
            keyGenerator.init(128);
            return keyGenerator.generateKey().getEncoded();
        } catch (NoSuchAlgorithmException e) {
            return null;
        }
    }
    
    /**
     * 在指定的目錄下生成key_info, key文件二拐,返回key_info文件
     * @param folder
     * @throws IOException 
     */
    private static Path genKeyInfo(String folder) throws IOException {
        // AES 密鑰
        byte[] aesKey = genAesKey();
        // AES 向量
        String iv = Hex.encodeHexString(genAesKey());
        
        // key 文件寫入
        Path keyFile = Paths.get(folder, "key");
        Files.write(keyFile, aesKey, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);

        // key_info 文件寫入
        StringBuilder stringBuilder = new StringBuilder();
        stringBuilder.append("key").append(LINE_SEPARATOR);                 // m3u8加載key文件網(wǎng)絡(luò)路徑
        stringBuilder.append(keyFile.toString()).append(LINE_SEPARATOR);    // FFmeg加載key_info文件路徑
        stringBuilder.append(iv);                                           // ASE 向量
        
        Path keyInfo = Paths.get(folder, "key_info");
        
        Files.write(keyInfo, stringBuilder.toString().getBytes(), StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);
        
        return keyInfo;
    }
    
    /**
     * 指定的目錄下生成 master index.m3u8 文件
     * @param fileName          master m3u8文件地址
     * @param indexPath         訪問子index.m3u8的路徑
     * @param bandWidth         流碼率
     * @throws IOException
     */
    private static void genIndex(String file, String indexPath, String bandWidth) throws IOException {
        StringBuilder stringBuilder = new StringBuilder();
        stringBuilder.append("#EXTM3U").append(LINE_SEPARATOR);
        stringBuilder.append("#EXT-X-STREAM-INF:BANDWIDTH=" + bandWidth).append(LINE_SEPARATOR);  // 碼率
        stringBuilder.append(indexPath);
        Files.write(Paths.get(file), stringBuilder.toString().getBytes(StandardCharsets.UTF_8), StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);
    }
    
    /**
     * 轉(zhuǎn)碼視頻為m3u8
     * @param source                源視頻
     * @param destFolder            目標(biāo)文件夾
     * @param config                配置信息
     * @throws IOException 
     * @throws InterruptedException 
     */
    public static void transcodeToM3u8(String source, String destFolder, TranscodeConfig config) throws IOException, InterruptedException {
        
        // 判斷源視頻是否存在
        if (!Files.exists(Paths.get(source))) {
            throw new IllegalArgumentException("文件不存在:" + source);
        }
        
        // 創(chuàng)建工作目錄
        Path workDir = Paths.get(destFolder, "ts");
        Files.createDirectories(workDir);
        
        // 在工作目錄生成KeyInfo文件
        Path keyInfo = genKeyInfo(workDir.toString());
        
        // 構(gòu)建命令
        List<String> commands = new ArrayList<>();
        commands.add("ffmpeg");         
        commands.add("-i")                      ;commands.add(source);                  // 源文件
        commands.add("-c:v")                    ;commands.add("libx264");               // 視頻編碼為H264
        commands.add("-c:a")                    ;commands.add("copy");                  // 音頻直接copy
        commands.add("-hls_key_info_file")      ;commands.add(keyInfo.toString());      // 指定密鑰文件路徑
        commands.add("-hls_time")               ;commands.add(config.getTsSeconds());   // ts切片大小
        commands.add("-hls_playlist_type")      ;commands.add("vod");                   // 點播模式
        commands.add("-hls_segment_filename")   ;commands.add("%06d.ts");               // ts切片文件名稱
        
        if (StringUtils.hasText(config.getCutStart())) {
            commands.add("-ss")                 ;commands.add(config.getCutStart());    // 開始時間
        }
        if (StringUtils.hasText(config.getCutEnd())) {
            commands.add("-to")                 ;commands.add(config.getCutEnd());      // 結(jié)束時間
        }
        commands.add("index.m3u8");                                                     // 生成m3u8文件
        
        // 構(gòu)建進(jìn)程
        Process process = new ProcessBuilder()
            .command(commands)
            .directory(workDir.toFile())
            .start()
            ;
        
        // 讀取進(jìn)程標(biāo)準(zhǔn)輸出
        new Thread(() -> {
            try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {
                String line = null;
                while ((line = bufferedReader.readLine()) != null) {
                    LOGGER.info(line);
                }
            } catch (IOException e) {
            }
        }).start();
        
        // 讀取進(jìn)程異常輸出
        new Thread(() -> {
            try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getErrorStream()))) {
                String line = null;
                while ((line = bufferedReader.readLine()) != null) {
                    LOGGER.info(line);
                }
            } catch (IOException e) {
            }
        }).start();
        
        
        // 阻塞直到任務(wù)結(jié)束
        if (process.waitFor() != 0) {
            throw new RuntimeException("視頻切片異常");
        }
        
        // 切出封面
        if (!screenShots(source, String.join(File.separator, destFolder, "poster.jpg"), config.getPoster())) {
            throw new RuntimeException("封面截取異常");
        }
        
        // 獲取視頻信息
        MediaInfo mediaInfo = getMediaInfo(source);
        if (mediaInfo == null) {
            throw new RuntimeException("獲取媒體信息異常");
        }
        
        // 生成index.m3u8文件
        genIndex(String.join(File.separator, destFolder, "index.m3u8"), "ts/index.m3u8", mediaInfo.getFormat().getBitRate());
        
        // 刪除keyInfo文件
        Files.delete(keyInfo);
    }
    
    /**
     * 獲取視頻文件的媒體信息
     * @param source
     * @return
     * @throws IOException
     * @throws InterruptedException
     */
    public static MediaInfo getMediaInfo(String source) throws IOException, InterruptedException {
        List<String> commands = new ArrayList<>();
        commands.add("ffprobe");    
        commands.add("-i")              ;commands.add(source);
        commands.add("-show_format");
        commands.add("-show_streams");
        commands.add("-print_format")   ;commands.add("json");
        
        Process process = new ProcessBuilder(commands)
                .start();
         
        MediaInfo mediaInfo = null;
        
        try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {
            mediaInfo = new Gson().fromJson(bufferedReader, MediaInfo.class);
        } catch (IOException e) {
            e.printStackTrace();
        }
        
        if (process.waitFor() != 0) {
            return null;
        }
        
        return mediaInfo;
    }
    
    /**
     * 截取視頻的指定時間幀服鹅,生成圖片文件
     * @param source        源文件
     * @param file          圖片文件
     * @param time          截圖時間 HH:mm:ss.[SSS]     
     * @throws IOException 
     * @throws InterruptedException 
     */
    public static boolean screenShots(String source, String file, String time) throws IOException, InterruptedException {
        
        List<String> commands = new ArrayList<>();
        commands.add("ffmpeg"); 
        commands.add("-i")              ;commands.add(source);
        commands.add("-ss")             ;commands.add(time);
        commands.add("-y");
        commands.add("-q:v")            ;commands.add("1");
        commands.add("-frames:v")       ;commands.add("1");
        commands.add("-f");             ;commands.add("image2");
        commands.add(file);
        
        Process process = new ProcessBuilder(commands)
                    .start();
        
        // 讀取進(jìn)程標(biāo)準(zhǔn)輸出
        new Thread(() -> {
            try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {
                String line = null;
                while ((line = bufferedReader.readLine()) != null) {
                    LOGGER.info(line);
                }
            } catch (IOException e) {
            }
        }).start();
        
        // 讀取進(jìn)程異常輸出
        new Thread(() -> {
            try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getErrorStream()))) {
                String line = null;
                while ((line = bufferedReader.readLine()) != null) {
                    LOGGER.error(line);
                }
            } catch (IOException e) {
            }
        }).start();
        
        return process.waitFor() == 0;
    }
}


UploadController,執(zhí)行轉(zhuǎn)碼操作

package com.demo.web.controller;

import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.util.HashMap;
import java.util.Map;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestPart;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;

import com.demo.ffmpeg.FFmpegUtils;
import com.demo.ffmpeg.TranscodeConfig;

@RestController
@RequestMapping("/upload")
public class UploadController {
    
    private static final Logger LOGGER = LoggerFactory.getLogger(UploadController.class);
    
    @Value("${app.video-folder}")
    private String videoFolder;

    private Path tempDir = Paths.get(System.getProperty("java.io.tmpdir"));
    
    /**
     * 上傳視頻進(jìn)行切片處理百新,返回訪問路徑
     * @param video
     * @param transcodeConfig
     * @return
     * @throws IOException 
     */
    @PostMapping
    public Object upload (@RequestPart(name = "file", required = true) MultipartFile video,
                        @RequestPart(name = "config", required = true) TranscodeConfig transcodeConfig) throws IOException {
        
        LOGGER.info("文件信息:title={}, size={}", video.getOriginalFilename(), video.getSize());
        LOGGER.info("轉(zhuǎn)碼配置:{}", transcodeConfig);
        
        // 原始文件名稱企软,也就是視頻的標(biāo)題
        String title = video.getOriginalFilename();
        
        // io到臨時文件
        Path tempFile = tempDir.resolve(title);
        LOGGER.info("io到臨時文件:{}", tempFile.toString());
        
        try {
            
            video.transferTo(tempFile);
            
            // 刪除后綴
            title = title.substring(0, title.lastIndexOf("."));
            
            // 按照日期生成子目錄
            String today = DateTimeFormatter.ofPattern("yyyyMMdd").format(LocalDate.now());
            
            // 嘗試創(chuàng)建視頻目錄
            Path targetFolder = Files.createDirectories(Paths.get(videoFolder, today, title));
            
            LOGGER.info("創(chuàng)建文件夾目錄:{}", targetFolder);
            Files.createDirectories(targetFolder);
            
            // 執(zhí)行轉(zhuǎn)碼操作
            LOGGER.info("開始轉(zhuǎn)碼");
            try {
                FFmpegUtils.transcodeToM3u8(tempFile.toString(), targetFolder.toString(), transcodeConfig);
            } catch (Exception e) {
                LOGGER.error("轉(zhuǎn)碼異常:{}", e.getMessage());
                Map<String, Object> result = new HashMap<>();
                result.put("success", false);
                result.put("message", e.getMessage());
                return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(result);
            }
            
            // 封裝結(jié)果
            Map<String, Object> videoInfo = new HashMap<>();
            videoInfo.put("title", title);
            videoInfo.put("m3u8", String.join("/", "", today, title, "index.m3u8"));
            videoInfo.put("poster", String.join("/", "", today, title, "poster.jpg"));
            
            Map<String, Object> result = new HashMap<>();
            result.put("success", true);
            result.put("data", videoInfo);
            return result;
        } finally {
            // 始終刪除臨時文件
            Files.delete(tempFile);
        }
    }
}

index.html,客戶端

<html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
        <script src="https://cdn.jsdelivr.net/hls.js/latest/hls.min.js"></script>
    </head>
    <body>
        選擇轉(zhuǎn)碼文件: <input name="file" type="file" accept="video/*" onchange="upload(event)">
        <hr/>
        <video id="video"  width="500" height="400" controls="controls"></video>
    </body>
    <script>
    
        const video = document.getElementById('video');
        
        function upload (e){
            let files = e.target.files
            if (!files) {
                return
            }
            
            // TODO 轉(zhuǎn)碼配置這里固定死了
            var transCodeConfig = {
                poster: "00:00:00.001", // 截取第1毫秒作為封面
                tsSeconds: 15,              
                cutStart: "",
                cutEnd: ""
            }
            
            // 執(zhí)行上傳
            let formData = new FormData();
            formData.append("file", files[0])
            formData.append("config", new Blob([JSON.stringify(transCodeConfig)], {type: "application/json; charset=utf-8"}))

            fetch('/upload', {
                method: 'POST',
                body: formData
            })
            .then(resp =>  resp.json())
            .then(message => {
                if (message.success){
                    // 設(shè)置封面
                    video.poster = message.data.poster;
                    
                    // 渲染到播放器
                    var hls = new Hls();
                    hls.loadSource(message.data.m3u8);
                    hls.attachMedia(video);
                } else {
                    alert("轉(zhuǎn)碼異常饭望,詳情查看控制臺");
                    console.log(message.message);
                }
            })
            .catch(err => {
                alert("轉(zhuǎn)碼異常仗哨,詳情查看控制臺");
                throw err
            })
        }
    </script>
</html>

使用

  1. 在配置文件中,配置到本地視頻目錄后啟動
  2. 打開頁面 localhost
  3. 點擊【選擇文件】铅辞,選擇一個視頻文件進(jìn)行上傳厌漂,等待執(zhí)行完畢(沒有做加載動畫)
  4. 后端轉(zhuǎn)碼完成后,會自動把視頻信息加載到播放器巷挥,此時可以手動點擊播放按鈕進(jìn)行播放

可以打開控制臺桩卵,查看上傳進(jìn)度,以及播放時的網(wǎng)絡(luò)加載信息

源碼

https://github.com/KevinBlandy/springboot-ffmpeg-demo


首發(fā):https://springboot.io/t/topic/3669

最后編輯于
?著作權(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
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來涝涤,“玉大人媚狰,你說我怎么就攤上這事±” “怎么了崭孤?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長糊肠。 經(jīng)常有香客問我辨宠,道長,這世上最難降的妖魔是什么货裹? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任嗤形,我火速辦了婚禮,結(jié)果婚禮上弧圆,老公的妹妹穿的比我還像新娘赋兵。我一直安慰自己,他們只是感情好墓阀,可當(dāng)我...
    茶點故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布毡惜。 她就那樣靜靜地躺著,像睡著了一般斯撮。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上扶叉,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天勿锅,我揣著相機與錄音,去河邊找鬼枣氧。 笑死溢十,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的达吞。 我是一名探鬼主播张弛,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了吞鸭?” 一聲冷哼從身側(cè)響起寺董,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎刻剥,沒想到半個月后遮咖,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡造虏,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年御吞,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(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
  • 正文 我出身青樓,卻偏偏與公主長得像夺饲,于是被迫代替她去往敵國和親奸汇。 傳聞我的和親對象是個殘疾皇子施符,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,037評論 2 355

推薦閱讀更多精彩內(nèi)容