WKWebView 的使用和封装

前言

项目中有个新闻资讯模块展示公司和相关行业的最新动态。
这个部分基本是以展示网页为主,内部可能会有一些 native 和 JS 代码的交互。
因为是新项目,所以决定采用 iOS 8 中新出的 WebKit。
本文是对 WebKit 框架中 WKWebView 的一些学习和封装

WKWebViewDemo 地址

UIWebView 和 WKWebView

这两者都是 iOS 中展示 web 相关的组件。前者 iOS 2.0 就有了,后者是 iOS 8.0 时候新加的。

网络中关于两者的差异和性能对比分析很多,这里不再赘述。只是说明一下苹果文档中的重要提示以及自己需要功能的实现:

官方文档中重要提示

文档中主要说了以下几点:

  • iOS 8 之后,应该用 WKWebView 代替 UIWebView。并可以设置 WKPreferencesjavaScriptEnabled 属性决定是否支持 web 内容执行 JavaScript 代码
  • iOS 10 之后必须在 info.plist 文件中包含要访问数据的描述。对于图库的访问必须包含 NSPhotoLibraryUsageDescriptionNSCameraUsageDescription 否则会直接 crash
  • 加载本地 HTML 文件使用: loadHTMLString:baseURL: 方法
  • 加载网络内容使用 : loadRequest: 方法
  • stopLoading 方法用来结束加载。loading 属性查看WK进程中是否加载中
  • goBackgoForward 方法可用于 WKWebView 的前进后退. canGoBackcanGoForward 属性来查看是否能前进后退
  • 通常WKWebView会自动识别电话号码,并把它设置成可打电话的链接。如果不用这个功能: 设置 dataDetectorTypes 属性中 UIDataDetectorTypes 的位字段不包含 UIDataDetectorTypePhoneNumber.
  • scalesPageToFit 属性用于第一次加载网页内容时候设置是否可以使用手势改变网页缩放。 设置后用户就可以手势缩放网页大小
  • 记录网页加载网络内容可以设置WKWebView 的 delegate 并遵守 UIWebViewDelegate 协议
  • 不要在 网页中嵌入 UIScrollView 及其子类,这样会导致手势等行为混乱

有了基本的概念,就可以去看一下 WKWebView 的具体文档了。如果怕官方文档麻烦也可以直接看网络上别人整理好的网络整理

下面是我整理的 WebView 和 H5 调用逻辑图:

特别说明一点:

0. OC 执行 JS 方法
- (void)evaluateJavaScript:(NSString *)javaScriptString completionHandler:(void (^)(id, NSError *error))completionHandler; 这个方法中webView调用JS,block只是成功或失败的回调 1. JS方法中
window.webkit.messageHandlers.webViewApp.postMessage(message);
作用是JS 向之前注册的 webViewApp 发送消息。 OC 端接到消息会调用 <WKScriptMessageHandler> 中下面代理方法
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message

功能需求

对于项目而言,网页功能无需太多,主要满足以下几点

基本展示功能

  1. 导航栏下显示加载进度条
  2. 导航栏 title 展示网页内容当前 title
  3. 网页内容的刷新、前进、后退
  4. 网页内容加载、刷新过程中 HUD 提示

JS和OC交互功能:交互的数据格式和方法名等需要和H5端具体协调

  1. App内登录后,访问 web 需要对应用户token
  2. 网页中点击超链接,新开页面处理,同上也需要拦截新URL请求并补全token参数
  3. 跟JS交互中对JS返回值的处理
  4. 简单JS代码注入,如资讯内容底部加一些赞和分享等功能<曾经就有接口只返回一段JS>

以上这些基本功能中基本展示功能都比较简单,和JS交互的部分需要和 H5 端小伙伴共同定义数据结构和互调的方法名、参数等。所以需要具体问题具体分析。项目以我自己 Demo 为例说一下。

功能实现

为实现功能主要封装了三个类

XYWKWebViewController  ---> 管理 webView 加载相关的代理方法
XYWKWebView ---> 封装 webView 请求相关方法
XYScriptMessage ---> 封装JS回调信息

XYWKWebView 核心功能

加载本地HTML文件

/**
* 加载本地HTML页面
*
* @param htmlName html页面文件名称
*/
- (void)loadLocalHTMLWithFileName:(nonnull NSString *)htmlName

实现代码

