细读 VirtualApk 之资源加载(上)

1. LoadedPlugin

由于插件是不安装的,为了宿主可以与插件正常工作,需要宿主可以加载插件的类,可以访问插件的静态资源和本地库.

LoadedPlugin 代表着插件 APK,一个 LoadedPlugin 对应一个插件,LoadedPlugin 不但保存了插件 APK 一切信息还负责打通宿主与插件之间的屏障,使宿主可以加载插件的类,可以加载插件的资源和本地库.

1.1 构造方法

// LoadedPlugin.java
protected Resources mResources;
public LoadedPlugin(PluginManager pluginManager, Context context, File apk) throws Exception {
    // ...
  this.mResources = createResources(context, getPackageName(), apk);
    // ...
}

LoadedPlugin 的成员变量 mResources 是访问插件静态资源的 Resources .构造方法内会调用 createResources() 初始化 LoadedPlugin.mResources.

这里 Context 是宿主 Application Context ,是初始化 VirtualApk 时传入的,请记住这对后面阅读有帮助.

1.2 LoadedPlugin.createResources()

// LoadedPlugin.java
protected Resources createResources(Context context, String packageName, File apk) throws Exception {
  if (Constants.COMBINE_RESOURCES) {
    return ResourcesManager.createResources(context, packageName, apk);
  } else {
    // 宿主 Resource
    Resources hostResources = context.getResources();
    // 构建只加载插件 APK 的 AssetManager
    AssetManager assetManager = createAssetManager(context, apk);
    // 构建新的 Resources ,只加载插件 apk 资源
    return new Resources(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
  }
}

// Constants.java
public class Constants {
        // ...
    public static final boolean COMBINE_RESOURCES = true;
    // ...
}

createResources() 目的是构建一个新的 Resources ,这个 Resources 可以同时访问宿主和插件的资源或者只可以访问插件资源,由常量 COMBINE_RESOURCES 控制,默认为 true.

为了方便先看只访问插件资源的情况,很简单,只需要构建一个新的只加载插件 APK AssetManager,然后用它构建新的 Resources 即可.

1.3 LoadedPlugin.createAssetManager()

// LoadedPlugin.java
protected AssetManager createAssetManager(Context context, File apk) throws Exception {
  AssetManager am = AssetManager.class.newInstance();
  Reflector.with(am).method("addAssetPath", String.class).call(apk.getAbsolutePath());
  return am;
}
  • createAssetManager() 首先通过 AssetManager.class.newInstance() 构建一个新的 AssetManager 实例.
  • 然后通过反射调用 AssetManager.addAssetPath() 把插件 apk 路径添加到 AssetManager 的加载路径.

现在回头看当 COMBINE_RESOURCES 为 true 时构建 Resources 需要调用 ResourcesManager.createResources().

2. VirtualAPK 的 ResourcesManager

本文中会出现两个 ResourcesManager ,一个是 VirtualApk 的 ResourcesManager 一个是系统的 ResourcesManager,下面这个是 VirtualApk 的 ResourcesManager.

VirtualApk.ResourcesManager 是创建和管理插件 Resources 的类,是 LoadedPlugin 打破墙壁让宿主访问插件资源的关键角色.

2.1 ResourcesManager.ResourcesManager.createResources()

// ResourcesManager.java
public static synchronized Resources createResources(Context hostContext, String packageName, File apk) throws Exception {
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
    // API >= 24
    return createResourcesForN(hostContext, packageName, apk);
  }

  // 构建新的 Resources
  Resources resources = ResourcesManager.createResourcesSimple(hostContext, apk.getAbsolutePath());
  // 用新的 Resources Hook 宿主 Resources ( API < 24 )
  ResourcesManager.hookResources(hostContext, resources);
  return resources;
}

实现方式以 Android7.0 (API - 24) 为界限分为两种:

  • 7.0 以前需要适配各种厂商的方式构建 Resources ,然后通过 Hook 的方式替代宿主进程正在使用的所有 Resources.

  • 7.0 之后通过利用系统 API 优雅地帮助我们构建新的 Resources,不需要适配厂商也不需要 Hook.

先从低版本实现说起,从 ResourcesManager.createResourcesSimple() 开始.

API 1 - 23 (Android1.0 - Android7.0)

2.2 ResourcesManager.createResourcesSimple()

// ResourcesManager.java
private static Resources createResourcesSimple(Context hostContext, String apk) throws Exception {
  // 宿主 Resources
  Resources hostResources = hostContext.getResources();
  Resources newResources = null;
  AssetManager assetManager;
  Reflector reflector = Reflector.on(AssetManager.class).method("addAssetPath", String.class);
  // 1.
  // 根据API版本不同构建新的 AssetManager 或者直接获取宿主的 AssetManager
  if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
    // API 1 - 20
    assetManager = AssetManager.class.newInstance();
    reflector.bind(assetManager);
    final int cookie1 = reflector.call(hostContext.getApplicationInfo().sourceDir);
    if (cookie1 == 0) {
      throw new RuntimeException("createResources failed, can't addAssetPath for " + hostContext.getApplicationInfo().sourceDir);
    }
  } else {
    // API 21 - 23
    assetManager = hostResources.getAssets();
    reflector.bind(assetManager);
  }
  // 调用 AssetManager.addAssetPath() 添加 apk 路径
  final int cookie2 = reflector.call(apk);
  if (cookie2 == 0) {
    // AssetManager.addAssetPath() 返回 0 表示失败
    throw new RuntimeException("createResources failed, can't addAssetPath for " + apk);
  }
  // 获取所有插件并调用 AssetManager.addAssetPath() 添加其他插件apk路径
  List<LoadedPlugin> pluginList = PluginManager.getInstance(hostContext).getAllLoadedPlugins();
  for (LoadedPlugin plugin : pluginList) {
    final int cookie3 = reflector.call(plugin.getLocation());
    if (cookie3 == 0) {
      throw new RuntimeException("createResources failed, can't addAssetPath for " + plugin.getLocation());
    }
  }
  
  // 2.
  // 针对不同厂商用相应方法构建新的 Resources
  if (isMiUi(hostResources)) {
    // 小米
    newResources = MiUiResourcesCompat.createResources(hostResources, assetManager);
  } else if (isVivo(hostResources)) {
    // Vivo
    newResources = VivoResourcesCompat.createResources(hostContext, hostResources, assetManager);
  } else if (isNubia(hostResources)) {
    // 努比亚
    newResources = NubiaResourcesCompat.createResources(hostResources, assetManager);
  } else if (isNotRawResources(hostResources)) {
    // 非原生
    newResources = AdaptationResourcesCompat.createResources(hostResources, assetManager);
  } else {
    // is raw android resources
    // 原生
    newResources = new Resources(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
  }
  // 3.
  // lastly, sync all LoadedPlugin to newResources
  // 使用新的 Resources 更新所有插件的 Resources
  for (LoadedPlugin plugin : pluginList) {
    plugin.updateResources(newResources);
  }

  return newResources;
}

// AssetManager.java (Android 5.0)
/**
 * Add an additional set of assets to the asset manager.  This can be
 * either a directory or ZIP file.  Not for use by applications.  Returns
 * the cookie of the added asset, or 0 on failure.
 * {@hide}
 */
public final int addAssetPath(String path) {
  synchronized (this) {
    int res = addAssetPathNative(path);
    makeStringBlocks(mStringBlocks);
    return res;
  }
}

createResourcesSimple() 看似很长,主要分三步:

  1. 根据系统版本的不同构建新的或者复用宿主 AssetManager ,然后通过反射调用 AssetManager.addAssetPath() 添加插件 apk 路径,最后遍历把所有 LoadedPlugins 的插件 APK 路径都调用 AssetManager.addAssetPath() 添加进去.

    • API 1-20: 创建新的 AssetManager 实例.
    • API 21-23: 获取宿主的 AssetManager 实例.
  2. 根据不同厂商创建新的 Resources 实例.

  3. 更新所有 LoadedPlugin 的 Resources.

每一步都在代码上标了注释.其中 addAssetPath() 返回值可以判断是否添加成功.

低版本构建 Resources 的时候不单只加载 LoadedPlugin 对应的插件 APK 资源,还添加其他插件 APK 的路径,最后还要更新所有插件的 Resources,所以所有 LoadedPlugin.mResources 指向的都是同一个 Resources,且这个 Resources 可以访问宿主和所有插件 APK 的资源.

下面稍微看下适配厂家构建 Resources 的代码.

// ResourcesManager.java
private static final class MiUiResourcesCompat {
  private static Resources createResources(Resources hostResources, AssetManager assetManager) throws Exception {
    Reflector reflector = Reflector.on("android.content.res.MiuiResources");
    Resources newResources = reflector.constructor(AssetManager.class, DisplayMetrics.class, Configuration.class)
      .newInstance(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
    return newResources;
  }
}

private static final class VivoResourcesCompat {
  private static Resources createResources(Context hostContext, Resources hostResources, AssetManager assetManager) throws Exception {
    Reflector reflector = Reflector.on("android.content.res.VivoResources");
    Resources newResources = reflector.constructor(AssetManager.class, DisplayMetrics.class, Configuration.class)
      .newInstance(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
    reflector.method("init", String.class).callByCaller(newResources, hostContext.getPackageName());
    reflector.field("mThemeValues");
    reflector.set(newResources, reflector.get(hostResources));
    return newResources;
  }
}

private static final class NubiaResourcesCompat {
  private static Resources createResources(Resources hostResources, AssetManager assetManager) throws Exception {
    Reflector reflector = Reflector.on("android.content.res.NubiaResources");
    Resources newResources = reflector.constructor(AssetManager.class, DisplayMetrics.class, Configuration.class)
      .newInstance(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
    return newResources;
  }
}

private static final class AdaptationResourcesCompat {
  private static Resources createResources(Resources hostResources, AssetManager assetManager) throws Exception {
    Resources newResources;
    try {
      Reflector reflector = Reflector.with(hostResources);
      newResources = reflector.constructor(AssetManager.class, DisplayMetrics.class, Configuration.class)
        .newInstance(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
    } catch (Exception e) {
      newResources = new Resources(assetManager, hostResources.getDisplayMetrics(), hostResources.getConfiguration());
    }

    return newResources;
  }
}

由于厂家定制 ROM 的原因会有厂家定制的 Resources ,当然原生的 Resources 是保留的,所以针对不适配的厂家就会创建原生的 Resources .通过反射调用构造方法创建实例.

ResourcesManager.createResourcesSimple() 到这里结束了,接下来看 ResourcesManager.hookResources(hostContext, resources)

2.3 ResourcesManager.hookResources()

public static void hookResources(Context base, Resources resources) {
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
    return;
  }
  // API 1 - 23
  try {
    // 1.反射替代 ContextImpl.mResources
    Reflector reflector = Reflector.with(base);
    reflector.field("mResources").set(resources);
    // 2.反射替代 ContextImpl.mPackageInfo.mResources
    Object loadedApk = reflector.field("mPackageInfo").get();
    Reflector.with(loadedApk).field("mResources").set(resources);

    // 获取宿主 ActivityThread
    Object activityThread = ActivityThread.currentActivityThread();
    Object resManager;
    // 3.根据版本不同获取 ResourcesManager
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
      // API 19 - 23
      resManager = android.app.ResourcesManager.getInstance();
    } else {
      // API 1 < 18
      resManager = Reflector.with(activityThread).field("mResourcesManager").get();
    }
    // 获取保存 Resources 弱引用的 ResourcesManager.mActiveResources
    // ArrayMap<ResourcesKey, WeakReference<Resources>>
    Map<Object, WeakReference<Resources>> map = Reflector.with(resManager).field("mActiveResources").get();
    Object key = map.keySet().iterator().next();
    // 把 Resources 添加到 map 中
    map.put(key, new WeakReference<>(resources));
  } catch (Exception e) {
    Log.w(TAG, e);
  }
}
  1. 首先用反射把新的 Resources 替换宿主 ContextImpl.mResources 指向的 Resources 对象.

