我在使用 NSURLSession 框架时有一些误解,这就是为什么我决定在没有 AFFramework/Alamofire 的情况下从头开始编写小应用程序。
我有一个 API 需要执行以下步骤来上传文件:
api/save我有一个这样配置的后台 session :
let configuration = NSURLSessionConfiguration.backgroundSessionConfigurationWithIdentifier("myBackground")
let session = NSURLSession(configuration: configuration, delegate: self, delegateQueue: nil)
我实现了两种方法:
func URLSession(session: NSURLSession, dataTask: NSURLSessionDataTask, didReceiveData data: NSData)
我在哪里汇总所有数据
和
func URLSession(session: NSURLSession, task: NSURLSessionTask, didCompleteWithError error: NSError?)
我在哪里将此数据转换为响应对象。如果这个响应对象对我来说非常重要。
一切正常,虽然应用程序在前台,但我在后台遇到问题。
案例一
应用程序在我开始上传数据后立即崩溃。根据 WWDC,我需要实现
func application(application: UIApplication, handleEventsForBackgroundURLSession identifier: String, completionHandler: () -> Void)
并在 didCompleteWithError 方法中调用此处理程序。但在调用此方法之前,我需要使用上传响应中的数据调用 api/save。
我怎样才能得到这些数据?
案例二
大部分情况相似。用户在上传过程中停止应用程序。比在几秒钟内加载应用程序,而 session 与我的任务一起工作。现在 session 调用didReceiveData,当然,一些数据丢失了。遇到这种情况怎么办?如何恢复响应数据?
最佳答案
您没有提到实现 URLSessionDidFinishEventsForBackgroundURLSession(NSURLSessionDelegate 方法)。你也真的想实现它。基本流程是:
在应用委托(delegate)的 handleEventsForBackgroundURLSession 中,您应该:
NSURLSession(它将开始接收与所有上传相关的委托(delegate)方法调用);和然后,在 URLSessionDidFinishEventsForBackgroundURLSession 中(处理完所有响应后),调用保存在 handleEventsForBackgroundURLSession 中的完成处理程序。 (确保将其分派(dispatch)到主队列。)
如果您正在执行所有这些操作,当后台 session 重新启动时,didReceiveData 调用将随对您的各种上传的响应一起进来。
我刚刚做了一个快速测试,上传了五张 20mb 的图片并立即终止了应用程序。然后,即使应用程序没有运行,我也看到这五个文件慢慢出现在我的服务器上(显然由守护进程处理)。当所有五个都完成后,应用程序在后台透明地重新启动,调用了 handleEventsForBackgroundURLSession(重新启动了 session ),它让所有的 didReceiveData 调用快速被调用,完成后,将调用 URLSessionDidFinishEventsForBackgroundURLSession,然后我的应用才调用已保存的完成处理程序。
就为什么这对您不起作用而言,还不足以诊断问题。可能性包括:
也许您不恰本地终止了该应用程序。您不能通过双击主页按钮并在那里终止应用程序来终止应用程序;您必须让它自己自然终止,或者出于诊断/测试目的,我通过在代码中调用 exit(0) 强制它终止。
可能在调用 handleEventsForBackgroundURLSession 时您没有重新启动 session 。
也许您调用提供的完成处理程序的时间过早(即在调用 URLSessionDidFinishEventsForBackgroundURLSession 之前)。
这很难说,但我怀疑您的实现中隐藏了一些不太正确的东西,并且很难根据提供的信息说出它是什么(假设它不是上述几点之一) ).不幸的是,调试这个后台 session 非常复杂,因为当应用程序终止时,它不再附加到调试器,所以你无法轻松调试应用程序由 iOS 自动重启后发生的事情。就个人而言,我要么使用 NSLog 消息并只观察设备控制台(以及观察服务器上显示的内容),要么在应用程序本身中构建一些持久的日志记录机制。
关于ios - NSURLSessionUploadTask 获取响应数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34760265/
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
有没有办法在这个简单的get方法中添加超时选项?我正在使用法拉第3.3。Faraday.get(url)四处寻找,我只能先发起连接后应用超时选项,然后应用超时选项。或者有什么简单的方法?这就是我现在正在做的:conn=Faraday.newresponse=conn.getdo|req|req.urlurlreq.options.timeout=2#2secondsend 最佳答案 试试这个:conn=Faraday.newdo|conn|conn.options.timeout=20endresponse=conn.get(url
这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下
我是Google云的新手,我正在尝试对其进行首次部署。我的第一个部署是RubyonRails项目。我基本上是在关注thisguideinthegoogleclouddocumentation.唯一的区别是我使用的是我自己的项目,而不是他们提供的“helloworld”项目。这是我的app.yaml文件runtime:customvm:trueentrypoint:bundleexecrackup-p8080-Eproductionconfig.ruresources:cpu:0.5memory_gb:1.3disk_size_gb:10当我转到我的项目目录并运行gcloudprevie
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳
我有一个存储主机名的Ruby数组server_names。如果我打印出来,它看起来像这样:["hostname.abc.com","hostname2.abc.com","hostname3.abc.com"]相当标准。我想要做的是获取这些服务器的IP(可能将它们存储在另一个变量中)。看起来IPSocket类可以做到这一点,但我不确定如何使用IPSocket类遍历它。如果它只是尝试像这样打印出IP:server_names.eachdo|name|IPSocket::getaddress(name)pnameend它提示我没有提供服务器名称。这是语法问题还是我没有正确使用类?输出:ge
我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c
我安装了ruby版本管理器,并将RVM安装的ruby实现设置为默认值,这样'哪个ruby'显示'~/.rvm/ruby-1.8.6-p383/bin/ruby'但是当我在emacs中打开inf-ruby缓冲区时,它使用安装在/usr/bin中的ruby。有没有办法让emacs像shell一样尊重ruby的路径?谢谢! 最佳答案 我创建了一个emacs扩展来将rvm集成到emacs中。如果您有兴趣,可以在这里获取:http://github.com/senny/rvm.el
我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的
假设我有这个范围:("aaaaa".."zzzzz")如何在不事先/每次生成整个项目的情况下从范围中获取第N个项目? 最佳答案 一种快速简便的方法:("aaaaa".."zzzzz").first(42).last#==>"aaabp"如果出于某种原因你不得不一遍又一遍地这样做,或者如果你需要避免为前N个元素构建中间数组,你可以这样写:moduleEnumerabledefskip(n)returnto_enum:skip,nunlessblock_given?each_with_indexdo|item,index|yieldit