关于iOS中原生和h5交互的知识总结(二)WKWebView

前言

目录

关于iOS中原生和h5交互的知识总结(一)UIWebView
关于iOS中原生和h5交互的知识总结(二)WKWebView

基于WKWebView的实现,请注意以下几点(可以全局搜索1~7查看代码中对应位置)

1.初始化
2.清理缓存
3.进度条
4.代理方法
5.交互*
6.addScriptMessageHandler导致不释放问题*
7.响应localStorage变化事件*

样例

我们写一个基类WebViewController,之后所有用到WKWebView的controller都可以继承这个基类,省着我们再去在每个controller中再去写一些关于webView的初始化,配置,代理等一堆东西了
WebViewController.h

#import <UIKit/UIKit.h>
#import <WebKit/WebKit.h>
@interface WebViewController : UIViewController

//跳转到该controller时需要加载的页面url
@property (nonatomic, copy) NSString *url;
//是否显示导航栏
@property (nonatomic, assign) BOOL hideNavigationBar;
//暴露进度条,方便子类修改样式,如果进度条样式全局统一,可以不暴露
@property (nonatomic, strong) UIProgressView *progressView;
//暴露webView,用于子类loadRequest
@property (nonatomic, strong) WKWebView *webView;
//暴露WKUserContentController,如果素有WKUserContentController操作都在基类中,可以不暴露
@property (nonatomic, strong) WKUserContentController *controller;

@end

WebViewController.m

#import "WebViewController.h"
#import "WebViewConstant.h"
#import "WebLoadingView.h"
#import "NotifyConstant.h"
#import "ProgressPoolSingleton.h"
#import "WeakScriptMessageDelegate.h"

#import "NavigationController.h"
#import "SendTableViewController.h"
#import "LoginTableViewController.h"
#import "VideoBrowseController.h"
#import "MemberAuthStatusViewController.h"

@interface WebViewController ()<WKUIDelegate,WKNavigationDelegate,WKScriptMessageHandler,UITabBarControllerDelegate,UIScrollViewDelegate,UINavigationControllerDelegate>
@property (nonatomic, assign) CGFloat contentOffsetY;
@end

@implementation WebViewController

- (UIStatusBarStyle)preferredStatusBarStyle
{
    return UIStatusBarStyleLightContent;
}

//默认隐藏状态栏
- (id)init
{
    self = [super init];
    if (self) {
        _hideNavigationBar = YES;
    }
    return self;
}

- (id)initWithCoder:(NSCoder *)aDecoder
{
    self = [super initWithCoder:aDecoder];
    if (self) {
        _hideNavigationBar = YES;
    }
    return self;
}

//释放时,移除KVO和scriptMessageHandler
- (void)dealloc
{
    [self.webView removeObserver:self forKeyPath:@"estimatedProgress"];
    [[NSNotificationCenter defaultCenter] removeObserver:self];
    
    [self.controller removeScriptMessageHandlerForName:@"getToken"];
    [self.controller removeScriptMessageHandlerForName:@"showWebScene"];
    [self.controller removeScriptMessageHandlerForName:@"showSendScene"];
    [self.controller removeScriptMessageHandlerForName:@"exitScene"];
    [self.controller removeScriptMessageHandlerForName:@"showLoginScene"];
    [self.controller removeScriptMessageHandlerForName:@"showVideoPlayScene"];
    [self.controller removeScriptMessageHandlerForName:@"showMemberCertificationScene"];
}

