- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个图像的 UICollectionView 和一个在选择任何图像时打开的完整 ImageView 。打开完整图像效果很好,但一旦我关闭它,整个应用程序就会卡住超过 5 秒。我读到主线程可能过载,这可以解释延迟,但我不确定要移动到哪些线程以及移动到哪里,如果这是问题的话。
import UIKit
class FullImageViewController: UIViewController , UIScrollViewDelegate {
var scroller: UIScrollView!
var width = CGFloat()
var images = [Image]()
var imageArray = [UIImage]()
var imgs: [UIImage] = [UIImage(named: "pug7")!, UIImage(named: "pug6")!, UIImage(named: "pug7")!, UIImage(named: "pug6")!]
var index = Int()
var fullImagesAdded = Bool()
init(width: CGFloat, images: [Image], index: Int, imageArray: [UIImage]) {
super.init(nibName: nil, bundle: nil)
self.width = width
self.images = images
self.index = index
self.imageArray = imageArray
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func viewDidLoad() {
super.viewDidLoad()
let downSwipe = UISwipeGestureRecognizer(target: self, action: #selector(FullImageViewController.swipeDown))
downSwipe.direction = .Down
view.addGestureRecognizer(downSwipe)
self.scroller = UIScrollView()
self.scroller.delegate = self
var i = 0
for img in images {
let aimageDetailViewController = ImageDetailViewController(index: self.index, image: img)
self.scroller.addSubview(aimageDetailViewController.view)
var aimageDetailViewFrame: CGRect = aimageDetailViewController.view.frame
aimageDetailViewFrame.origin.x = (super.view.frame.width) * CGFloat(i)
aimageDetailViewController.view.frame = aimageDetailViewFrame
i += 1
}
self.scroller.backgroundColor = UIColor.blackColor()
self.scroller.contentSize = CGSizeMake(self.view.frame.width * CGFloat(images.count), self.view.frame.height)
self.scroller.bounces = false
self.scroller.showsHorizontalScrollIndicator = false
self.scroller.setContentOffset(CGPoint(x: self.view.frame.width * CGFloat(index), y: 0), animated: false)
self.scroller.pagingEnabled = true
self.view.addSubview(scroller)
}
func swipeDown(){
dispatch_async(dispatch_get_main_queue()) {
self.dismissViewControllerAnimated(true, completion: nil)
}
}
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
scroller.frame = view.bounds
}
}
这是我在 UICollectionView 中调用完整 ImageView 的地方:
func collectionView(collectionView: UICollectionView, didSelectItemAtIndexPath indexPath: NSIndexPath) {
let fullImage = FullImageViewController(width: CGFloat(imageURLSCache.count), images: imageURLSCache, index: Int(indexPath.row), imageArray: images)
self.presentViewController(fullImage, animated: true, completion: nil)
}
谢谢!
这是我用来下载完整 ImageView 和 Collection View 的图像的扩展
extension UIImageView {
func downloadFrom(link link:String?, cache: NSCache ) {
image = UIImage(named: "logo")
self.alpha = 0.3
let data: NSData? = cache.objectForKey(link!) as? NSData
if let cachedData = data {
let image = UIImage(data: cachedData)
dispatch_async(dispatch_get_main_queue(), {() in
self.image = image
self.alpha = 1
images.append(self.image!)
})
return
}
if link != nil, let url = NSURL(string: link!) {
NSURLSession.sharedSession().dataTaskWithURL(url) { data, response, error in
guard let data = data where error == nil else {
print("\nerror on download \(error)")
return
}
if let httpResponse = response as? NSHTTPURLResponse where httpResponse.statusCode != 200 {
print("statusCode != 200; \(httpResponse.statusCode)")
return
}
dispatch_async(dispatch_get_main_queue()) {
print("\ndownload completed \(url.lastPathComponent!)")
self.image = UIImage(data: data)
self.alpha = 1
cache.setObject(data, forKey: link!)
}
}.resume()
} else {
dispatch_async(dispatch_get_main_queue(), {
self.image = UIImage(named: "logo")
})
}
}
}
最佳答案
func swipeDown(){
dispatch_async(dispatch_get_main_queue()) {
self.dismissViewControllerAnimated(true, completion: nil)
}
}
此方法已设置为 UISwipeGestureRecognizer
的操作。它将由手势识别器在主线程上调用,因此您不需要将其分派(dispatch)到主队列。这会增加延迟,具体取决于队列的繁忙程度。
将其更改为:
func swipeDown() {
self.dismissViewControllerAnimated(true, completion: nil)
}
有关您添加的附加信息的更新:
UIImageView
上的扩展不是很好(即,当使用它作为缓存的键时,它假设 link
不是 nil
,但稍后检查它是否为 nil
,但不可能是这样,否则你已经崩溃了)。但它至少使用 dataTaskWithURL
方法在后台进行下载。
我会尝试在调度到后台调用中调用它,如下所示,以确保您不会因缓存检查等而使主线程重载。我假设您正在安全地为此扩展创建缓存,并且当然会重用所有调用的缓存,而不是为每个调用创建一个新的缓存(我已经看到这样做了)。
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)) {
imageView.downloadFrom(....)
}
这是一个可供使用的替代扩展。这是我经常使用的一个。它会为您创建缓存,因此您不必担心在扩展之外执行此操作。它还具有确认自图像开始下载以来尚未在图像上设置另一个 URL 的好处。例如,当 ImageView 位于可重复使用的单元上时,这一点很重要。创建单元格,将图像 url 设置为 x,当它仍在下载时,单元格滚动到屏幕外并重新使用,图像 url 设置为 y,然后 x 下载完成并显示适用于旧单元格的图像。
代码如下:
// UIImageView+XCGAdditions.swift
// Cerebral Gardens
//
// Created by Dave Wood
// Copyright © Cerebral Gardens. All rights reserved.
import UIKit
extension UIImageView {
func setImageWithLink(link: String) {
guard let url = NSURL(string: link) else { return }
setImageWithURL(url: url)
}
func setImageWithURL(url url: NSURL) {
let checkInt = Int(arc4random_uniform(100000))
tag = checkInt
struct Statics {
static var onceToken: dispatch_once_t = 0
static var imageCache: NSCache!
}
dispatch_once(&Statics.onceToken) { () -> Void in
Statics.imageCache = NSCache()
}
if let cachedImage = Statics.imageCache.objectForKey(url) as? UIImage {
self.image = cachedImage
return
}
NSURLSession.sharedSession().dataTaskWithURL(url, completionHandler: { (data: NSData?, response: NSURLResponse?, error: NSError?) -> () in
guard let data = data where error == nil,
let image = UIImage(data: data) where self.tag == checkInt else { return }
dispatch_async(dispatch_get_main_queue()) { () -> () in
Statics.imageCache.setObject(image, forKey: url)
self.image = image
}
}).resume()
}
}
关于swift - 关闭 UIViewController 后 UI 卡住 5 秒,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36815317/
我正在我的 xamarin.forms 应用程序中实现扫描仪功能,为此我正在使用 iOS native AVCaptureSession。但我的问题是在扫描或捕获 session 处于事件状态并且设备
所以我目前正在为我的项目制作一个音乐应用程序,它允许用户创建自己的音乐播放列表。但是,当我单击显示媒体选择器按钮时,它只显示白屏,当包含媒体选择器的 View 是 Initial View Contr
当我尝试在模拟器中启动 AVD 时,会出现一个小窗口(见图片),5 秒后它说没有响应并一直保持这种状态直到我关闭它。 我在网上搜索并尝试了所有解决方案,但都没有成功 在 BIOS 中启用了虚拟化 已安
尝试使用以下命令从视频中提取特定帧(删除了文件的特定名称!: ffmpeg -i video.mp4 -vf "select-gte(n\,6956)"-vframes 10262 文件夹/帧%d.j
我怎么知道终端正在继续工作而不中断它? 我已经运行了以下 git 命令: clone git://ligo-vcs.phys.uwm.edu/lalsuite.gituote 一段时间后它似乎被卡住了
我对 WPF 中的数据网格有一个奇怪的问题。我正在为我的应用程序使用 MVVM 模式,并且我的 View 模型实现了 idataerrorinfo 接口(interface)。每当我在添加新行后在我的
我有这个 Excel 文件,当我输入数据时它卡住了。例如,我双击一个单元格,输入数据,然后按“输入”。它会卡住而不是进入下面的细胞。按几次“enter”不会解冻程序,唯一有效的是用鼠标选择另一个单元格
我有线程池的任务队列,每个任务都有卡住锁定其正在使用的所有资源的倾向。并且除非重新启动服务,否则这些无法释放。 ThreadPool 中有没有办法知道它的线程已经被卡住?我有一个使用超时的想法(虽然我
我制作了以下小程序来确定内存是否用于 freeze(X,Goal) 之类的目标回收时 X变得无法访问: %:- use_module(library(freeze)). % Ciao Prolog n
我有一个使用 swing 的简单 java 应用程序。然而,当我执行程序时,框架将会出现,但我无法单击任何地方,并且按钮仅在几秒钟后出现。我对 javas Swing 库非常陌生,所以我可能会丢失一些
我正在尝试创建一个简单的 TCP 客户端服务器应用程序接口(interface)用户可以在按下相应按钮时启动或停止服务器我创建了一个 StartServer 按钮,当用户按下按钮时它应该连接到服务
我正在尝试从 ftp 服务器下载文件,但在检索文件时卡住了。我正在使用 commons-net-3.6.jar 我注意到的事情 当我使用 ftpClient.enterRemotePassiveMod
我正在尝试编写一个函数,该函数将能够找到位于我系统上的可执行文件搜索路径中的任意可执行文件。我遇到了一些输入会导致 SearchPathW 的问题无限期地卡住,我不确定到底发生了什么。 std::op
我的 Nativescript 应用程序的许多页面中都有 RadSideDrawer。主应用程序组件有一个 page-router-outlet并且所有其他页面都通过导航加载到此组件中。带抽屉的页面包
我有一个最小的服务器,它等待客户端连接,然后他启动一个线程,将回复发送回客户端,问题是回复。 这是服务器的代码: int port = 1234; ServerSocket servSock =
我有一个使用 C# 的 WinForms 应用程序。我尝试从文件中读取一些数据并将其插入到数据表中。虽然此操作很忙,但我的表单卡住并且我无法移动它。有谁知道我该如何解决这个问题? 最佳答案 这可能是因
在我们学校最新的项目中,我遇到了一些问题。我想观察新条目的路径,该路径是由文件导向器按钮选择的,但如果我选择任何文件,整个窗口都会卡住...我猜它被卡住,因为调用了“observePath”方法,但我
当我输入一百万作为输入数字时,我的程序卡住了。我该如何解决这个问题? 我尝试将第二个 for 循环分离为第二个函数,但没有成功。 import java.io.*; public class Arra
早上好编译我的应用程序时,我在 Android Studio 上遇到问题。我在构建时没有收到关于 app:transformClassesWithDexBuilderForDebug 的任何输出错误,
我正在使用以下触发器 DELIMITER ; CREATE TRIGGER updateCount AFTER INSERT ON user_info FOR EACH ROW BEGIN UPDA
我是一名优秀的程序员,十分优秀!