  2. 用新的 Resources 替换宿主 ContextImpl.mPackageInfo.mResources 指向的对象.

  3. 根据系统版本的不同获取系统的 ResourcesManager.

    • API 1 - 18: 反射获取 ActivityThread.mResourcesManager.

    • API 19 - 23: 直接创建系统 ResourcesManager 实例.

  4. 反射把新的 Resources 添加到 ResourcesManager.mActiveResources.

Android 7.0 以前的 ResourcesManager 会有一个 Map 保存 Resources 的弱引用和他对应的 ResourcesKey.

// ResourcesManager.java (Android 4.4.4)
final ArrayMap<ResourcesKey, WeakReference<Resources> > mActiveResources = new ArrayMap<ResourcesKey, WeakReference<Resources> >();

低版本创建 Resources 并替换的实现到这里结束了,下面从 createResourcesForN() 开始看高版本的实现.


API 24 - lastest (Android7.0 - lastest)

2.4 ResourcesManager.createResourcesForN()

首先看方法注释

/**
 * Use System Apis to update all existing resources.
 * <br/>
 * 1. Update ApplicationInfo.splitSourceDirs and LoadedApk.mSplitResDirs
 * <br/>
 * 2. Replace all keys of ResourcesManager.mResourceImpls to new ResourcesKey
 * <br/>
 * 3. Use ResourcesManager.appendLibAssetForMainAssetPath(appInfo.publicSourceDir, "${packageName}.vastub") to update all existing resources.
 * <br/>
 * <p>
 * see android.webkit.WebViewDelegate.addWebViewAssetPath(Context)
 */

使用系统 API 更新所有 Resources (宿主+插件)

  1. 更新 ApplicationInfo.splitSourceDirs 和 LoadedApk.mSplitResDirs.

  2. 在 ResourcesManager.mResourceImpls 中用新的 ResourcesKey 替换所有旧的 ResourcesKey.