//添加addScriptMessageHandler
//WeakScriptMessageDelegate这个类是防止添加到self,导致不能释放的问题
//6.addScriptMessageHandler导致不释放问题
- (void)addScriptMessageHandler
{
    WeakScriptMessageDelegate *delegate = [[WeakScriptMessageDelegate alloc] initWithDelegate:self];
    [self.controller addScriptMessageHandler:delegate name:@"getToken"];
    [self.controller addScriptMessageHandler:delegate name:@"showWebScene"];
    [self.controller addScriptMessageHandler:delegate name:@"showSendScene"];
    [self.controller addScriptMessageHandler:delegate name:@"exitScene"];
    [self.controller addScriptMessageHandler:delegate name:@"showLoginScene"];
    [self.controller addScriptMessageHandler:delegate name:@"showVideoPlayScene"];
    [self.controller addScriptMessageHandler:delegate name:@"showMemberCertificationScene"];
}

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.contentOffsetY = self.hideNavigationBar == YES ? 20 : 0;
    self.view.backgroundColor = [UIColor colorWithRed:150.0/255.0 green:0 blue:10.0/255.0 alpha:1];
    self.navigationController.delegate = self;

    self.webView.scrollView.delegate = self;
    self.webView.scrollView.bounces = NO;

    // 1.初始化
    WKWebViewConfiguration *configuration = [[WKWebViewConfiguration alloc] init];
    self.controller = [[WKUserContentController alloc] init];
    [self addScriptMessageHandler];
    
    //7.响应localStorage变化事件
    configuration.userContentController = self.controller;
    configuration.processPool = [ProgressPoolSingleton sharedInstance].processPool;

    self.webView = [[WKWebView alloc] initWithFrame:CGRectMake(0, self.contentOffsetY, self.view.bounds.size.width, self.view.bounds.size.height-20) configuration:configuration];
    self.webView.scrollView.bounces = YES;
    self.webView.UIDelegate = self;
    self.webView.navigationDelegate = self;
    [self.view addSubview:self.webView];
    //3.进度条
    [self.view addSubview:self.progressView];
    //观察加载进度
    [self.webView addObserver:self forKeyPath:@"estimatedProgress"
                      options:(NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld) context:nil];

    if (self.url) {
        [self.webView loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:self.url]]];
    }
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardWillShow:) name:UIKeyboardWillHideNotification object:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardWillHide:) name:UIKeyboardWillShowNotification object:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(refresh) name:HRefreshWebVCNotification object:nil];
    [[WebLoadingView sharedInstance] showOnView:self.view];
    
//    [self debugButton];
}

//调试按钮
- (void)debugButton
{
    UIButton *button = [UIButton buttonWithType:UIButtonTypeSystem];
    [button setTitle:@"点我" forState:UIControlStateNormal];
    [button setBackgroundColor:[UIColor greenColor]];
    button.frame = CGRectMake(0, 400, 100, 100);
    [button addTarget:self action:@selector(debugButtonAction) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:button];
}

//2.清理缓存
- (void)debugButtonAction
{
    NSSet *websiteDataTypes = [NSSet setWithObjects:WKWebsiteDataTypeDiskCache,WKWebsiteDataTypeOfflineWebApplicationCache,WKWebsiteDataTypeMemoryCache,WKWebsiteDataTypeLocalStorage,nil];
    NSDate *dateFrom = [NSDate dateWithTimeIntervalSince1970:0];
    
    @weakify(self)
    [[WKWebsiteDataStore defaultDataStore] removeDataOfTypes:websiteDataTypes modifiedSince:dateFrom completionHandler:^{
        @strongify(self)
        [self.webView reload];
    }];
}

- (void)refresh
{
    [self.webView reload];
}

- (void)keyboardWillShow:(NSNotification *)notifycation
{

}

- (void)keyboardWillHide:(NSNotification *)notifycation
{
    
}

//控制是否显示导航条
- (void)navigationController:(UINavigationController *)navigationController willShowViewController:(UIViewController *)viewController animated:(BOOL)animated
{
    if ([viewController isEqual:self]) {
        [self.navigationController setNavigationBarHidden:self.hideNavigationBar animated:YES];
    } else {
        if ([viewController isKindOfClass:WebViewController.class]) {
            [self.navigationController setNavigationBarHidden:((WebViewController *)viewController).hideNavigationBar animated:YES];
        } else {
            [self.navigationController setNavigationBarHidden:YES animated:YES];
        }
    }
}

