我的团队最近推出了一个应用程序,该应用程序出现了很多 SIGTRAP 崩溃。以前我发现这些修复起来相对简单,因为它是一个糟糕的强制转换或在有问题的函数中将隐式解包的可选设置为 nil 的问题。这次虽然我找不到任何类似的东西。我最好的猜测是,由于日历错误,可能 TimeBlock 对象之一或其属性为 nil。
我们的应用程序是一个 session 组织者,可以根据空闲时间、冲突和 session TimeBlocks 向用户显示他们的 native iOS 日历事件。我可以访问几个崩溃的用户的日历。
Apple SigTrap 定义
Swift code will terminate with this exception type if an unexpected condition is encountered at runtime such as:
- a non-optional type with a nil value
- a failed forced type conversion
崩溃函数
/**
Updates the conflictHours and meetingHours according to the timeblocks
it is used as quick light reference to the button
*/
func updateTimeHours(timeblocks : [Timeblock]) {
for timeblock in timeblocks {
switch timeblock {
case is MeetingTimeblock:
for i in timeblock.startHour...timeblock.endHour {
self.meetingHours[i] = true
}
break
case is ConflictTimeblock:
for i in timeblock.startHour...timeblock.endHour {
self.conflictsHours[i] = true
}
break
default: break
}
}
updateButtonByOffset(offset: self.scrollTimeline.contentOffset.x)
}
崩溃函数的调用
/**
This function inits the variables and button layout according to the timeblocks
*/
func handleTimeblocksDependantComponents() {
buttonLayout()
guard Scheduler.sharedInstance.timelines.count > SharedGlobals.Calendar.TODAY_INDEX else {
return
}
updateTimeHours(timeblocks : (Scheduler.sharedInstance.timelines[SharedGlobals.Calendar.TODAY_INDEX].timeblocks))
}
设置页眉高度
/**
Adjusts the height of the header depending on whether there are hosted meetings or
meeting VIP's or not.
*/
private func setHeaderHeight() {
self.tableView.tableHeaderView = self.headerView
let hostedMeetings = OverviewInteractor.getHostedMeetings(dayIndex: SharedGlobals.Calendar.SELECTED_DAY)
let vips = OverviewInteractor.getVIPS(dayIndex: SharedGlobals.Calendar.SELECTED_DAY)
self.tableView.beginUpdates()
if let headerView = self.tableView.tableHeaderView {
var height = 360.0
if(vips.count == 0) { height -= 80.0 }
if(hostedMeetings.count == 0) { height -= 80.0 }
headerView.frame.size.height = CGFloat(height)
}
self.tableView.endUpdates()
}
时间 block 定义
// The Timeblock parent class. It simply holds a start and end time and provides its own duration. Not to be used as such
public class Timeblock {
public let startTime: Date
public let endTime: Date
/// Returns the hour the Timeblock starts, in current timezone
public var startHour: Int {
get {
return Calendar.current.component(.hour, from: startTime)
}
}
/// Returns the hour the Timeblocks ends, in current timezone
public var endHour: Int {
get {
return Calendar.current.component(.hour, from: endTime)
}
}
/// Returns the minutes the Timeblocks starts
public var startMinutes: Int {
get {
return Calendar.current.component(.minute, from: startTime)
}
}
/// Returns the minutes the Timeblocks ends
public var endMinutes: Int {
get {
return Calendar.current.component(.minute, from: endTime)
}
}
/**
Initialises the instance with a start and end time
- Parameters:
- startTime: The start time of the timeblock
- endTime: The end time of the timeblock
*/
public init(startTime: Date, endTime: Date) {
self.startTime = startTime
self.endTime = endTime
}
/**
Provides the Timeblock's duration in the form of a DateInterval
- warning: Only available on iOS 10.0 and up
- returns: A DateInterval of the duration of the Timeblock
*/
@available(iOS 10.0, *)
public func getTimeInterval() -> DateInterval {
return DateInterval(start: self.startTime, end: self.endTime)
}
/**
Provides the Timeblock's duration in the form of a `Double` (number of seconds)
- returns: The number of seconds that this Timeblock goes on for
*/
public func getDuration() -> Double {
return self.endTime.timeIntervalSince(self.startTime)
}
}
崩溃报告
Incident Identifier: 98D4F477-C57B-4767-B957-E9EA2E0EE3EA CrashReporter Key: 0000000000000000000000000000000000000000 Hardware Model: undefined Process: xxxxxxx [784] Identifier:
com.xxx.xxx.xx.xxxxxxx Version: 4.0.3 Code Type:
arm64Date/Time: Sun Dec 24 2017 09:55:23 GMT+0000 (GMT) Launch Time: Invalid Date OS Version: undefined 11.0.3 (15A432) Report Version: 105
Exception Type: SIGTRAP Exception Subtype: undefined
Thread 0 name: Thread 0 Crashed: 0 CallIn
0x0000000102c224e4 specialized TimelineHeader.updateTimeHours(timeblocks:) (TimelineHeader.swift:0) 1 CallIn 0x0000000102c20af0 TimelineHeader.handleTimeblocksDependantComponents() (TimelineHeader.swift:0) 2 CallIn
0x0000000102c7a28c specialized MeetingTableViewController.tableView(:viewForHeaderInSection:) (TimelineHeader.swift:78) 3 CallIn
0x0000000102c75d54 @objc MeetingTableViewController.tableView(:viewForHeaderInSection:) (MeetingTableViewController.swift:0) 4 UIKit
0x000000018d1157d8 -[UITableView _delegateViewForHeaderInSection:] (UIKit) 5 UIKit 0x000000018d11def0 96-[UITableView _sectionHeaderView:withFrame:forSection:floating:reuseViewIfPossible:willDisplay:]_block_invoke (UIKit) 6 UIKit 0x000000018cdf1a14 +[UIView(Animation) performWithoutAnimation:] (UIKit) 7 UIKit 0x000000018d11dc60 -[UITableView _sectionHeaderView:withFrame:forSection:floating:reuseViewIfPossible:willDisplay:] (UIKit) 8 UIKit 0x000000018cfc6c04 -[_UITableViewUpdateSupport(Private) _setupAnimationsForExistingHeadersAndFooters] (UIKit) 9 UIKit 0x000000018cfc1070 -[_UITableViewUpdateSupport _setupAnimations] (UIKit) 10 UIKit 0x000000018cfc0944 -[UITableView _updateWithItems:updateSupport:] (UIKit) 11 UIKit 0x000000018cfa8448 -[UITableView endCellAnimationsWithContext:] (UIKit) 12 UIKit 0x000000018cfa46e4 -[UITableView endUpdates] (UIKit) 13 CallIn 0x0000000102c761a4 MeetingTableViewController.setHeaderHeight() (MeetingTableViewController.swift:398) 14 CallIn
0x0000000102c7a7c8 specialized closure #1 in MeetingTableViewController.refreshTable(:) (MeetingTableViewController.swift:513) 15 CallIn
0x0000000102c7aacc partial apply for closure #1 in MeetingTableViewController.refreshTable(_:) (MeetingTableViewController.swift:0) 16 CallIn
0x0000000102cd17f0 thunk for @callee_owned () -> () (LoginPageViewController.swift:0) 17 libdispatch.dylib
0x000000018327ea54 _dispatch_call_block_and_release (libdispatch.dylib) 18 libdispatch.dylib
0x000000018327ea14 _dispatch_client_callout (libdispatch.dylib) 19 libdispatch.dylib 0x000000018328b698 _dispatch_main_queue_callback_4CF$VARIANT$mp (libdispatch.dylib) 20 CoreFoundation 0x00000001838aa544 __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE (CoreFoundation) 21 CoreFoundation 0x00000001838a8120 __CFRunLoopRun (CoreFoundation) 22 CoreFoundation
0x00000001837c7e58 CFRunLoopRunSpecific (CoreFoundation) 23 GraphicsServices 0x0000000185674f84 GSEventRunModal (GraphicsServices) 24 UIKit
0x000000018ce4767c UIApplicationMain (UIKit) 25 CallIn
0x0000000102c02c08 main (AppDelegate.swift:18) 26 libdyld.dylib
0x00000001832e456c start (libdyld.dylib)
最佳答案
复制
这是由于 Timeblock.startHour 和 Timeblock.endHour 被 Int 值弄乱造成的。查看我们的用户日历后,我注意到他们都有在午夜结束的 session 。
因此对于在下午 5 点到午夜之间进行的 session ,我们会发生以下情况。
我在 timeblock.startHour...timeblock.endHour
成为
我在 0...17
这会给出一个非常精确的错误,告诉我一旦我重现了崩溃,我就无法向后迭代,不幸的是,崩溃报告中没有出现,这有点误导。
修复
我们限制了创建 timeblock.endHour 属性的时间,因此第二天的 > 0:00 更改为 23:59 以防止这种情况。修复只是将上限应用到 >= 0:00,因此在午夜结束的 session 不会被视为多天 session 。
future 的重构可能会将所有在整点结束的时间设置为前一小时的结束时间,因为从技术上讲,如果您的 session 在上午 11:00 结束,您仍有空闲时间。我还想在 Timeblock 对象中处理这一切,而不是过滤用于创建它的参数。
关于ios - 如何确定我的 SIGTrap 崩溃的原因?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47991496/
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/
我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为
为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar