Android学习整理- 8 -MediaPlayer 放歌


Android学习整理 - 系列


目录:

Mediaplayer
常用的API
生命周期
播放文件来源
最基本实现放歌(file文件形式)
网络流放歌
内容提供器(content provider)播放
使用锁
播放错误,监听处理
注意事项
在服务里实现后台放歌


Mediaplayer

在Android中播放音频文件一般是使用MediaPlayer类实现的,它具有很全面的控制方法,支持多种格式。

常用的API

方法名 功能描述
setDataSource() 设置要播放的音频文件的位置
prepare() 在开始播放之前调用这个方法完成准备工作
start() 开始播放或者继续播放音频
pause() 暂停播放音频
reset() 将MediaPlayer对象重置到开始创建的状态,prepare()之前
seekTo() 从指定的位置开始播放音频
stop() 停止播放音频。从生命周期(在下面)看出调用这个方法后,MediaPlayer无法再start()播放音频
release() 释放掉与MediaPlayer有关的资源,防止占用过多内存
isPlaying() 当前是否在播放音频
getDuration() 载入音频文件的长度
setAudioStreamType 设置资源为音频流

生命周期

MediaPlayer生命周期

播放文件来源

  • file
  • 网络流
  • 内容提供器

最基本实现放歌(file文件形式)

基本步骤:

  1. 获取音频文件路径
  2. 用setDataSource()方法设置音频文件的路径
  3. 调用prepare()方法使MediaPlayer进入到准备状态
  4. 调用start()方法开始播放
  5. 调用pause()可以停止播放
  6. 调用reset()方法重置播放
  7. 调用stop()方法停止播放音频,从生命周期看出无法直接再start()
  8. 释放内存

实践

首先下载一个mp3文件mu.mp3放到sd卡根目录(建议用真机)

这里要查看内存卡资源,先声明权限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
...其他的不用写了,允许一个权限,该权限所在的权限组就全部同意了

另外还有Android 6.0以上的运行时权限在代码里写(程序运行时申请权限)

接着布局文件,书写三个控制按钮,play,pause,reset

<LinearLayout
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:orientation="horizontal">

        <Button
            android:id="@+id/play"
            android:layout_width="0dp"
            android:layout_height="40dp"
            android:layout_weight="1"
            android:text="play"/>

        <Button
            android:id="@+id/pause"
            android:layout_width="0dp"
            android:layout_height="40dp"
            android:layout_weight="1"
            android:text="pause"/>

        <Button
            android:id="@+id/reset"
            android:layout_width="0dp"
            android:layout_height="40dp"
            android:layout_weight="1"
            android:text="reset"/>
    </LinearLayout>

接着Activity中,

