Xcode 8 with "Release: Fastest, Smallest [-Os]" 有一些奇怪的问题,在某些情况下功能不正常
Xcode 8 with "Release: Fastest, Smallest [-Os]" have some weird issue and not functional well in some case
我的项目中有 1 个目标和 3 个构建配置
Debug
Production
& Release
虽然我 运行 我的应用程序具有 debug
或 production
配置并且它工作正常没有任何问题,但是当我将我的 build configuration
更改为 release
它给了我一些奇怪的问题,比如一些字符串值没有存储在 plist
文件中,当我调试我的发布模式时,我没有在我的调试器中得到任何值。
在对我的代码和构建设置进行过多调查以及 SO
的一些建议后,我在我的构建设置中做了以下更改,如下所述。
在 Apple LLVM 8.0 代码生成下更改优化级别 header
在优化级别以下出现问题:
已解决以下优化级别更改的问题:
不知道为什么它在 None[-O0].
谁能帮我打上面的补丁?
这是一个敏感部分,上面的优化级别受到影响。
我保存的书签代码在我有价值的地方,但它仍然在 plist
中存储 NULL
。
- (IBAction)bookmark:(id)sender{
DimensionModel *aDimensionModel = APPDELEGATE.selected_dimensionModel;
EmirateModel *aEmirateModel = APPDELEGATE.selected_emirateModel;
DivisionModel *aDivisionModel = APPDELEGATE.selected_divisionModel;
__weak NSString *aStrVCName = self.bookMarkViewModel.selectedStrVCName;
//Add Title string in Dictionary and then in Plist
__weak NSString *aStrTitle = [NSString stringWithFormat:@"%@",aDimensionModel.dimension_description];
aStrTitle = [aStrTitle stringByReplacingOccurrencesOfString:@"(null)" withString:@""];
if (IS_ENGLISH) {
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@ - %@ - %@",aDivisionModel.division_name,aEmirateModel.emirate_name,[self.bookMarkViewModel.strDataType isEqualToString:@"Number"]?@"(0)":@"(%)"]];
}else{
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@ - %@ - %@",aDivisionModel.division_name,[Utilities getEmirateNameinArabicByCode:aEmirateModel.emirate_code],[self.bookMarkViewModel.strDataType isEqualToString:@"Number"]?@"(0)":@"(%)"]];
}
if (APPDELEGATE.selectedDimRecordTimeModel.record_year == -1) {
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@",[LanguageUtility getLocalizedStringForKey:@"All Years"]]];
}else{
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %ld",(long)APPDELEGATE.selectedDimRecordTimeModel.record_year]];
}
NSMutableDictionary *aMutDicGraphDetail = [NSMutableDictionary new];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",self.bookMarkViewModel.strSelectedDim] forKey:@"DimCode"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aStrVCName] forKey:@"VCName"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",PLIST_TITLE_BOOKMARK] forKey:@"PlistBookmark"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",IS_ENGLISH?@"ENGLISH":@"ARABIC"] forKey:@"Language"];
if (self.bookMarkViewModel.strSelectedDim == nil) {
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",[LanguageUtility getLocalizedStringForKey:aStrVCName]] forKey:@"Title"];
}else{
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aStrTitle] forKey:@"Title"];
}
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",APPDELEGATE.selectedDimRecordTimeModel.record_time] forKey:@"RecordTime"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%ld",(long)APPDELEGATE.selectedDimRecordTimeModel.record_year] forKey:@"RecordYear"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aEmirateModel.emirate_code] forKey:@"EmirateCode"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",self.bookMarkViewModel.strDataType] forKey:@"DataType"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aDivisionModel.division_code] forKey:@"DivisionCode"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%d",APPDELEGATE.isDataTypeSwitchHide] forKey:@"switchHide"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",APPDELEGATE.strYearOrMonthFromRbtnCtrl] forKey:@"YearOrMonthFromRbtnCtrl"];
[aMutDicGraphDetail setObject:@(self.bookMarkViewModel.selectedIndexPath).stringValue forKey:DimSelectedIndex];
/* Check for RegionVC and CountryVC*/
if ([aStrVCName isEqualToString:@"RegionMainVC"] || [aStrVCName isEqualToString:@"CountryMainVC"]){
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",self.bookMarkViewModel.strComparison] forKey:@"Comparison"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%ld",self.bookMarkViewModel.intSelectedRegionId] forKey:@"RegionId"];
}
[self.bookMarkViewModel saveBookmark:aMutDicGraphDetail];
aStrVCName = nil;
}
使用上面的代码,所有值都已成功存储到 plist
中,但是 aStrTitle
中有值并且仍然存储为 null
.
我在迁移到 xcode 8 和 swift 3 时遇到了类似的问题。事实证明,swift 不喜欢我从基础 [=31] 覆盖函数的事实=]es 在扩展中。参见
出于某种原因,编译器并没有抱怨它,更糟糕的是,"strange issues" 只在 Fastest, Smallest [-Os]
优化级别开始发生。
似乎在 Fastest, Smallest [-Os]
优化级别中,编译器将调用基础 class 中的函数,而不是扩展中的函数。在 None [-O0]
优化级别,编译器将从扩展中调用函数。
例如:
public class BaseController : UIViewController {
override func reloadInterface() {
println("ok")
}
}
public class NewController : BaseController {
override func viewDidLoad() {
super.viewDidLoad()
reloadInterface()
}
}
extension NewController {
override func reloadInterface() {
println("extended function")
}
}
将在 none 优化级别打印出 "extended function",并将在最快优化级别打印出 "ok"...
简而言之,不要覆盖 swift 扩展中的函数
您的问题出在这一行:
__weak NSString *aStrTitle = [NSString stringWithFormat:@"%@",aDimensionModel.dimension_description];
+[NSString stringWithFormat:]
创建一个保留计数为零的新字符串。这通常没问题,因为该值要么分配给 strong
变量,要么作为参数传递给方法调用。在这两种情况下,它的保留计数都会增加并且值保持不变。但是您正在将这个新字符串分配给 __weak
变量!弱变量不影响保留计数,因此在执行此行后它保持为零。因此创建的字符串在创建后立即被删除。
解决方案很简单:删除 __weak
说明符即可。
将局部变量声明为弱变量没有意义(除非它们被块捕获)。还有像这样的"cleanups":aStrVCName = nil;
根本没有效果。
- 方法执行完毕后,所有局部变量都会自动释放。
aStrVCName
无论如何都很弱,所以它永远不会影响它所指向的对象的保留计数。
我的项目中有 1 个目标和 3 个构建配置
Debug
Production
& Release
虽然我 运行 我的应用程序具有 debug
或 production
配置并且它工作正常没有任何问题,但是当我将我的 build configuration
更改为 release
它给了我一些奇怪的问题,比如一些字符串值没有存储在 plist
文件中,当我调试我的发布模式时,我没有在我的调试器中得到任何值。
在对我的代码和构建设置进行过多调查以及 SO
的一些建议后,我在我的构建设置中做了以下更改,如下所述。
在 Apple LLVM 8.0 代码生成下更改优化级别 header
在优化级别以下出现问题:
已解决以下优化级别更改的问题:
不知道为什么它在 None[-O0].
谁能帮我打上面的补丁?
这是一个敏感部分,上面的优化级别受到影响。
我保存的书签代码在我有价值的地方,但它仍然在 plist
中存储 NULL
。
- (IBAction)bookmark:(id)sender{
DimensionModel *aDimensionModel = APPDELEGATE.selected_dimensionModel;
EmirateModel *aEmirateModel = APPDELEGATE.selected_emirateModel;
DivisionModel *aDivisionModel = APPDELEGATE.selected_divisionModel;
__weak NSString *aStrVCName = self.bookMarkViewModel.selectedStrVCName;
//Add Title string in Dictionary and then in Plist
__weak NSString *aStrTitle = [NSString stringWithFormat:@"%@",aDimensionModel.dimension_description];
aStrTitle = [aStrTitle stringByReplacingOccurrencesOfString:@"(null)" withString:@""];
if (IS_ENGLISH) {
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@ - %@ - %@",aDivisionModel.division_name,aEmirateModel.emirate_name,[self.bookMarkViewModel.strDataType isEqualToString:@"Number"]?@"(0)":@"(%)"]];
}else{
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@ - %@ - %@",aDivisionModel.division_name,[Utilities getEmirateNameinArabicByCode:aEmirateModel.emirate_code],[self.bookMarkViewModel.strDataType isEqualToString:@"Number"]?@"(0)":@"(%)"]];
}
if (APPDELEGATE.selectedDimRecordTimeModel.record_year == -1) {
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@",[LanguageUtility getLocalizedStringForKey:@"All Years"]]];
}else{
aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %ld",(long)APPDELEGATE.selectedDimRecordTimeModel.record_year]];
}
NSMutableDictionary *aMutDicGraphDetail = [NSMutableDictionary new];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",self.bookMarkViewModel.strSelectedDim] forKey:@"DimCode"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aStrVCName] forKey:@"VCName"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",PLIST_TITLE_BOOKMARK] forKey:@"PlistBookmark"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",IS_ENGLISH?@"ENGLISH":@"ARABIC"] forKey:@"Language"];
if (self.bookMarkViewModel.strSelectedDim == nil) {
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",[LanguageUtility getLocalizedStringForKey:aStrVCName]] forKey:@"Title"];
}else{
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aStrTitle] forKey:@"Title"];
}
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",APPDELEGATE.selectedDimRecordTimeModel.record_time] forKey:@"RecordTime"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%ld",(long)APPDELEGATE.selectedDimRecordTimeModel.record_year] forKey:@"RecordYear"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aEmirateModel.emirate_code] forKey:@"EmirateCode"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",self.bookMarkViewModel.strDataType] forKey:@"DataType"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",aDivisionModel.division_code] forKey:@"DivisionCode"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%d",APPDELEGATE.isDataTypeSwitchHide] forKey:@"switchHide"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",APPDELEGATE.strYearOrMonthFromRbtnCtrl] forKey:@"YearOrMonthFromRbtnCtrl"];
[aMutDicGraphDetail setObject:@(self.bookMarkViewModel.selectedIndexPath).stringValue forKey:DimSelectedIndex];
/* Check for RegionVC and CountryVC*/
if ([aStrVCName isEqualToString:@"RegionMainVC"] || [aStrVCName isEqualToString:@"CountryMainVC"]){
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%@",self.bookMarkViewModel.strComparison] forKey:@"Comparison"];
[aMutDicGraphDetail setObject:[NSString stringWithFormat:@"%ld",self.bookMarkViewModel.intSelectedRegionId] forKey:@"RegionId"];
}
[self.bookMarkViewModel saveBookmark:aMutDicGraphDetail];
aStrVCName = nil;
}
使用上面的代码,所有值都已成功存储到 plist
中,但是 aStrTitle
中有值并且仍然存储为 null
.
我在迁移到 xcode 8 和 swift 3 时遇到了类似的问题。事实证明,swift 不喜欢我从基础 [=31] 覆盖函数的事实=]es 在扩展中。参见
出于某种原因,编译器并没有抱怨它,更糟糕的是,"strange issues" 只在 Fastest, Smallest [-Os]
优化级别开始发生。
似乎在 Fastest, Smallest [-Os]
优化级别中,编译器将调用基础 class 中的函数,而不是扩展中的函数。在 None [-O0]
优化级别,编译器将从扩展中调用函数。
例如:
public class BaseController : UIViewController {
override func reloadInterface() {
println("ok")
}
}
public class NewController : BaseController {
override func viewDidLoad() {
super.viewDidLoad()
reloadInterface()
}
}
extension NewController {
override func reloadInterface() {
println("extended function")
}
}
将在 none 优化级别打印出 "extended function",并将在最快优化级别打印出 "ok"...
简而言之,不要覆盖 swift 扩展中的函数
您的问题出在这一行:
__weak NSString *aStrTitle = [NSString stringWithFormat:@"%@",aDimensionModel.dimension_description];
+[NSString stringWithFormat:]
创建一个保留计数为零的新字符串。这通常没问题,因为该值要么分配给 strong
变量,要么作为参数传递给方法调用。在这两种情况下,它的保留计数都会增加并且值保持不变。但是您正在将这个新字符串分配给 __weak
变量!弱变量不影响保留计数,因此在执行此行后它保持为零。因此创建的字符串在创建后立即被删除。
解决方案很简单:删除 __weak
说明符即可。
将局部变量声明为弱变量没有意义(除非它们被块捕获)。还有像这样的"cleanups":aStrVCName = nil;
根本没有效果。
- 方法执行完毕后,所有局部变量都会自动释放。
aStrVCName
无论如何都很弱,所以它永远不会影响它所指向的对象的保留计数。