android mvp分層架構(gòu)優(yōu)雅寫代碼

背景

看了好多android技術博客,寫android分層架構(gòu)的博客越來越多吊档,有mvc篙议、mvp、mvvm怠硼、clean等各式各樣的鬼贱,而mvp異常火熱香璃,然而每個人對mvp的定義又是不同这难,寫法自然也是千紫萬紅。

目的

寫一個實用分層清晰的mvp架構(gòu)

主題

mvp無非 model(數(shù)據(jù))葡秒、view(界面)姻乓、presenter(邏輯)。model對應本地持久化或遠程服務端數(shù)據(jù)眯牧,而在筆者看來其實就是對應一個bean對象蹋岩,然而這個bean對象由遠程服務器或本地持久化而得到,因而此層需封裝網(wǎng)絡請求和本地持久化学少;view對應activity剪个、fragment以及它們對應的xml布局文件,這層只負責做ui顯示版确;presenter對應邏輯處理層扣囊,所做的事情很多乎折,包括網(wǎng)絡請求操作、讀取緩存數(shù)據(jù)操作如暖、算法計算等等笆檀。

接下來寫代碼來分析筆者認為優(yōu)雅的mvp分層架構(gòu)忌堂,開始看一下項目分組盒至,如下圖所示:

QQ圖片20160908140950.png

從上圖我們看到module下有四個分組,分別對應:contract士修、model枷遂、presenter、views棋嘲。相信大部分童鞋對contract有點疑惑酒唉,這個分組是干啥用的呢?contract是作為契約沸移,目的是將presenter痪伦、views等接口集中關聯(lián)起來,便于統(tǒng)一管理。

QQ圖片20160930154403.png

打開契約分組雹锣,我們看到四個接口和一個類网沾,分別為IActivityLifeCycle(Activity生命周期接口類)、IBaseActivity(Activity接口基類 界面層的)蕊爵、IBaseFragment(Fragment接口基類 界面層的)辉哥、IBasePresenter(邏輯層基類 邏輯層的)、UserInfoContract(用戶信息契約類攒射,關聯(lián)view層與presenter層接口,方便統(tǒng)一管理)醋旦。

IActivityLifeCycle代碼如下:

/**
 * @className: IActivityLifeCycle
 * @classDescription: 生命周期接口(為了實現(xiàn)Activity UI層生命周期映射到邏輯層)
 * @author: leibing
 * @createTime: 2016/8/11
 */
public interface IActivityLifeCycle {
    void onCreate();
    void onRestart();
    void onStart();
    void onResume();
    void onPause();
    void onStop();
    void onDestroy();
}

IActivityLifeCycle 作為一個activity生命周期接口,為了將activity生命周期映射到對應presenter層会放,便于邏輯層能更好的處理跟activity生命周期有關的事件饲齐。

IBaseActivity代碼如下:

package com.ym.mvpdemo.module.contract;

/**
 * @className: IBaseActivity
 * @classDescription: activity接口基類
 * @author: leibing
 * @createTime: 2016/8/11
 */
public interface IBaseActivity<T> {
    // 設置邏輯
    void setPresenter(T mIActivityPresenter);
    // 設置生命周期
    void setILifeCycle(IActivityLifeCycle mIActivityLifeCycle);
}

IBaseActivity作為一個view層的activity接口基類,主要是設置邏輯層和設置生命周期咧最,將邏輯層與界面層綁定起來捂人,將activity生命周期映射到邏輯層去。

IBaseFragment代碼如下:

package com.ym.mvpdemo.module.contract;

/**
 * @className:IBaseFragment
 * @classDescription:Fragment接口基類
 * @author: leibing
 * @createTime: 2016/8/12
 */
public interface IBaseFragment<T> {
    // 設置邏輯
    void setPresenter(T mIFragmentPresenter);
}

IBaseFragment作為一個view層的fragment接口基類窗市,主要是設置邏輯層先慷,將邏輯層與界面層綁定起來。