private MediaPlayer mMediaPlayer = new MediaPlayer();

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //初始化button
        iniView();
        //判断权限够不够,不够就给
        if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(MainActivity.this, new String[]{
                    Manifest.permission.WRITE_EXTERNAL_STORAGE
            }, 1);
        } else {
            //够了就设置路径等,准备播放
            iniMediaPlayerFile();
        }
    }

    //获取到权限回调方法
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        switch (requestCode) {
            case 1:
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    iniMediaPlayerFile();
                } else {
                    Toast.makeText(this, "权限不够获取不到音乐,程序将退出", Toast.LENGTH_SHORT).show();
                    finish();
                }
                break;
            default:
                break;
        }
    }

  /*
  * 添加file文件到MediaPlayer对象并且准备播放音频
  **/
  private void iniMediaPlayerFile() {
        //获取文件路径
        File file = new File(Environment.getExternalStorageDirectory(), "mu.mp3");
        try {
            //此处的两个方法需要捕获IO异常
            //设置音频文件到MediaPlayer对象中
            mMediaPlayer.setDataSource(file.getPath());
            //让MediaPlayer对象准备
            mMediaPlayer.prepare();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

  @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.play:
                if (!mMediaPlayer.isPlaying()) {
                   //如果还没开始播放,就开始
                    mMediaPlayer.start();
                }
                break;
            case R.id.pause:
                if (mMediaPlayer.isPlaying()) {
                    //如果在播放,就暂停
                    mMediaPlayer.pause();
                }

                break;
            case R.id.reset:
                if (mMediaPlayer.isPlaying()) {
                    //如果在播放,就恢复到初始状态
                    mMediaPlayer.reset();
                    //重新添加file文件到MediaPlayer对象并且准备播放音频
                    iniMediaPlayerFile();
                }
                break;

    @Override
        protected void onDestroy() {
            super.onDestroy();
            if (mMediaPlayer != null) {
                //停止播放音频
                mMediaPlayer.stop();
                //释放内存
                mMediaPlayer.release();
            }
        
        }

保证sd卡根目录下有音频文件mu.mp3,启动程序,这个时候就写好了播放器了

音频看不见,截图不放了

网络流放歌

其实是下载完了再放,和酷狗那些播放器不一样

String url = "http://........"; // 你的音频URL
MediaPlayer mediaPlayer = new MediaPlayer();
mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
mediaPlayer.setDataSource(url);
mediaPlayer.prepare(); // 可能会很久,为了缓冲
mediaPlayer.start();

Activity里初始化一个新的MediaPlayer对象mediaPlayer2

private MediaPlayer mediaPlayer2 = new MediaPlayer();

布局文件加入3个控制按钮

<TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="网络流形式"/>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:orientation="horizontal">

        <Button
            android:id="@+id/play2"
            android:layout_width="0dp"
            android:layout_height="40dp"
            android:layout_weight="1"
            android:text="play"/>

        <Button
            android:id="@+id/pause2"
            android:layout_width="0dp"
            android:layout_height="40dp"
            android:layout_weight="1"
            android:text="pause"/>

        <Button
            android:id="@+id/reset2"
            android:layout_width="0dp"
            android:layout_height="40dp"
            android:layout_weight="1"
            android:text="reset"/>
    </LinearLayout>

添加方法iniMediaPlayerInternet()

//初始化网络流播放
 private void iniMediaPlayerInternet() {

        String url = "http://win.web.rc03.sycdn.kuwo.cn/9c1f523c86e747c0d7219559befbc493/58ca8370/resource/a2/23/27/1060703964.aac";
        //设置资源为音频流
        mediaPlayer2.setAudioStreamType(AudioManager.STREAM_MUSIC);
        try {
            mediaPlayer2.setDataSource(url);
            mediaPlayer2.prepare();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

在判断权限的加入

          .....
ActivityCompat.requestPermissions(MainActivity.this, new String[]{
                    Manifest.permission.WRITE_EXTERNAL_STORAGE
            }, 1);
        }else{
             iniMediaPlayerFile();
            //加到这里
            iniMediaPlayerInternet();
        .....
}```

按钮事件
       case R.id.play2:
            if (!mediaPlayer2.isPlaying()) {
                mediaPlayer2.start();
            }
            break;
        case R.id.pause2:
            if (mediaPlayer2.isPlaying()) {
                mediaPlayer2.pause();
            }
            break;
        case R.id.reset2:
            if (mediaPlayer2.isPlaying()) {
                mediaPlayer2.reset();
                iniMediaPlayerInternet();
            }
            break;

最后同file播放形式要释放内存

记得声明网络权限,

<uses-permission android:name="android.permission.INTERNET"/>


编译安装,听到歌。。。

#内容提供器(content provider)播放

**步骤**

首先获取歌的列表信息

ContentResolver contentResolver = getContentResolver();
Uri uri = android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
Cursor cursor = contentResolver.query(uri, null, null, null, null);
if (cursor == null) {
// 查询失败,则处理错误
} else if (!cursor.moveToFirst()) {
//moveToFirst()
//This method will return false if the cursor is empty.
//设备里没歌
} else {
int titleColumn = cursor.getColumnIndex(android.provider.MediaStore.Audio.Media.TITLE);
int idColumn = cursor.getColumnIndex(android.provider.MediaStore.Audio.Media._ID);
do {
long thisId = cursor.getLong(idColumn);//这个id就是播放的id
String thisTitle = cursor.getString(titleColumn);
// ...process entry...
} while (cursor.moveToNext());
}
cursor.close();


> ###注意
这里要把cursor关掉

接着

//这个id就是播放的id,对应扫描到音乐id,可以在下面控制
long id = /* retrieve it from somewhere */;
Uri contentUri = ContentUris.withAppendedId(
android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, id);

mMediaPlayer = new MediaPlayer();
mMediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
mMediaPlayer.setDataSource(getApplicationContext(), contentUri);

// ...prepare and start...


具体例子

同上添加


···
private long[] idBag;//申请一个存歌曲id的long数组
···

private void iniMediaPlayercontentUri() {

    ContentResolver contentResolver = getContentResolver();
    Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
    Cursor cursor = contentResolver.query(uri, null, null, null, null);

    if (cursor == null) {
        Toast.makeText(this, "query failed,这里可以处理错误", Toast.LENGTH_SHORT).show();
    } else if (!cursor.moveToFirst()) {
        Toast.makeText(this, "没有歌在设备上", Toast.LENGTH_SHORT).show();
    } else {
        Log.d(TAG, "有歌在设备");
        Toast.makeText(this, "有歌在设备", Toast.LENGTH_SHORT).show();
        int titleColumn = cursor.getColumnIndex(MediaStore.Audio.Media.TITLE);
        int idColumn = cursor.getColumnIndex(MediaStore.Audio.Media._ID);
        //歌曲数量
        int totalMusic = cursor.getCount();
        idBag = new long[totalMusic];
        do {
            long thisId = cursor.getLong(idColumn);
            String thisTitle = cursor.getString(titleColumn);
            //这里还可以写更多过程获取更多媒体的详细数据
            idBag[i] = thisId;
            i++;
            Log.d(TAG,  "当前歌曲id:"+thisId);
            Log.d(TAG,  "当前歌曲title:"+thisTitle);
        } while (cursor.moveToNext());
    }
 
    //具体放歌
    long id = idBag[0];
    Uri contentUri = ContentUris.withAppendedId(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, id);
    mediaPlayer3.setAudioStreamType(AudioManager.STREAM_MUSIC);
    try {
        mediaPlayer3.setDataSource(this, contentUri);
        mediaPlayer3.prepare();
    } catch (IOException e) {
        e.printStackTrace();
    }

}

####贴一下全部代码


MainActivity

public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private static final String TAG = "MainActivity";
private Button palyButton;
private Button pauseButton;
private Button resetButton;
private Button palyButton2;
private Button pauseButton2;
private Button resetButton2;
private Button palyButton3;
private Button pauseButton3;
private Button resetButton3;

private long[] idBag;
private int i = 0;

private MediaPlayer mMediaPlayer = new MediaPlayer();
private MediaPlayer mediaPlayer2 = new MediaPlayer();
private MediaPlayer mediaPlayer3 = new MediaPlayer();

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    iniView();
    //判断权限够不够,不够就给
    if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
        ActivityCompat.requestPermissions(MainActivity.this, new String[]{
                Manifest.permission.WRITE_EXTERNAL_STORAGE
        }, 1);
    } else {
        iniMediaPlayerFile();
        iniMediaPlayerInternet();
        iniMediaPlayercontentUri();
    }

}

private void iniMediaPlayercontentUri() {

    ContentResolver contentResolver = getContentResolver();
    Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
    Cursor cursor = contentResolver.query(uri, null, null, null, null);

    if (cursor == null) {
        Toast.makeText(this, "query failed,这里可以处理错误", Toast.LENGTH_SHORT).show();
    } else if (!cursor.moveToFirst()) {
        Toast.makeText(this, "没有歌在设备上", Toast.LENGTH_SHORT).show();
    } else {
        Log.d(TAG, "有歌在设备");
        Toast.makeText(this, "有歌在设备", Toast.LENGTH_SHORT).show();
        int titleColumn = cursor.getColumnIndex(MediaStore.Audio.Media.TITLE);
        int idColumn = cursor.getColumnIndex(MediaStore.Audio.Media._ID);
        //歌曲数量
        int totalMusic = cursor.getCount();
        idBag = new long[totalMusic];
        do {
            long thisId = cursor.getLong(idColumn);
            String thisTitle = cursor.getString(titleColumn);
            //这里还可以写更多过程获取更多媒体的详细数据
            idBag[i] = thisId;
            i++;
            Log.d(TAG,  "当前歌曲id:"+thisId);
            Log.d(TAG,  "当前歌曲title:"+thisTitle);
        } while (cursor.moveToNext());
    }

    long id = idBag[0];
    Uri contentUri = ContentUris.withAppendedId(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, id);
    mediaPlayer3.setAudioStreamType(AudioManager.STREAM_MUSIC);
    try {
        mediaPlayer3.setDataSource(this, contentUri);
        mediaPlayer3.prepare();
    } catch (IOException e) {
        e.printStackTrace();
    }

}


private void iniMediaPlayerInternet() {


    String url = "http://win.web.rc03.sycdn.kuwo.cn/9c1f523c86e747c0d7219559befbc493/58ca8370/resource/a2/23/27/1060703964.aac";
    //设置资源为音频流
    mediaPlayer2.setAudioStreamType(AudioManager.STREAM_MUSIC);
    try {
        mediaPlayer2.setDataSource(url);
        mediaPlayer2.prepare();
    } catch (IOException e) {
        e.printStackTrace();
    }
}


private void iniMediaPlayerFile() {
    //获取文件路径
    File file = new File(Environment.getExternalStorageDirectory(), "mu.mp3");
    try {
        //此处的两个方法需要捕获IO异常
        //设置音频文件到MediaPlayer对象中
        mMediaPlayer.setDataSource(file.getPath());
        //让MediaPlayer对象准备
        mMediaPlayer.prepare();
    } catch (IOException e) {
        e.printStackTrace();
    }
}

//获取到权限回调方法
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    switch (requestCode) {
        case 1:
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                iniMediaPlayerFile();
            } else {
                Toast.makeText(this, "权限不够获取不到音乐,程序将退出", Toast.LENGTH_SHORT).show();
                finish();
            }
            break;
        default:
            break;
    }
}



@Override
public void onClick(View v) {
    switch (v.getId()) {
        case R.id.play:
            if (!mMediaPlayer.isPlaying()) {
                mMediaPlayer.start();
            }
            break;
        case R.id.pause:
            if (mMediaPlayer.isPlaying()) {
                mMediaPlayer.pause();
            }

            break;
        case R.id.reset:
            if (mMediaPlayer.isPlaying()) {
                mMediaPlayer.reset();
                iniMediaPlayerFile();
            }
            break;
        case R.id.play2:
            if (!mediaPlayer2.isPlaying()) {
                mediaPlayer2.start();
            }
            break;
        case R.id.pause2:
            if (mediaPlayer2.isPlaying()) {
                mediaPlayer2.pause();
            }

            break;
        case R.id.reset2:
            if (mediaPlayer2.isPlaying()) {
                mediaPlayer2.reset();
                iniMediaPlayerInternet();
            }
            break;
        case R.id.play3:
            if (!mediaPlayer3.isPlaying()) {
                mediaPlayer3.start();
                Log.d(TAG, "播放");
            }else {
                Log.d(TAG, "播放shibai2");
            }
            break;
        case R.id.pause3:
            if (mediaPlayer3.isPlaying()) {
                mediaPlayer3.pause();
            }

            break;
        case R.id.reset3:
            if (mediaPlayer3.isPlaying()) {
                mediaPlayer3.reset();
                iniMediaPlayercontentUri();
            }
            break;
        default:
            break;
    }
}

@Override
protected void onDestroy() {
    super.onDestroy();
    if (mMediaPlayer != null) {
        mMediaPlayer.stop();
        mMediaPlayer.release();
    }

}


private void iniView() {
    palyButton = (Button) findViewById(R.id.play);
    pauseButton = (Button) findViewById(R.id.pause);
    resetButton = (Button) findViewById(R.id.reset);
    palyButton.setOnClickListener(this);
    pauseButton.setOnClickListener(this);
    resetButton.setOnClickListener(this);

    palyButton2 = (Button) findViewById(R.id.play2);
    pauseButton2 = (Button) findViewById(R.id.pause2);
    resetButton2 = (Button) findViewById(R.id.reset2);
    palyButton2.setOnClickListener(this);
    pauseButton2.setOnClickListener(this);
    resetButton2.setOnClickListener(this);

    palyButton3 = (Button) findViewById(R.id.play3);
    pauseButton3 = (Button) findViewById(R.id.pause3);
    resetButton3 = (Button) findViewById(R.id.reset3);
    palyButton3.setOnClickListener(this);
    pauseButton3.setOnClickListener(this);
    resetButton3.setOnClickListener(this);
}

}

布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.minminaya.mediaplayer.MainActivity">

<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="file文件形式"/>

<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="50dp"
    android:orientation="horizontal">

    <Button
        android:id="@+id/play"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="play"/>

    <Button
        android:id="@+id/pause"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="pause"/>

    <Button
        android:id="@+id/reset"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="reset"/>
</LinearLayout>

<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="网络流形式"/>

<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="50dp"
    android:orientation="horizontal">

    <Button
        android:id="@+id/play2"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="play"/>

    <Button
        android:id="@+id/pause2"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="pause"/>

    <Button
        android:id="@+id/reset2"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="reset"/>
</LinearLayout>

<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="内容提供器(本机设备音乐)形式"/>

<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="50dp"
    android:orientation="horizontal">

    <Button
        android:id="@+id/play3"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="play"/>

    <Button
        android:id="@+id/pause3"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="pause"/>

    <Button
        android:id="@+id/reset3"
        android:layout_width="0dp"
        android:layout_height="40dp"
        android:layout_weight="1"
        android:text="reset"/>
</LinearLayout>

</LinearLayout>



![2017-03-16 22_17_16-MediaPlayer - [E__workspace_as20160914_MediaPlayer] - [app] - ..._app_src_main_r.png](http://upload-images.jianshu.io/upload_images/3515789-9e2280b5885e9424.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
#使用锁

CPU

> To ensure that the CPU continues running while your [MediaPlayer](https://developer.android.google.cn/reference/android/media/MediaPlayer.html)
 is playing, call the [setWakeMode()](https://developer.android.google.cn/reference/android/media/MediaPlayer.html#setWakeMode(android.content.Context, int))
 method when initializing your [MediaPlayer](https://developer.android.google.cn/reference/android/media/MediaPlayer.html)
. Once you do, the [MediaPlayer](https://developer.android.google.cn/reference/android/media/MediaPlayer.html)
 holds the specified lock while playing and** releases **the lock when paused or stopped:

mMediaPlayer = new MediaPlayer();
// ... other initialization here ...
mMediaPlayer.setWakeMode(getApplicationContext(), PowerManager.PARTIAL_WAKE_LOCK);


WIFI

> However, the wake lock acquired in this example guarantees only that the CPU remains awake. If you are streaming media over the network and you are using Wi-Fi, you probably want to hold a [WifiLock](https://developer.android.google.cn/reference/android/net/wifi/WifiManager.WifiLock.html)
 as well, which you must acquire and release manually. So, when you start preparing the [MediaPlayer](https://developer.android.google.cn/reference/android/media/MediaPlayer.html)
 with the remote URL, you should create and acquire the Wi-Fi lock. 

WifiLock wifiLock = ((WifiManager) getSystemService(Context.WIFI_SERVICE))
.createWifiLock(WifiManager.WIFI_MODE_FULL, "mylock");

wifiLock.acquire();


停止播放释放锁

wifiLock.release();


#播放错误,监听处理

在某种情况下播放时可能会报错,为了处理这种情况可以设置MediaPlayer.OnErrorListener监听,来恢复之前的状态,记得先释放它

mMediaPlayer.setOnErrorListener(new MediaPlayer.OnErrorListener() {
@Override
public boolean onError(MediaPlayer mp, int what, int extra) {
return false;
}
});


---

#注意事项
哈哈哈,我又忘记了

- 注意释放内存,不然会GG
- 使用contentProvider加载外部音频放在子线程里,不然ANR

---

#在服务里实现后台放歌

[Service 与 MediaPlayer学习后音乐播放器的实现](http://www.jianshu.com/p/836a3d7a8247)

---

参考自
[Google-Mediaplayer](https://developer.android.google.cn/guide/topics/media/mediaplayer.html)

[Mediaplayer参考](https://developer.android.google.cn/reference/android/media/MediaPlayer.html)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 161,513评论 4 369
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,312评论 1 305
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 111,124评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,529评论 0 217
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,937评论 3 295
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,913评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,084评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,816评论 0 205
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,593评论 1 249
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,788评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,267评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,601评论 3 261
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,265评论 3 241
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,158评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,953评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,066评论 2 285
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,852评论 2 277

推荐阅读更多精彩内容