#pragma mark - KVC
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
    if (object == self.webView && [keyPath isEqualToString:@"estimatedProgress"]) {
        CGFloat newprogress = [[change objectForKey:NSKeyValueChangeNewKey] doubleValue];
        if (newprogress == 1) {
            self.progressView.hidden = YES;
            [self.progressView setProgress:0 animated:NO];
        }else {
            self.progressView.hidden = NO;
            [self.progressView setProgress:newprogress animated:YES];
        }
    }
}

//5.交互
#pragma mark - WKScriptMessageHandler
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message
{
    NSDictionary *dictionary = message.body;
    NSString *nativeCallback;
    if (dictionary &&
        [dictionary isKindOfClass:NSDictionary.class]) {
        nativeCallback = dictionary[NATIVE_CALLBACK];
    }
    
    if ([message.name isEqualToString:@"getToken"]) {
        
        NSString *token = [[[LocalSaveManager sharedInstance] getToken] translateN];
        NSString *responseToken;
        
        if (token) {
            responseToken = [NSString stringWithFormat:@"'%@'",token];
        } else {
            responseToken = @"'invalid_token'";
        }
        
        NSString *evaluateJavaScript = [NSString stringWithFormat:@"%@(%@)",nativeCallback,responseToken];
        
        @weakify(self)
        dispatch_async(dispatch_get_main_queue(), ^ {
            @strongify(self)
            [self.webView evaluateJavaScript:evaluateJavaScript completionHandler:^(id _Nullable result, NSError * _Nullable error) {
                NSLog(@"result:%@",result);
                NSLog(@"error:%@",error);
            }];
        });
    } else if ([message.name isEqualToString:@"exitScene"]) {
        [self.navigationController popViewControllerAnimated:YES];
    } else if ([message.name isEqualToString:@"showWebScene"]) {
        WebViewController *webViewController = [[WebViewController alloc] init];
        webViewController.hidesBottomBarWhenPushed = YES;
        BOOL hideNavigationBar = [dictionary[@"hideNavigationBar"] boolValue];
        webViewController.hideNavigationBar = hideNavigationBar;
        webViewController.url = dictionary[@"url"];
        [self.navigationController pushViewController:webViewController animated:YES];
    } else if ([message.name isEqualToString:@"showSendScene"]) {
        NSString *token = [[LocalSaveManager sharedInstance] getToken];
        if (token) {
            SendTableViewController *sendTableViewController = [self.storyboard instantiateViewControllerWithIdentifier:@"SendTableViewController"];
            [self.navigationController pushViewController:sendTableViewController animated:YES];
        } else {
            LoginTableViewController *loginTableViewController = [self.storyboard instantiateViewControllerWithIdentifier:@"LoginTableViewController"];
            NavigationController *nav = [[NavigationController alloc] initWithRootViewController:loginTableViewController];
            [self presentViewController:nav animated:YES completion:nil];
        }
    } else if ([message.name isEqualToString:@"showLoginScene"]) {
        LoginTableViewController *loginTableViewController = [self.storyboard instantiateViewControllerWithIdentifier:@"LoginTableViewController"];
        NavigationController *nav = [[NavigationController alloc] initWithRootViewController:loginTableViewController];
        [self presentViewController:nav animated:YES completion:nil];
    } else if ([message.name isEqualToString:@"showVideoPlayScene"]) {
        VideoBrowseController *videoBrowseController = [[VideoBrowseController alloc] init];
        videoBrowseController.videoUrl = dictionary[@"video_url"];
        videoBrowseController.thumbnail = dictionary[@"thumbnail_url"];
        [self presentViewController:videoBrowseController animated:YES completion:nil];
    } else if ([message.name isEqualToString:@"showMemberCertificationScene"]) {
        NSString *token = [[LocalSaveManager sharedInstance] getToken];
        if (token) {
            MemberAuthStatusViewController *memberAuthStatusViewController = [[UIStoryboard storyboardWithName:@"Service" bundle:nil] instantiateViewControllerWithIdentifier:@"MemberAuthStatusViewController"];
            NavigationController *nav = [[NavigationController alloc] initWithRootViewController:memberAuthStatusViewController];
            [self presentViewController:nav animated:YES completion:nil];
        } else {
            LoginTableViewController *loginTableViewController = [self.storyboard instantiateViewControllerWithIdentifier:@"LoginTableViewController"];
            NavigationController *nav = [[NavigationController alloc] initWithRootViewController:loginTableViewController];
            [self presentViewController:nav animated:YES completion:nil];
        }
    } else {
        
    }
}