IBasePresenter代碼如下:

package com.ym.mvpdemo.module.contract;

/**
 * @className: IBasePresenter
 * @classDescription: 邏輯層基類
 * @author: leibing
 * @createTime: 2016/8/11
 */
public interface IBasePresenter {
    // 邏輯層開始執(zhí)行方法
    void start();
}

IBasePresenter作為一個邏輯層基類咨察,主要做界面層與邏輯層綁定之后论熙,邏輯層初始化工作。

UserInfoContract比較重要摄狱,大家仔細看看脓诡,代碼如下:

package com.ym.mvpdemo.module.contract;

import com.ym.mvpdemo.module.model.UserInfoModel;

/**
 * @className: UserInfoContract
 * @classDescription: 用戶信息契約類
 * @author: leibing
 * @createTime: 2016/8/11
 */
public class UserInfoContract {
    /**
     * 用戶信息activity中用于更新UI的方法集合
     * @interfaceName: IUserInfoActivity
     * @interfaceDescription: View接口
     * @author: leibing
     * @createTime: 2016/08/23
     */
    public interface IUserInfoActivity extends IBaseActivity<IUserInfoActivityPresenter> {
        void showLoading();//展示加載框
        void dismissLoading();//取消加載框展示
        void showUserInfo(UserInfoModel userInfoModel);//將網(wǎng)絡請求得到的用戶信息回調(diào)
        String loadUserId();//假設接口請求需要一個userId
    }

    /**
     * 用戶信息Fragment中用于更新UI的方法集合
     * @interfaceName: IFragment
     * @interfaceDescription: Fragment接口
     * @author: leibing
     * @createTime: 2016/08/23
     */
    public interface IUserInfoFragment extends IBaseFragment<IUserInfoFragmentPresenter> {
        void showData(); // 假定顯示數(shù)據(jù)
    }

    /**
     * 用戶信息activity邏輯層需要使用的方法集合
     * @interfaceName: IUserInfoActivityPresenter
     * @interfaceDescription: 用戶信息Activity邏輯層接口
     * @author: leibing
     * @createTime: 2016/08/23
     */
    public interface IUserInfoActivityPresenter extends IBasePresenter {
        void loadUserInfo();
    }

    /**
     * 用戶信息Fragment邏輯層需要使用的方法集合
     * @interfaceName: IUserInfoFragmentPresenter
     * @interfaceDescription: 用戶信息Fragment邏輯層接口
     * @author: leibing
     * @createTime: 2016/08/23
     */
    public interface IUserInfoFragmentPresenter extends IBasePresenter {
        void loadData();
    }
}

UserInfoContract作為一個契約類无午,將界面層與邏輯層接口進行集中管理,便于提高接口可讀性祝谚。

契約分組分析完了宪迟,然后我們再看model,model結(jié)構(gòu)圖如下所示:

QQ圖片20160930160414.png

至此交惯,我們看到httprequest次泽、data兩個分組,這兩分組分別對應網(wǎng)絡請求封裝和數(shù)據(jù)持久化封裝席爽,這兩塊封裝看個人了意荤,沒有一個絕對的方案,我們還看到一個UserInfoModel只锻,這里我就先放外面了玖像,沒放httprequest和data里面去了,這里具體封裝具體放對應的位置齐饮,網(wǎng)絡請求封裝可以參考基于Retrofit捐寥、OkHttp、Gson封裝通用網(wǎng)絡框架祖驱、持久化數(shù)據(jù)封裝可以參考android基于xml實現(xiàn)的對象緩存方案握恳。

接下來我們分析View層,View分組如下圖所示:

QQ圖片20160930161058.png

view分組就一個activity和一個fragment羹膳,其實這層很簡單睡互,主要做更新ui的工作,代碼結(jié)構(gòu)也比較清晰陵像,筆者在activity里面將對應生命周期映射到了其邏輯層上面去了就珠,這樣省去了在生命周期上面 view層往邏輯層寫方法的麻煩。

