# iOS9 URL Scheme
---
在最近iOS9带来的更新中,有一项关于URL Scheme的变化,具体内容是:在iOS9的SDK中,若要通过URL Scheme访问其他APP,则需要事先将该URL加入程序的白名单中。
##影响
这个URL Scheme机制的改变主要影响了两个api:canOpenURL和openURL。在iOS9以前,如果要判断系统中有没有APP能够处理相应的URL,直接调用canOpenURL就可以了,openURL同理。而在iOS9之后,若要访问某个URL,则必须在程序的info.plist中把访问的URL提前设置好,将其加入白名单中,才能够顺利使用canOpenURL和openURL,否则,系统会拒绝访问,直接返回NO。
##如何设置
要设置访问APP的URL Scheme,需要在info.plist中添加一个字段:`LSApplicationQueriesSchemes`,其值为由String组成的Array类型,每一个String代表一个URL Scheme。
##系统Log提示
当URL Scheme没有设置成功时,使用canOpenURL或者openURL时,系统会打出Log:
`canOpenURL: failed for URL: "urlscheme://" - error: "This app is not allowed to query for scheme urlscheme"`
而当URL设置成功,但没有找到相应的处理APP时,系统的Log是这样的:`canOpenURL: failed for URL: "urlscheme://" - error: "(null)"`
当URL Scheme没有正常工作时,可以通过系统的Log定位原因。
##一次坑爹的经历
最近在项目中就出现了这样的需求,需要通过检测设备是否安装了某个应用,来决定是否打开某个功能点,当时直接使用了canOpenURL来检测,结果理所应当的在iOS9上出现了问题,系统提示如下:
canOpenURL: failed for URL: "urlscheme://" - error: "This app is not allowed to query for scheme url scheme"
于是通过谷歌来的解决方案在info.plist中添加了`LSApplicationQueriesSchemes`这个键,但是,canOpenURL依然返回NO,why?我百思不得其解,只能继续各种查资料,看文档,发现连苹果官方提供的解决方案也就只是这样。
后来也不知在哪里看来的资料,说是要在CFBundleURLType里加上指定的URL Scheme,我看这URL Scheme可能是一回事,于是就试着加了上去,发现果然可以,就开心的觉得自己解决了问题。
直到后来测试时发现,`canOpenURL`永远返回`YES`了,再仔细看下`CFBundleURLType`代表的意思,才发现自己蠢透了,这两个东西根本就不是一回事,`CFBundleURLType`指定App能够处理的URL Scheme,而`LSApplicationQueriesSchemes`则指定App可能要访问的URL Scheme白名单,两者的方向根本是相反的,而我之前加的`CFBundleURLType`,是在告诉系统我的App能处理该URL Scheme,自己发出的URL请求自己消化了,所以调用`canOpenURL`才返回了`YES`,这根本就是骗自己嘛=。=
那为何我已经在info.plist中加了白名单字段,canOpenURL还是返回NO呢,反复几次测试后,我才从系统Log中发现一些端倪,第二次系统Log的error信息为null,而WWDC是这样说的:
![iOS9URLScheme](http://img.zoww.net/iOS9URLScheme.png)
好嘛,null是说指定的App没有装的意思,于是看了眼设备,发现下错了iPad版。。(╯‵□′)╯︵┻━┻
整个过程到此水落石出了,实际上适配iOS9的URL Scheme就只是添加一个字段这么简单,但我却折腾了蛮长的时间,还差点搞了一个bug上线,真是不应该。说这个排bug经历坑爹,事实上是我自己坑爹了,如果我对CFBundleURLTypes有过认识,也就不会多出后面的事了。这次经历暴露出的许多问题:
> * 对info.plist的常用字段没有认识
> * 出现问题时,没有仔细观察问题的表现,错过了排查问题的好时机
> * 对解决问题盲目乐观,没有测试完全
就这样吧,下一篇写写info.plist。