//4.代理方法
#pragma mark - WKUIDelegate
- (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler
{
    completionHandler();
    UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"提示" message:message preferredStyle:UIAlertControllerStyleAlert];
    UIAlertAction *action = [UIAlertAction actionWithTitle:@"确认" style:UIAlertActionStyleDefault handler:nil];
    [alertController addAction:action];
    [self presentViewController:alertController animated:YES completion:nil];
}

#pragma mark - WKNavigationDelegate
- (void)webView:(WKWebView *)webView didStartProvisionalNavigation:(null_unspecified WKNavigation *)navigation
{
    //    [self rotate];
}

- (void)webView:(WKWebView *)webView didFinishNavigation:(null_unspecified WKNavigation *)navigation
{
    //    [self stopRotation];
    [[WebLoadingView sharedInstance] hide];
}

- (void)webView:(WKWebView *)webView didFailNavigation:(null_unspecified WKNavigation *)navigation withError:(NSError *)error
{
    //    [self stopRotation];
    [[WebLoadingView sharedInstance] hide];
}

#pragma mark - get method
- (UIProgressView *)progressView
{
    if (!_progressView)
    {
        _progressView = [[UIProgressView alloc] initWithFrame:CGRectMake(0, self.contentOffsetY, self.view.bounds.size.width, 2)];
        _progressView.tintColor = [UIColor greenColor];
        _progressView.trackTintColor = [UIColor whiteColor];
    }
    return _progressView;
}

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

解答1.2.3.4请自行百度吧,这里主要说下5.6.7

5.交互*

6.addScriptMessageHandler导致不释放问题*

WKWebView和UIWebView js与原生交互是完全不同的,我们需要向messageHandlers中注册标识,js端通过这些标识来与原生交互,WeakScriptMessageDelegate是解决问题6,下面会给出代码

WeakScriptMessageDelegate *delegate = [[WeakScriptMessageDelegate alloc] initWithDelegate:self];
[self.controller addScriptMessageHandler:delegate name:@"getToken"];
[self.controller addScriptMessageHandler:delegate name:@"showWebScene"];
[self.controller addScriptMessageHandler:delegate name:@"showSendScene"];
[self.controller addScriptMessageHandler:delegate name:@"exitScene"];
[self.controller addScriptMessageHandler:delegate name:@"showLoginScene"];
[self.controller addScriptMessageHandler:delegate name:@"showVideoPlayScene"];
[self.controller addScriptMessageHandler:delegate name:@"showMemberCertificationScene"];

然后在WKScriptMessageHandler的代理方法中去处理这些标识(完整代码见WebViewController.m),这里着重说一下getToken()这个方法,还记得讲UIWebView的时候吗,token是可以直接return回去的。可是在WKWebView中是没法在下面的代理方法写return的,所以js调用原生后,原生要给js传值,只能通过下面的方法

- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message
{
    NSDictionary *dictionary = message.body;
    NSString *nativeCallback;
    //nativeCallback是js端定义的方法,native端取到js端定义的方法名后带着token去执行js端的这个方法,取到的就是js端定义的 getTokenCallback这个方法名
    if (dictionary &&
        [dictionary isKindOfClass:NSDictionary.class]) {
        nativeCallback = dictionary[NATIVE_CALLBACK];
    }
    
    if ([message.name isEqualToString:@"getToken"]) {
        
       //获取native端的token
        NSString *token = [[[LocalSaveManager sharedInstance] getToken] translateN];
        NSString *responseToken;
        
        if (token) {
            responseToken = [NSString stringWithFormat:@"'%@'",token];
        } else {
            responseToken = @"'invalid_token'";
        }
        
       //拼接js端的方法和native的token作为方法参数,然后执行,把token回传给js端
        NSString *evaluateJavaScript = [NSString stringWithFormat:@"%@(%@)",nativeCallback,responseToken];
        
        @weakify(self)
        dispatch_async(dispatch_get_main_queue(), ^ {
            @strongify(self)
            [self.webView evaluateJavaScript:evaluateJavaScript completionHandler:^(id _Nullable result, NSError * _Nullable error) {
                NSLog(@"result:%@",result);
                NSLog(@"error:%@",error);
            }];
        });
    } 
}

下面给出WeakScriptMessageDelegate类的代码
WeakScriptMessageDelegate.h

#import <Foundation/Foundation.h>
#import <WebKit/WebKit.h>
@interface WeakScriptMessageDelegate : NSObject<WKScriptMessageHandler>

@property (nonatomic, weak) id<WKScriptMessageHandler> scriptDelegate;

- (instancetype)initWithDelegate:(id<WKScriptMessageHandler>)scriptDelegate;

@end

WeakScriptMessageDelegate.m

#import "WeakScriptMessageDelegate.h"

@implementation WeakScriptMessageDelegate

- (instancetype)initWithDelegate:(id<WKScriptMessageHandler>)scriptDelegate {
    self = [super init];
    if (self) {
        _scriptDelegate = scriptDelegate;
    }
    return self;
}

- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message {
    [self.scriptDelegate userContentController:userContentController didReceiveScriptMessage:message];
}

@end

接下来附上js端的调用代码,js端的代码不太想解释了,看不懂的同学多看几眼相信会看懂的

function isAndroid() {
    var u = navigator.userAgent;
    var isAndroid = u.indexOf('Android') > -1 || u.indexOf('Adr') > -1;
    return isAndroid;
}

function isIOS() {
    var u = navigator.userAgent;
    var isIOS = !!u.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/);
    return isIOS;
}

// public
// {
//  request:request,
//  callback:callback,
// }
var getTokenOptionCallback = null;
function getToken(option) {
    option = option || {};
    option.request = option.request || {};
    option.callback = option.callback || function () {};
    option.complete = option.complete || false;
    option.response = option.response || {};

    if (isIOS()) {
        if (option.complete != true) {
            getTokenOptionCallback = option.callback;
            option.request.nativeCallback = "getTokenCallback";
            window.webkit.messageHandlers.getToken.postMessage(option.request);
        } else {

            option.callback = getTokenOptionCallback;
            option.callback(option.response);
        }
    } else if (isAndroid()) {
        var token = jsObject.getToken(option.username);
        option.callback(token);
    }
}

// private (called by native)
// response
// be same with option.request.nativeCallback
function getTokenCallback(response) {
    var complete = true;
    getToken({
        complete:complete,
        response:response,
    });
}

// public , don't need nativeCallback
function showWebScene(url,hideNavigationBar) {
    if (isIOS()) {
        window.webkit.messageHandlers.showWebScene.postMessage({"url":url});
    } else if (isAndroid()) {
        jsObject.showWebScene();
    }
}

// public , don't need nativeCallback
function showSendScene() {
    if (isIOS()) {
        window.webkit.messageHandlers.showSendScene.postMessage(null);
    } else if (isAndroid()) {
        jsObject.showSendScene();
    }
}

// public , don't need nativeCallback
function exitScene() {
    if (isIOS()) {
        window.webkit.messageHandlers.exitScene.postMessage(null);
    } else if (isAndroid()) {
        jsObject.exitScene();
    }
}

// public , don't need nativeCallback
function showLoginScene() {
    if (isIOS()) {
        window.webkit.messageHandlers.showLoginScene.postMessage(null);
    } else if (isAndroid()) {
        jsObject.exitScene();
    }
}

// public , don't need nativeCallback
function showVideoPlayScene(thumbnail_url,video_url) {
    var parameter = {
        "thumbnail_url":thumbnail_url,
        "video_url":video_url
    };
    if (isIOS()) {
        window.webkit.messageHandlers.showVideoPlayScene.postMessage(parameter);
    } else if (isAndroid()) {
        jsObject.showVideoPlayScene(parameter);
    }
}
getToken({
   request:{
      username:'hliu',
      password:'123456',
  },callback:function(token) {
      alert(x+","+token);
  },
});

7.响应localStorage变化事件*

不清楚localStorage事件及用途的同学可以自行百度一下,简单说下我们当时的需求,我们从A界面通过导航控制器跳转到B界面,然后希望,B界面的修改能在A界面中得到响应,在safari中试了,不同窗口间是可以响应localStorage变化事件的,可是在iOS中却死活的不行。后来经过调研终于知道原因,每个WKWebView的实例都会有自己的进程池WKProcessPool,不同进程池中是不能及时响应localStorage变化的,所以现在是让所有的WKWebView使用同一个进程池,就能让不同的WKWebView及时的响应localStorage变化事件

    configuration.userContentController = self.controller;
    configuration.processPool = [ProgressPoolSingleton sharedInstance].processPool;

下面给出ProgressPoolSingleton的代码
ProgressPoolSingleton.h

#import <Foundation/Foundation.h>
#import <WebKit/WebKit.h>
@interface ProgressPoolSingleton : NSObject

+ (instancetype)sharedInstance;

@property (nonatomic, strong) WKProcessPool *processPool;

@end

ProgressPoolSingleton.m

#import "ProgressPoolSingleton.h"

@implementation ProgressPoolSingleton

+ (instancetype)sharedInstance
{
    static ProgressPoolSingleton *progressPoolSingleton;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        progressPoolSingleton = [[ProgressPoolSingleton alloc] init];
    });
    return progressPoolSingleton;
}

- (id)init
{
    self = [super init];
    if (self) {
        self.processPool = [WKProcessPool new];
    }
    return self;
}

@end

附上js样例代码
storage_event.html

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<title>Examples</title>
<meta name="description" content="">
<meta name="keywords" content="">
<link href="" rel="stylesheet">
</head>
<body>

    <div id="ddd">数字</div>
    <script type="text/javascript">

        var storage = window.localStorage;
        var value = parseInt(storage.count_n);
        document.getElementById("ddd").innerHTML = value;

        window.addEventListener('storage', onStorageChange);

        function onStorageChange(e) {
            console.log(e.key);
            console.log(e.newValue);

            var value = parseInt(storage.count_n);
            document.getElementById("ddd").innerHTML = value;
        }


    </script>
</body>
</html>

storage_dispach.html

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<title>Examples</title>
<meta name="description" content="">
<meta name="keywords" content="">
<link href="" rel="stylesheet">
</head>
<body>
    <script type="text/javascript" >

        var index = 0;
        if (localStorage.count_n == undefined) {
            localStorage.count_n = index;
        }

        function clickMethod() {
            var value = parseInt(localStorage.count_n) + 1;
            localStorage.count_n = value;
            alert(value);
        }
    </script>

    In Page 1
    <button id="addBtn"
            onclick="clickMethod();">
            Add
    </button>
    <button id="clearBtn"
            onclick="localStorage.clear()">
            Clear
    </button>
</body>
</html>

再附上我的web相关目录


屏幕快照 2018-01-24 下午2.45.45.png

总结

讲解的可能有些粗糙,不过附上了大量代码,小伙伴们看了,应该会有所收获的。日后还会对文章进行修改,完善,增加新的内容。还有哪儿不懂的小伙伴,欢迎留言,我会一一解答。

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

推荐阅读更多精彩内容