activity代碼如下:

package com.ym.mvpdemo.module.views.userinfo;

import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.view.ViewPager;
import android.support.v7.app.AppCompatActivity;
import android.widget.TextView;
import android.widget.Toast;
import com.ym.mvpdemo.R;
import com.ym.mvpdemo.adapter.ViewpagerAdapter;
import com.ym.mvpdemo.module.contract.IActivityLifeCycle;
import com.ym.mvpdemo.module.contract.UserInfoContract;
import com.ym.mvpdemo.module.model.UserInfoModel;
import com.ym.mvpdemo.module.presenter.userinfo.UserInfoActivityPresenter;
import java.util.ArrayList;
import java.util.List;
import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;

/**
 * @className: UserInfoActivity
 * @classDescription: UI層(Activity)
 * @author: leibing
 * @createTime: 2016/8/11
 */
public class UserInfoActivity extends AppCompatActivity implements UserInfoContract.IUserInfoActivity{
    // 切換Tab常量
    public final static int HOME_INDEX = 0;
    public final static int CZH_INDEX = 1;
    public final static int ME_INDEX = 2;
    // Activity邏輯層接口
    private UserInfoContract.IUserInfoActivityPresenter mIActivityPresenter;
    // 生命周期接口
    private IActivityLifeCycle mIActivityLifeCycle;
    // Fragment
    private UserInfoFragment mHomeFragment;
    private UserInfoFragment mCzhFragment;
    private UserInfoFragment mMineFragment;
    // Fragement列表
    private List<Fragment> mFragmentList;
    // 標題列表
    private List<String> mTitleList;