  3. 调用 ResourcesManager.appendLibAssetForMainAssetPath(appInfo.publicSourceDir, "${packageName}.vastub") 更新所有资源.

参考 android.webkit.WebViewDelegate.addWebViewAssetPath(Context)

下面贴出源码,带有每一步的标注,一步一步地看.

// ResourcesManager.java
@TargetApi(Build.VERSION_CODES.N)
private static Resources createResourcesForN(Context context, String packageName, File apk) throws Exception {
  long startTime = System.currentTimeMillis();
  String newAssetPath = apk.getAbsolutePath();
  ApplicationInfo info = context.getApplicationInfo();
  // 宿主的 base.apk 路径
  String baseResDir = info.publicSourceDir;

  // 1.1 更新 ApplicationInfo.splitSourceDirs
  info.splitSourceDirs = append(info.splitSourceDirs, newAssetPath);
  // 反射获取 ContextImpl.mPackageInfo
  LoadedApk loadedApk = Reflector.with(context).field("mPackageInfo").get();
  Reflector rLoadedApk = Reflector.with(loadedApk).field("mSplitResDirs");
  // 反射获取 LoadedApk.mSplitResDirs 即所有分包地址 splitResDirs
  String[] splitResDirs = rLoadedApk.get();
  // 1.2 更新 LoadedApk.mSplitResDirs
  rLoadedApk.set(append(splitResDirs, newAssetPath));
  // 构建新的 ResourcesManager
  final android.app.ResourcesManager resourcesManager = android.app.ResourcesManager.getInstance();
  // 获取 ResourcesManager.mResourceImpls
  // ArrayMap<ResourcesKey, WeakReference<ResourcesImpl>>
  ArrayMap<ResourcesKey, WeakReference<ResourcesImpl>> originalMap = Reflector.with(resourcesManager).field("mResourceImpls").get();

  synchronized (resourcesManager) {
    HashMap<ResourcesKey, WeakReference<ResourcesImpl>> resolvedMap = new HashMap<>();

    // 2.在 ResourcesManager.mResourceImpls 中用新的 ResourcesKey 替换所有旧的 ResourcesKey
    if (Build.VERSION.SDK_INT >= 28
        || (Build.VERSION.SDK_INT == 27 && Build.VERSION.PREVIEW_SDK_INT != 0)) { // P Preview
      // API 28 - lastest or Android P Preview
      ResourcesManagerCompatForP.resolveResourcesImplMap(originalMap, resolvedMap, context, loadedApk);

    } else {
      // API 24 - 27
      ResourcesManagerCompatForN.resolveResourcesImplMap(originalMap, resolvedMap, baseResDir, newAssetPath);
    }

    originalMap.clear();
    originalMap.putAll(resolvedMap);
  }

  // 3.
  // 刷新 ResourcesManager.mResourceImpls 并添加指向插件的 ResourceImpl
  android.app.ResourcesManager.getInstance().appendLibAssetForMainAssetPath(baseResDir, packageName + ".vastub");

  Resources newResources = context.getResources();

  // 4.
  // lastly, sync all LoadedPlugin to newResources
  // 最后把新的 Resources 同步到所有插件 LoadedPlugin 中
  for (LoadedPlugin plugin : PluginManager.getInstance(context).getAllLoadedPlugins()) {
    plugin.updateResources(newResources);
  }

  Log.d(TAG, "createResourcesForN cost time: +" + (System.currentTimeMillis() - startTime) + "ms");
  return newResources;
}
Step1

首先看 ApplicationInfo.splitSourceDirs 和 LoadedApk.mSplitResDirs 是啥:

// ApplicationInfo.java
/**
 * Full paths to zero or more split APKs, indexed by the same order as {@link #splitNames}.
 */
public String[] splitSourceDirs;

// LoadedApk.java
private String[] mSplitResDirs;

从 ApplicationInfo 注释可以看出 splitSourceDirs 保存该 APP 的所有分包 apk 的绝对路径,可以推断 LoadedApk.mSplitResDirs 也是保存一样的值.

通过反射得到这两个数组然后分别调用 ResourcesManager.append() 把插件 apk 路径添加到这两个数组中.

// ResourcesManager.java
private static String[] append(String[] paths, String newPath) {
  if (contains(paths, newPath)) {
    return paths;
  }

  final int newPathsCount = 1 + (paths != null ? paths.length : 0);
  // 创建一个新的数组
  final String[] newPaths = new String[newPathsCount];
  if (paths != null) {
    System.arraycopy(paths, 0, newPaths, 0, paths.length);
  }
  newPaths[newPathsCount - 1] = newPath;
  return newPaths;
}
Step2

第二步是最复杂的一步.上面说到这一步要替换 ResourcesManager 中保存 ResourceImpls 的 Map 的 key ,但其实还会生成新的指向插件 apk 的 ResourceImpls .

注意:Android 7.0 之前 Resources 是访问资源的实现类,Android 7.0 之后真正访问资源的是 ResourceImpl,而 Resources 变成一个代理类,所以 ResourcesManager 保存的是 ResourceImpl 而不是 Resources,如果对 ResourceImpl 有所修改,还需要替换的 ResourceImpl 对应的 Resources 持有的 ResourceImpl 引用.

// ResourcesManager.java
/**
 * A mapping of ResourceImpls and their configurations. These are heavy weight objects
 * which should be reused as much as possible.
 */
private final ArrayMap<ResourcesKey, WeakReference<ResourcesImpl>> mResourceImpls =
  new ArrayMap<>();

ResourcesManager.mResourceImpls 保存了所有以ResourcesKey 为 key 的 ResourceImpls.

第二步实现步骤:

  1. 首先获取 ResourcesManager.ResourceImpls 作为原始 Map 变量 originalMap.
  2. 创建一个新的 Map 变量 resolvedMap.
  3. 根据 Android 版本的不同调用 ResourcesManagerCompatForP.resolveResourcesImplMap() 或者 ResourcesManagerCompatForN.resolveResourcesImplMap() 根据 originalMap 往 resolvedMap 中添加 ResourceImpls 对应刷新后的 ResourcesKey.
  4. 把 originalMap 内容替换成 resolvedMap 的数据.由于 originalMap 一直引用 ResourcesManager.ResourceImpls 没有改变,所以操作 originalMap 相当于操作 ResourcesManager.ResourceImpls,第二步完成.

我们重点看第二步中的第三小步,其他都简单易懂.

API 24 - 27 (Android7.0 - Android8.1)

首先从低版本开始,Android API 在 24 - 27 之间的调用 ResourcesManagerCompatForN.resolveResourcesImplMap().

2.5 ResourcesManagerCompatForN.resolveResourcesImplMap()
// ResourcesManager.java
private static final class ResourcesManagerCompatForN {
  @TargetApi(Build.VERSION_CODES.KITKAT)
  public static void resolveResourcesImplMap(
    Map<ResourcesKey, WeakReference<ResourcesImpl>> originalMap, 
    Map<ResourcesKey, WeakReference<ResourcesImpl>> resolvedMap, 
    String baseResDir, 
    String newAssetPath) throws Exception {
    for (Map.Entry<ResourcesKey, WeakReference<ResourcesImpl>> entry : originalMap.entrySet()) {
      Log.d(TAG, "resolveResourcesImplMap:" + entry.getKey().mResDir);
      ResourcesKey key = entry.getKey();
      // 判断该 ResourcesImpl 的资源路径是否指向宿主 APK 路径
      if (Objects.equals(key.mResDir, baseResDir)) {
        // 把插件的路径添加到宿主的 mResDir 中生成新的 ResourcesKey
        resolvedMap.put(new ResourcesKey(key.mResDir,
                                         append(key.mSplitResDirs, newAssetPath),
                                         key.mOverlayDirs,
                                         key.mLibDirs,
                                         key.mDisplayId,
                                         key.mOverrideConfiguration,
                                         key.mCompatInfo), entry.getValue());
      } else {
        // 非宿主(系统资源)就直接添加到 resolvedMap
        resolvedMap.put(key, entry.getValue());
      }
    }
  }
}

// Resourceskey.java
public ResourcesKey(@Nullable String resDir,
                        @Nullable String[] splitResDirs,
                        @Nullable String[] overlayDirs,
                        @Nullable String[] libDirs,
                        int displayId,
                        @Nullable Configuration overrideConfig,
                        @Nullable CompatibilityInfo compatInfo){
  // ...
}

首选遍历 originalMap,每次循环都需要判断 ResourcesImpl 是否指向宿主的 APK 路径:

  • true: 调用 ResourcesKey 构造方法,其中方法参数 splitResDirs 在原 ResourcesKey.splitResDirs 基础上添加了插件 apk 路径,其余参数保持不变生成新的 ResourcesKey,并把旧的 ResourcesImpl 和新的 ResourcesKey 存到 resolvedMap.
  • false: 直接用原本的 key 把 ResourcesKey 存到 resolvedMap.

所以 resolveResourcesImplMap() 的核心逻辑就是从 originalMap 中找到指向宿主的 ResourcesImpl 然后把重新构建新的 ResourcesKey 且把插件 APK 路径添加到参数中,然后把新的 key 和旧的 ResourcesImpl 添加到 resolvedMap.注意这里只是更新了带有插件 APK 路径的 ResourcesKey, ResourcesImpl 没变依然不能访问插件资源,这个问题留着后面解决.

高版本请接着阅读细读 VirtualApk 之资源加载(下)

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

推荐阅读更多精彩内容