Xcode 8 с "Release: Fastest, Smallest [-Os]" имеет некоторую странную проблему и не работает в некотором смысле

У меня есть 1 цель в моем проекте и 3 конфигурации сборки

Debug Production и Release

Пока я запускаю свое приложение с конфигурацией Debug или Production, и он отлично работает без проблем, но когда я меняю свой build configuration на Release, он дает мне некоторые странные проблемы, такие как некоторые строковые значения. не сохраняя в файле plist, и когда я отлаживаю мой режим выпуска, я не получил никакого значения в моем отладчике.

После слишком большого изучения моего кода и настройки сборки и некоторой формы предложения SO я сделал ниже изменение в настройке сборки, как указано ниже.

Измените Уровень оптимизации в заголовке <Генерация кодa > LLVM 8.0

Получение проблемы ниже Уровень оптимизации:

введите описание изображения здесь

Решенная проблема с нижеследующим изменением уровня оптимизации:

введите описание изображения здесь

Не знаю, почему он работает в None [-O0].

Может ли кто-нибудь помочь мне на этом патче?

Вот одна чувствительная часть, на которой выше уровень оптимизации.

Мой сохраненный код закладки, где у меня есть значение, но при этом он сохраняет NULL в plist.

- (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"][email protected]"(0)":@"(%)"]];
    }else{
        aStrTitle = [aStrTitle stringByAppendingString:[NSString stringWithFormat:@" - %@ - %@ - %@",aDivisionModel.division_name,[Utilities getEmirateNameinArabicByCode:aEmirateModel.emirate_code],[self.bookMarkViewModel.strDataType isEqualToString:@"Number"][email protected]"(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:@"%@",[email protected]"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.

Ответы

Ответ 1

Ваша проблема в этой строке:

__weak NSString *aStrTitle = [NSString stringWithFormat:@"%@",aDimensionModel.dimension_description];

+[NSString stringWithFormat:] создает новую строку со значением удержания, равным нулю. И это обычно хорошо, поскольку это значение либо назначается переменной strong, либо передается как параметр вызова метода. В обоих случаях значение счетчика увеличивается, а значение остается нетронутым. Но вы назначаете эту новую строку переменной __weak! Слабые переменные не влияют на счет сохранения, поэтому он остается нулевым после выполнения этой строки. И поэтому созданная строка удаляется сразу после создания.

Решение прост: удалите спецификатор __weak, и вам будет хорошо идти.

Нет смысла объявлять локальные переменные слабыми (если они не будут захвачены блоками). Также "очистка", как эта: aStrVCName = nil; не имеет никакого эффекта.

  • Все локальные переменные автоматически освобождаются, когда метод заканчивает выполнение.
  • aStrVCName в любом случае слаб, поэтому он никогда не влияет на количество отсчетов объектов, на которые он указывает.

Ответ 2

У меня была аналогичная проблема при переносе на xcode 8 и swift 3. Оказывается, быстрому не нравится тот факт, что я переопределяю функции из базовых классов в расширениях. см. Переопределение методов в расширениях Swift

По какой-то причине компилятор не жалуется на это, и, чтобы усугубить ситуацию, "странные проблемы" начались только на уровне оптимизации Fastest, Smallest [-Os].

Похоже, что на уровне оптимизации Fastest, Smallest [-Os] компилятор вызовет функцию из базового класса, а не из функции расширения. На уровне оптимизации 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")
    }
}

распечатает "расширенную функцию" на одном уровне оптимизации и выведет "ok" в самый быстрый уровень оптимизации...

чтобы сделать рассказ коротким, не переопределяйте функции в быстрых расширениях