    @BindView(R.id.tv_name) TextView nameTv;
    @BindView(R.id.tv_age) TextView ageTv;
    @BindView(R.id.tv_address) TextView addressTv;
    @BindView(R.id.vpg_main) ViewPager mainPager;
    @BindView(R.id.tv_main_home) TextView mainHomeTv;
    @BindView(R.id.tv_main_czh) TextView mainCzhTv;
    @BindView(R.id.tv_main_me) TextView mainMeTv;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        // 綁定ButterKnife
        ButterKnife.bind(this);
        // 初始化list
        initList();
        // 初始化Fragment
        initFragment();
        // 初始化邏輯
        new UserInfoActivityPresenter(this);
        mIActivityPresenter.start();
        // View映射onCreate生命周期到Presenter
        mIActivityLifeCycle.onCreate();
    }

    /**
     * 初始化列表
     * @author leibing
     * @createTime 2016/8/11
     * @lastModify 2016/8/11
     * @return
     */
    private void initList() {
        mFragmentList = new ArrayList<>();
        mTitleList = new ArrayList<>();
    }

    /**
     * 初始化Fragment
     * @author leibing
     * @createTime 2016/8/11
     * @lastModify 2016/8/11
     * @param
     * @return
     */
    private void initFragment() {
        // 首頁
        mHomeFragment = new UserInfoFragment();
        Bundle bundle = new Bundle();
        bundle.putSerializable(UserInfoFragment.PAGE_INDEX, HOME_INDEX);
        mHomeFragment.setArguments(bundle);
        mFragmentList.add(mHomeFragment);

        // 車智匯
        mCzhFragment = new UserInfoFragment();
        bundle = new Bundle();
        bundle.putSerializable(UserInfoFragment.PAGE_INDEX, CZH_INDEX);
        mCzhFragment.setArguments(bundle);
        mFragmentList.add(mCzhFragment);

        // 我的
        mMineFragment = new UserInfoFragment();
        bundle = new Bundle();
        bundle.putSerializable(UserInfoFragment.PAGE_INDEX, ME_INDEX);
        mMineFragment.setArguments(bundle);
        mFragmentList.add(mMineFragment);

        // ViewPager適配
        ViewpagerAdapter mAdapter = new ViewpagerAdapter(
                getSupportFragmentManager(), mFragmentList, mTitleList);
        mainPager.setAdapter(mAdapter);
        mainPager.setOffscreenPageLimit(3);
        mainPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageSelected(int position) {
            }

            @Override
            public void onPageScrollStateChanged(int state) {
            }

            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
                if (positionOffset == 0.0)
                    switchTab(position);
            }
        });
        mainPager.setCurrentItem(0);
    }

    /**
     * 切換Tab頁
     * @author leibing
     * @createTime 2016/5/6
     * @lastModify 2016/5/6
     * @param index
     * @return
     */
    private void switchTab(int index){
        switch (index){
            case HOME_INDEX:
                mainHomeTv.setTextColor(getResources().getColor(R.color.main_home_text_blue));
                mainCzhTv.setTextColor(getResources().getColor(R.color.main_home_text_gray));
                mainMeTv.setTextColor(getResources().getColor(R.color.main_home_text_gray));
                break;
            case CZH_INDEX:
                mainHomeTv.setTextColor(getResources().getColor(R.color.main_home_text_gray));
                mainCzhTv.setTextColor(getResources().getColor(R.color.main_home_text_blue));
                mainMeTv.setTextColor(getResources().getColor(R.color.main_home_text_gray));
                break;
            case ME_INDEX:
                mainHomeTv.setTextColor(getResources().getColor(R.color.main_home_text_gray));
                mainCzhTv.setTextColor(getResources().getColor(R.color.main_home_text_gray));
                mainMeTv.setTextColor(getResources().getColor(R.color.main_home_text_blue));
                break;
        }
    }

    @Override
    protected void onRestart() {
        // View映射onRestart生命周期到Presenter
        mIActivityLifeCycle.onRestart();
        super.onRestart();
    }

    @Override
    protected void onStart() {
        super.onStart();
        // View映射onStart生命周期到Presenter
        mIActivityLifeCycle.onStart();
    }

    @Override
    protected void onResume() {
        super.onResume();
        // View映射onResume生命周期到Presenter
        mIActivityLifeCycle.onResume();
    }

    @Override
    protected void onPause() {
        // View映射onPause生命周期到Presenter
        mIActivityLifeCycle.onPause();
        super.onPause();
    }

    @Override
    protected void onStop() {
        // View映射onStop生命周期到Presenter
        mIActivityLifeCycle.onStop();
        super.onStop();
    }

    @Override
    protected void onDestroy() {
        // View映射onDestroy生命周期到Presenter
        mIActivityLifeCycle.onDestroy();
        super.onDestroy();
    }

    @Override
    public void showLoading() {
        Toast.makeText(this, "正在加載", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void dismissLoading() {
        Toast.makeText(this, "加載完成", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void showUserInfo(UserInfoModel userInfoModel) {
        if (userInfoModel != null) {
            nameTv.setText(userInfoModel.getName());
            ageTv.setText(String.valueOf(userInfoModel.getAge()));
            addressTv.setText(userInfoModel.getAddress());
        }
    }

    @Override
    public String loadUserId() {
        return "1000";//假設需要查詢的用戶信息的userId是1000
    }

    @Override
    public void setPresenter(UserInfoContract.IUserInfoActivityPresenter mIActivityPresenter) {
        this.mIActivityPresenter = mIActivityPresenter;
    }

    public void setILifeCycle(IActivityLifeCycle mIActivityLifeCycle) {
        this.mIActivityLifeCycle = mIActivityLifeCycle;
    }

    @OnClick(R.id.ly_main_home) void mainOnClick() {
        mainPager.setCurrentItem(HOME_INDEX, false);
    }

    @OnClick(R.id.ly_main_czh) void czhOnClick() {
        mainPager.setCurrentItem(CZH_INDEX, false);
    }

    @OnClick(R.id.ly_main_me) void meOnClick() {
        mainPager.setCurrentItem(ME_INDEX, false);
    }
}

fragment代碼如下:

package com.ym.mvpdemo.module.views.userinfo;

import android.app.Activity;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import android.widget.Toast;
import com.ym.mvpdemo.R;
import com.ym.mvpdemo.module.contract.UserInfoContract;
import com.ym.mvpdemo.module.presenter.userinfo.UserInfoFragmentPresenter;
import butterknife.BindView;
import butterknife.ButterKnife;

/**
 * @className: UserInfoFragment
 * @classDescription: Ui層(IFragment)
 * @author: leibing
 * @createTime: 2016/8/11
 */
public class UserInfoFragment extends Fragment implements UserInfoContract.IUserInfoFragment {
    // 頁面常量
    public final static String PAGE_INDEX = "page_index";
    // 頁面數(shù)字
    private int pageIndex;
    // UI回調(diào)
    private UserInfoContract.IUserInfoFragmentPresenter mIFragmentPresenter;
    // 判斷是否當前Fragment
    private boolean isVisibleToUser = false;

    @BindView(R.id.tv_fgm)
    TextView fgmTv;

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        pageIndex = (int) getArguments().getSerializable(PAGE_INDEX) + 1;
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_main, null);
        // 綁定ButterKnife
        ButterKnife.bind(this, view);

        fgmTv.setText("第"+ pageIndex + "頁");
        if (isVisibleToUser) {
            new UserInfoFragmentPresenter(this);
            mIFragmentPresenter.start();
        }
        return view;
    }

    @Override
    public void showData() {
        Toast.makeText(getActivity(), "這是第" + pageIndex + "個頁面", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void setPresenter(UserInfoContract.IUserInfoFragmentPresenter mIFragmentPresenter) {
        this.mIFragmentPresenter = mIFragmentPresenter;
    }

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        this.isVisibleToUser = isVisibleToUser;
    }
}

view層分析完后醒颖,接下來我們來分析壓軸戲:presenter(邏輯層)妻怎,邏輯層分組情況如下圖所示:

QQ圖片20160930161532.png

我們可以看到邏輯層分組里面包含一個activity邏輯層類、fragment邏輯層類泞歉。
接下來我們看下activity邏輯層類逼侦,代碼如下:

package com.ym.mvpdemo.module.presenter.userinfo;

import android.os.Handler;
import com.ym.mvpdemo.module.contract.IActivityLifeCycle;
import com.ym.mvpdemo.module.contract.UserInfoContract;
import com.ym.mvpdemo.module.model.UserInfoModel;

/**
 * @className: UserInfoActivityPresenter
 * @classDescription: 用戶信息activity邏輯層
 * @author: leibing
 * @createTime: 2016/8/11
 */
public class UserInfoActivityPresenter implements UserInfoContract.IUserInfoActivityPresenter, IActivityLifeCycle {
    // 用戶信息activity接口
    private UserInfoContract.IUserInfoActivity mIUserInfoActivity;

    /**
     * 構(gòu)造函數(shù)
     * @author leibing
     * @createTime 2016/08/23
     * @lastModify 2016/08/23
     * @param mIUserInfoActivity 用戶信息activity接口
     * @return
     */
    public UserInfoActivityPresenter(UserInfoContract.IUserInfoActivity mIUserInfoActivity) {
        this.mIUserInfoActivity = mIUserInfoActivity;
        // 設置邏輯
        mIUserInfoActivity.setPresenter(this);
        // 設置生命周期
        mIUserInfoActivity.setILifeCycle(this);
    }

    @Override
    public void loadUserInfo() {
        String userId = mIUserInfoActivity.loadUserId();
        System.out.println("ddddddddddddddddddddddddddd userId = " + userId);
        mIUserInfoActivity.showLoading();//接口請求前顯示loading
        //這里模擬接口請求回調(diào)-
        new Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                //模擬接口返回的json,并轉(zhuǎn)換為javaBean
                UserInfoModel userInfoModel = new UserInfoModel("小寶", 1, "杭州");
                mIUserInfoActivity.showUserInfo(userInfoModel);
                mIUserInfoActivity.dismissLoading();
            }
        }, 3000);
    }

    @Override
    public void start() {
        loadUserInfo();
    }

    @Override
    public void onRestart() {
        System.out.println("ddddddddddddddddddddd present onRestart");
    }

    @Override
    public void onCreate() {
        System.out.println("ddddddddddddddddddddd present onCreate");
    }

    @Override
    public void onStart() {
        System.out.println("ddddddddddddddddddddd present onStart");
    }

    @Override
    public void onResume() {
        System.out.println("ddddddddddddddddddddd present onResume");
    }

    @Override
    public void onPause() {
        System.out.println("ddddddddddddddddddddd present onPause");
    }

    @Override
    public void onStop() {
        System.out.println("ddddddddddddddddddddd present onStop");
    }

    @Override
    public void onDestroy() {
        System.out.println("ddddddddddddddddddddd present onDestroy");
    }
}

上面代碼主要是將activity 界面層與邏輯層關聯(lián)起來腰耙、實現(xiàn)生命周期映射接口榛丢、實現(xiàn)邏輯層接口,有不懂得童鞋可以結(jié)合view層多看看就明白了挺庞,fragment邏輯層類似activity晰赞,就不多做分析了。

童鞋們,筆者寫的這個mvp是不是很簡單掖鱼?大家可以嘗試去寫下然走。

項目地址: MvpDemo

效果圖如下:

MvpDemo.gif

關于作者

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市戏挡,隨后出現(xiàn)的幾起案子芍瑞,更是在濱河造成了極大的恐慌,老刑警劉巖褐墅,帶你破解...
    沈念sama閱讀 218,036評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件拆檬,死亡現(xiàn)場離奇詭異,居然都是意外死亡掌栅,警方通過查閱死者的電腦和手機秩仆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來猾封,“玉大人,你說我怎么就攤上這事噪珊∩卧担” “怎么了?”我有些...
    開封第一講書人閱讀 164,411評論 0 354
  • 文/不壞的土叔 我叫張陵痢站,是天一觀的道長磷箕。 經(jīng)常有香客問我,道長阵难,這世上最難降的妖魔是什么岳枷? 我笑而不...
    開封第一講書人閱讀 58,622評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮呜叫,結(jié)果婚禮上空繁,老公的妹妹穿的比我還像新娘。我一直安慰自己朱庆,他們只是感情好盛泡,可當我...
    茶點故事閱讀 67,661評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著娱颊,像睡著了一般傲诵。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上箱硕,一...
    開封第一講書人閱讀 51,521評論 1 304
  • 那天拴竹,我揣著相機與錄音,去河邊找鬼剧罩。 笑死栓拜,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播菱属,決...
    沈念sama閱讀 40,288評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼钳榨,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了纽门?” 一聲冷哼從身側(cè)響起薛耻,我...
    開封第一講書人閱讀 39,200評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎赏陵,沒想到半個月后饼齿,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,644評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡蝙搔,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,837評論 3 336
  • 正文 我和宋清朗相戀三年缕溉,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片吃型。...
    茶點故事閱讀 39,953評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡证鸥,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出勤晚,到底是詐尸還是另有隱情枉层,我是刑警寧澤,帶...
    沈念sama閱讀 35,673評論 5 346
  • 正文 年R本政府宣布赐写,位于F島的核電站鸟蜡,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏挺邀。R本人自食惡果不足惜揉忘,卻給世界環(huán)境...
    茶點故事閱讀 41,281評論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望端铛。 院中可真熱鬧泣矛,春花似錦、人聲如沸沦补。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,889評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽夕膀。三九已至虚倒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間产舞,已是汗流浹背魂奥。 一陣腳步聲響...
    開封第一講書人閱讀 33,011評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留易猫,地道東北人耻煤。 一個月前我還...
    沈念sama閱讀 48,119評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親哈蝇。 傳聞我的和親對象是個殘疾皇子棺妓,可洞房花燭夜當晚...
    茶點故事閱讀 44,901評論 2 355

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