- (void)loadLocalHTMLWithFileName:(nonnull NSString *)htmlName {

    NSString *path = [[NSBundle mainBundle] bundlePath];
NSURL *baseURL = [NSURL fileURLWithPath:path];
NSString * htmlPath = [[NSBundle mainBundle] pathForResource:htmlName
ofType:@"html"];
NSString * htmlCont = [NSString stringWithContentsOfFile:htmlPath
encoding:NSUTF8StringEncoding
error:nil]; // WKWebView 的 loadHTMLString: 方法
[self loadHTMLString:htmlCont baseURL:baseURL];
}

加载网络内容

// 加载网络内容,根据是否有参数选不同方法
- (void)loadRequestWithRelativeUrl:(nonnull NSString *)relativeUrl; - (void)loadRequestWithRelativeUrl:(nonnull NSString *)relativeUrl params:(nullable NSDictionary *)params;

实现代码

- (void)loadRequestWithRelativeUrl:(NSString *)relativeUrl params:(NSDictionary *)params {

    NSURL *url = [self generateURL:relativeUrl params:params];

    [self loadRequest:[NSURLRequest requestWithURL:url]];
} - (NSURL *)generateURL:(NSString*)baseURL params:(NSDictionary*)params { self.webViewRequestUrl = baseURL;
self.webViewRequestParams = params; NSMutableDictionary *param = [NSMutableDictionary dictionaryWithDictionary:params]; NSMutableArray* pairs = [NSMutableArray array]; //可以在这里将token参数添加进去,这样就可以实现补全token功能
for (NSString* key in param.keyEnumerator) {
NSString *value = [NSString stringWithFormat:@"%@",[param objectForKey:key]]; NSString* escaped_value = (__bridge_transfer NSString *)CFURLCreateStringByAddingPercentEscapes(NULL,
(__bridge CFStringRef)value,
NULL,
(CFStringRef)@"!*'\"();:@&=+$,/?%#[]% ",
kCFStringEncodingUTF8); [pairs addObject:[NSString stringWithFormat:@"%@=%@", key, escaped_value]];
} NSString *query = [pairs componentsJoinedByString:@"&"];
baseURL = [baseURL stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; NSString* url = @"";
if ([baseURL containsString:@"?"]) {
url = [NSString stringWithFormat:@"%@&%@",baseURL, query];
}
else {
url = [NSString stringWithFormat:@"%@?%@",baseURL, query];
}
//绝对地址
if ([url.lowercaseString hasPrefix:@"http"]) {
return [NSURL URLWithString:url];
}
else {
return [NSURL URLWithString:url relativeToURL:self.baseUrl];
}
}

XYWKWebViewController 核心功能

这是一个 Controller,建议创建新的Controller继承XYWKWebViewController 来使用,这样可以把不同的页面区分开,每个页面加载的url和相关的业务逻辑都可以单独处理,代码易读,也容易维护。如果项目后期添加功能也好处理
XYWKWebViewController主要完成了对一些功能的封装,比如进度条、页面title以及 webView 的生命周期。

进度条和title都是通过KVO实现

if (self.shouldShowProgress) {
[self.webView addObserver:self forKeyPath:@"estimatedProgress" options:NSKeyValueObservingOptionNew context:NULL];
} if (self.isUseWebPageTitle) {
[self.webView addObserver:self forKeyPath:@"title" options:NSKeyValueObservingOptionNew context:NULL];
}

设置title 和 progressView 直接是自己简单写了一个 View

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context {
if ([keyPath isEqualToString:@"estimatedProgress"]) { if (object == self.webView) {
[self showLoadingProgress:self.webView.estimatedProgress andTintColor:[UIColor colorWithRed:24/255.0 green:124/255.0 blue:244/255.0f alpha:1.0]];
}
else{
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
}
else if ([keyPath isEqualToString:@"title"]){
if (object == self.webView) {
if ([self isUseWebPageTitle]) {
self.title = self.webView.title;
}
}
else{
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
}
else {
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
}

OC 与 JS 之间交互的处理

这部分是可定制化功能最多的,遇到的问题也是最多的。WKWebView 和 JS 之间的交互需要设置 ScriptMessageHandler 如下。

- (instancetype)initWithFrame:(CGRect)frame configuration:(WKWebViewConfiguration *)configuration {
self = [super initWithFrame:frame configuration:configuration];
if (self) { if (configuration) {
//文档中说
//Adds a script message handler.
//Adding a script message handler with name name causes the JavaScript function window.webkit.messageHandlers.name.postMessage(messageBody) to be defined in all frames in all web views that use the user content controller. // 这里就是设置 网页中 JS Message handler
// 通过 “name” 注册之后,JS 内部函数 window.webkit.messageHandlers.“name”.postMessage(messageBody) 就被定义到整个用户的Web内容的控制器中。
//后面的JS调用OC也是通过 “name” 联系的
[configuration.userContentController addScriptMessageHandler:self name:@"webViewApp"];
} //默认允许系统自带的侧滑后退
self.allowsBackForwardNavigationGestures = YES;
}
return self;
}

然后实现 WKScriptMessageHandler 代理


// JS 调用 OC 的回调。JS 向OC 发送消息会调用这个方法
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message { NSLog(@"得到的 JS message 是 :%@",message.body);
if ([message.body isKindOfClass:[NSDictionary class]]) { NSDictionary *body = (NSDictionary *)message.body; // 这里是对 JS 消息的一个处理,用自己定义的消息类型,封装并发送给代理去外部处理,具体格式需要工作中和H5共同制定 XYScriptMessage *msg = [XYScriptMessage new];
[msg setValuesForKeysWithDictionary:body]; if (self.xy_messageHandlerDelegate && [self.xy_messageHandlerDelegate respondsToSelector:@selector(xy_webView:didReceiveScriptMessage:)]) {
[self.xy_messageHandlerDelegate xy_webView:self didReceiveScriptMessage:msg];
}
} }

其中自定义的 XYScriptMessage 如下

/**
* WKWebView与JS调用时参数规范实体
*/
@interface XYScriptMessage : NSObject /**
* 方法名
* 用来确定Native App的执行逻辑
*/
@property (nonatomic, copy) NSString *method; /**
* 方法参数
* json字符串
*/
@property (nonatomic, copy) NSDictionary *params; /**
* 回调函数名
* Native App执行完后回调的JS方法名
*/
@property (nonatomic, copy) NSString *callback; @end

同时提供delegate方法供XYWKWebViewController实现

/**
* JS调用原生方法处理,其中方法名都需要和 H5 端相互协调
*/
- (void)xy_webView:(XYWKWebView *)webView didReceiveScriptMessage:(XYScriptMessage *)message { NSLog(@"webView method:%@",message.method); //返回上一页
if ([message.method isEqualToString:@"tobackpage"]) {
[self.navigationController popViewControllerAnimated:YES];
}
//打开新页面
else if ([message.method isEqualToString:@"openappurl"]) { NSString *url = [message.params objectForKey:@"url"];
if (url.length) {
XYWKWebViewController *webViewController = [[XYWKWebViewController alloc] init];
webViewController.url = url; [self.navigationController pushViewController:webViewController animated:YES];
}
}
}

使用方法

一个提供四类使用功能,使用方法建议直接继承 XYWKWebViewController。

class WebViewController: XYWKWebViewController {

    override func viewDidLoad() {
super.viewDidLoad() /// #用法0: 直接加载对应的地址 <没有参数>
//self.webView.loadRequest(withRelativeUrl: "https://www.httpbin.org/") /// #用法1: 直接加载对应的地址 <有参数>
//let params = ["name":"xiaoyou",
// "password" : "123456#/HTTP_Methods/get_get"]
//self.webView.loadRequest(withRelativeUrl: "https://www.httpbin.org/", params: params) /// #用法2: 直接加载本地HTML文件 <没有参数>
self.webView.loadLocalHTML(withFileName: "main") /// #用法3: JS 注入,添加一些方法 <这里的原生坐标和JS之间无法直接相对应>
let margin : CGFloat = 6.0
let padding : CGFloat = 10.0
let width = UIScreen.main.bounds.size.width - (margin * 2.0) - (margin * 7.0 + padding)
let btnWidth = (width - padding - 5) / 2.0 let styleJS = """
<style type="text/css">
#foot {
border:solid 10px #600;
padding:\(padding)px;
margin:\(margin)px;
clear:both;
width:\(width)px
}
#share {
border:solid 1px #600;
padding:2px;
margin:2px;
clear:both;
width:\(btnWidth)px;
heiht:150px
}
#like {
border:solid 1px #600;
padding:2px;
margin:2px;
clear:both;
width:\(btnWidth)px;
heiht:50px
}
</style>
""" let funcJS = """
\t\t\tfunction testFunc(text){\n
\t\t\t\tvar message = \"点我干什么\";\n
\t\t\t\twindow.webkit.messageHandlers.webViewApp.postMessage(message);\n
\t\t\t\talert(text);\n
\t\t\t}\n
""" let footerJS = """
\t<button onclick=\"testFunc('http://www.baidu.com/')\">自己添加的Footer的Button一个</button><br /><br /><br />\n
\t <div id=\"foot\">底部说明 <br />
<button id=\"share\" onclick=\"testFunc('分享')\">分享</button>
<button id=\"like\" onclick=\"testFunc('点赞')\">点赞</button><br />
</div>
"""
self.webView.loadLocalHTML("main", withAddingStyleJS: styleJS, funcJS: funcJS, footerJS: footerJS) /// 设置导航
self.navigationItem.leftBarButtonItem = UIBarButtonItem(title: "返回", style: .plain, target: self, action: #selector(backAction));
self.navigationItem.rightBarButtonItem = UIBarButtonItem(title: "调用JS", style: .plain, target: self, action: #selector(callJS));
}
} /// #用法4: OC 调用JS方法。这里可以调用JS,把H5需要的参数传给他们
/// 这里是JS 回调方法
extension WebViewController{ @objc func backAction() {
self.dismiss(animated: true, completion: nil)
} @objc func callJS() {
self.webView.callJS("call('Hello World!')") { (response) in
print("\(String(describing: response))")
}
} /// 这里是重写了WebView接受到JS消息的回调,需要调用super方法才能执行内部方法,否则这里只是打印
override func xy_webView(_ webView: XYWKWebView, didReceive message: XYScriptMessage) { // 如果完全自定义的js方法处理,无需重写父类,自行实现即可
super.xy_webView(webView, didReceive: message)
print(message)
} }

具体见Demo

遇到的问题

HTML 中超链接,需要打开新页面的"_blank"处理

小结

WebKit 的小封装能实现目前所需功能,但很多内容还需要在需要的时候去探究,希望能帮到同样学习的小伙伴。

如果看完有收获,不妨点个赞,让我也更有分享的动力!

header{font-size:1em;padding-top:1.5em;padding-bottom:1.5em}
.markdown-body{overflow:hidden}
.markdown-body>div,.markdown-body>article{width:100%}
aside.sidebar{float:none;padding:0 18px 1px;background-color:#f7f7f7;border-top:1px solid #e0e0e0}
.flex-content,article img,article video,article .flash-video,aside.sidebar img{max-width:100%;height:auto}
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left{float:left;margin-right:1.5em}
.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right{float:right;margin-left:1.5em}
.basic-alignment.center,article img.center,article video.center,article .center.flash-video,aside.sidebar img.center{display:block;margin:0 auto 1.5em}
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left,.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right{margin-bottom:.8em}
.toggle-sidebar,.no-sidebar .toggle-sidebar{display:none}

.markdown-body img,.markdown-body video,.markdown-body .flash-video{ -webkit-border-radius:0.3em;-moz-border-radius:0.3em;-ms-border-radius:0.3em;-o-border-radius:0.3em;border-radius:0.3em;-webkit-box-shadow:rgba(0,0,0,0.15) 0 1px 4px;-moz-box-shadow:rgba(0,0,0,0.15) 0 1px 4px;box-shadow:rgba(0,0,0,0.15) 0 1px 4px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;border:#fff 0.5em solid;}

.markdown-body img,.markdown-body video{max-width: 100%;}
.markdown-body video,.markdown-body .flash-video{margin:0 auto 1.5em}
.markdown-body video{display:block;width:100%}
.markdown-body .flash-video>div{position:relative;display:block;padding-bottom:56.25%;padding-top:1px;height:0;overflow:hidden}
.markdown-body .flash-video>div iframe,.markdown-body .flash-video>div object,.markdown-body .flash-video>div embed{position:absolute;top:0;left:0;width:100%;height:100%}
.markdown-body>footer{padding-bottom:2.5em;margin-top:2em}
.markdown-body>footer p.meta{margin-bottom:.8em;font-size:.85em;clear:both;overflow:hidden}

body,pre{ background:#fdf6e3 url('') top left;}
body{ background-color: #f8f8f8;}
pre{-webkit-border-radius:0.4em;-moz-border-radius:0.4em;-ms-border-radius:0.4em;-o-border-radius:0.4em;border-radius:0.4em;border:1px solid #e7dec3;line-height:1.45em;font-size:13px;margin-bottom:2.1em;padding:.8em 1em;color:#586e75;overflow:auto}
.markdown-body code{background: none;}
h3.filename+pre{-moz-border-radius-topleft:0px;-webkit-border-top-left-radius:0px;border-top-left-radius:0px;-moz-border-radius-topright:0px;-webkit-border-top-right-radius:0px;border-top-right-radius:0px}
p code,li code{display:inline-block;white-space:no-wrap;background:#fff;font-size:.8em;line-height:1.5em;color:#555;border:1px solid #ddd;-webkit-border-radius:0.4em;-moz-border-radius:0.4em;-ms-border-radius:0.4em;-o-border-radius:0.4em;border-radius:0.4em;padding:0 .3em;margin:-1px 0}
p pre code,li pre code{font-size:1em !important;background:none;border:none}

/*

Orginal Style from ethanschoonover.com/solarized (c) Jeremy Hull

*/

.hljs {
display: block;
overflow-x: auto;
padding: 0.5em;
background: #fdf6e3;
color: #657b83;
-webkit-text-size-adjust: none;
}

.hljs-comment,
.diff .hljs-header,
.hljs-doctype,
.hljs-pi,
.lisp .hljs-string {
color: #93a1a1;
}

/* Solarized Green */
.hljs-keyword,
.hljs-winutils,
.method,
.hljs-addition,
.css .hljs-tag,
.hljs-request,
.hljs-status,
.nginx .hljs-title {
color: #859900;
}

/* Solarized Cyan */
.hljs-number,
.hljs-command,
.hljs-string,
.hljs-tag .hljs-value,
.hljs-rule .hljs-value,
.hljs-doctag,
.tex .hljs-formula,
.hljs-regexp,
.hljs-hexcolor,
.hljs-link_url {
color: #2aa198;
}

/* Solarized Blue */
.hljs-title,
.hljs-localvars,
.hljs-chunk,
.hljs-decorator,
.hljs-built_in,
.hljs-identifier,
.vhdl .hljs-literal,
.hljs-id,
.css .hljs-function,
.hljs-name {
color: #268bd2;
}

/* Solarized Yellow */
.hljs-attribute,
.hljs-variable,
.lisp .hljs-body,
.smalltalk .hljs-number,
.hljs-constant,
.hljs-class .hljs-title,
.hljs-parent,
.hljs-type,
.hljs-link_reference {
color: #b58900;
}

/* Solarized Orange */
.hljs-preprocessor,
.hljs-preprocessor .hljs-keyword,
.hljs-pragma,
.hljs-shebang,
.hljs-symbol,
.hljs-symbol .hljs-string,
.diff .hljs-change,
.hljs-special,
.hljs-attr_selector,
.hljs-subst,
.hljs-cdata,
.css .hljs-pseudo,
.hljs-header {
color: #cb4b16;
}

/* Solarized Red */
.hljs-deletion,
.hljs-important {
color: #dc322f;
}

/* Solarized Violet */
.hljs-link_label {
color: #6c71c4;
}

.tex .hljs-formula {
background: #eee8d5;
}
-->

WKWebView 的使用和封装的更多相关文章

  1. [iOS 开发] WebViewJavascriptBridge 从原理到实战 · Shannon's Blog

    前言:iOS 开发中,h5 和原生实现通信有多种方式, JSBridge 就是最常用的一种,各 JSBridge 类库的实现原理大同小异,这篇文章主要是针对当前使用最为广泛的 WebViewJavas ...

  2. iOS网络3—UIWebView与WKWebView使用详解

    一.整体介绍 UIWebView自iOS2就有,WKWebView从iOS8才有,毫无疑问WKWebView将逐步取代笨重的UIWebView.通过简单的测试即可发现UIWebView占用过多内存,且 ...

  3. IOS进阶之WKWebView

    前言 Xcode8发布以后,编译器开始不支持IOS7,所以很多应用在适配IOS10之后都不在适配IOS7了,其中包括了很多大公司,网易新闻,滴滴出行等.因此,我们公司的应用也打算淘汰IOS7. 支持到 ...

  4. WKWebView与JS交互,UIWebView+JavascriptCore和JS交互

    最近一直在做有关Swift和JavaScript交互的程序,所以有关UIWebView和WKWebView在使用上的差别在此总结下: UIWebView: (1)创建 var webView: UIW ...

  5. iOS 开发之路(使用WKWebView加载Html5) 四

    基于Swift 3 . Xcode 8 . iOS 10 下的WKWebView的使用. 首先是WKWebView的基本用法: var wk:WKWebView! var progBar:UIProg ...

  6. 使用WKWebView替换UIWebView

    开发App的过程中,常常会遇到在App内部加载网页,通常用UIWebView加载.这个自iOS2开始使用的网页加载器一直是开发的心病:加载速度慢,占用内存多,优化困难.如果加载网页多,还可能因为过量占 ...

  7. 使用iOS8 WKWebView的浏览器模块,脉冲动画层-b

    KINWebBrowser是一个可嵌入app的浏览器模块. 它使用iOS 8的 WKWebView API编写,同时在iOS 7上使用UIWebView来兼容. 测试环境: Xcode 6.0 iOS ...

  8. iOS开发之Xcode8推出的WKWebView与UIWebView的使用

    一.整体介绍 UIWebView自iOS2就有,WKWebView从iOS8才有,毫无疑问WKWebView将逐步取代笨重的UIWebView.通过简单的测试即可发现UIWebView占用过多内存,且 ...

  9. iOS开发之WKWebView代替UIWebView

    前言 Xcode8发布以后,编译器开始不支持IOS7,所以很多应用在适配IOS10之后都不在适配IOS7了,其中包括了很多大公司,网易新闻,滴滴出行等.因此,我们公司的应用也打算淘汰IOS7. 支持到 ...

随机推荐

  1. linux 动态静态库

    库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行.库分静态库和动态库两种.  1  静态库和动态库的区别1.1. 静态函数库    (1)静态函数库的名字一般是lib[name].a( ...

  2. 从javascript的循环问题来看待闭包本质

    第一次接触这个问题还是在我刚开始学js的时候,当时就是一头雾水,时隔一年多了,突然又想起了这个问题,在这个春气盎然的周末,我就坐下来研究下并把结果和大家分享下: 先看代码:demo.html < ...

  3. 使用JAVA实现模拟登陆并发送新浪微博(非调用新浪API)

    没有调用新浪的API,在程序中加入自己的帐号和密码就能发送微博,代码完全在后台运行,不用打开浏览器. 用了HtmlUnit这个库来模拟登录还有发送微博. 先上效果图: 这个是刚登陆上获取第一页的信息. ...

  4. Java发送邮件Utils

    /** * 类文件说明 * */ public class SendMail { Logger log = Logger.getLogger(SendMail.class); /** * 发送邮件 * ...

  5. Catch That Cow(bfs)

    Description Farmer John has been informed of the location of a fugitive cow and wants to catch her i ...

  6. Leetcode:Substring with Concatenation of All Words分析和实现

    题目大意是传入一个字符串s和一个字符串数组words,其中words中的所有字符串均等长.要在s中找所有的索引index,使得以s[index]为起始字符的长为words中字符串总长的s的子串是由wo ...

  7. 使用Fuel安装openstack

    一.前言 Fuel是OpenStack的开源部署和管理工具.作为OpenStack社区的开发贡献者,它为OpenStack.OpenStack相关社区项目以及OpenStack插件的部署和管理提供了直 ...

  8. mfs教程(三)

    mfs文件系统(三) 使用  MooseFS 一.挂载文件系统 启动管理服务器(master server)和数据服务器(chunkservers) (chunkservers一个是必需的,但至少两个 ...

  9. Ros学习——移动机器人Ros导航详解及源码解析

    1 执行过程 1.运行仿真机器人fake_turtlebot.launch:加载机器人模型——启动机器人仿真器——发布机器人状态 2.运行amcl节点fake_amcl.launch:加载地图节点ma ...

  10. 【转】request的cache-control和response cache-control不同点

    原文地址:http://www.cnblogs.com/lwhkdash/archive/2012/11/04/2748291.html HTTP协议中,关于一些头域的解释很模糊,网上的解释有些甚至是 ...