IPhone (iOS): копирование файлов из основного пакета в папку документов вызывает сбой
Я пытаюсь настроить приложение таким образом, чтобы при первом запуске в каталог документов копировался ряд файлов, расположенных в папке "Populator" в главном комплекте.
Моя текущая реализация такова:
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
NSString *documentsDirectory = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];
NSString *sourcePath = [[[NSBundle mainBundle] resourcePath] stringByAppendingPathComponent:@"Populator"];
NSString *folderPath = [documentsDirectory stringByAppendingPathComponent:@"Files"];
NSLog(@"Source Path: %@\n Documents Path: %@ \n Folder Path: %@", sourcePath, documentsDirectory, folderPath);
NSError *error;
[[NSFileManager defaultManager] copyItemAtPath:sourcePath
toPath:folderPath
error:&error];
NSLog(@"Error description-%@ \n", [error localizedDescription]);
NSLog(@"Error reason-%@", [error localizedFailureReason]);
....
return YES;
}
Однако это приводит к сбою при первом запуске со следующими консольными сообщениями (но файлы копируются). В следующий раз, когда приложение будет открыто, оно не сработает.
2010-07-13 15:14:26.418 AppName[5201:207] Source Path: /Users/jack/Library/Application Support/iPhone Simulator/3.2/Applications/1076C1FA-60B0-4AC7-8CD4-74F81472DAE6/AppName.app/Populator
Documents Path: /Users/jack/Library/Application Support/iPhone Simulator/3.2/Applications/1076C1FA-60B0-4AC7-8CD4-74F81472DAE6/Documents
Folder Path: /Users/jack/Library/Application Support/iPhone Simulator/3.2/Applications/1076C1FA-60B0-4AC7-8CD4-74F81472DAE6/Documents/Files
2010-07-13 15:14:26.466 AppName[5201:207] *** +[AppNameAppDelegate localizedDescription]: unrecognized selector sent to class 0xa79c
2010-07-13 15:14:26.475 AppName[5201:207] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '*** +[AppNameAppDelegate localizedDescription]: unrecognized selector sent to class 0xa79c'
2010-07-13 15:14:26.495 AppName[5201:207] Stack: (
40911435,
2569270537,
41183227,
40645910,
40642578,
9142,
2815466,
2819475,
2844680,
2826401,
2858055,
49271164,
40452156,
40448072,
2817668,
2850273,
8776,
8630
)
Есть ли у кого-нибудь какие-либо предложения относительно того, что происходит не так? У меня уже есть некоторый код, созданный для реализации функциональности "только при первом запуске", но не включил его здесь для ясности.
Спасибо
Ответы
Ответ 1
Я не знаю много о программировании на iPhone или объекте C, но из любопытства, что такое ошибка в этом случае, если операция копирования действительно сработала? Могут ли быть повреждены строки журнала, если не было ошибок?
[edit] Кроме того, разрешено ли вам копировать все содержимое такого подкаталога? (Опять же, я не знаком с API iOS, просто определяя возможные источники ошибок на основе того, что я знаю о других языках/API)
Ответ 2
NSError *error;
Вы объявляете локальную переменную без ее инициализации. Поэтому он будет заполнен мусором.
[[NSFileManager defaultManager] copyItemAtPath:sourcePath
toPath:folderPath
error:&error];
Если в этой строке не возникает ошибка, статус мусора error
все равно останется.
NSLog(@"Error description-%@ \n", [error localizedDescription]);
Теперь вы отправляете сообщение в случайное, неинициализированное местоположение. Это источник аварии.
Чтобы избежать этого, выполните инициализацию error
до nil
.
NSError* error = nil;
// ^^^^^
Или распечатайте ошибку только тогда, когда -copyItemAtPath:…
возвращает NO (в котором error
правильно заполнено).
if (![[NSFileManager defaultManager] copyItemAtPath:sourcePath ...]) {
NSLog(...);
}
Ответ 3
Я только что прочитал свой код и нашел проблему. Как указывает Шон Эдвардс выше, нет ошибки, если она преуспеет - следовательно, сбой.
Вот мой новый код для заинтересованных:
if([[NSFileManager defaultManager] copyItemAtPath:sourcePath toPath:folderPath error:&error]){
NSLog(@"File successfully copied");
} else {
NSLog(@"Error description-%@ \n", [error localizedDescription]);
NSLog(@"Error reason-%@", [error localizedFailureReason]);
}
Ответ 4
вы должны проверить, существует ли файл уже! Затем скопируйте.
+ (BOOL) getFileExistence: (NSString *) filename
{
BOOL IsFileExists = NO;
NSArray *documentPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentsDir = [documentPaths objectAtIndex:0];
NSString *favsFilePath = [documentsDir stringByAppendingPathComponent:filename];
NSFileManager *fileManager = [NSFileManager defaultManager];
// Check if the database has already been created in the users filesystem
if ([fileManager fileExistsAtPath:favsFilePath])
{
IsFileExists = YES;
}
return IsFileExists;
}
+ (NSString *)dataFilePath:(NSString *)filename {
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *docDirectory = [paths objectAtIndex:0];
return [docDirectory stringByAppendingPathComponent:filename];
}
- (void)copyFileToLocal:(NSString *)filename
{
if (![AppDelegate getFileExistence:filename])
{
NSError *error;
NSString *file = [[NSBundle mainBundle] pathForResource:filename ofType:nil];
if (file)
{
if([[NSFileManager defaultManager] copyItemAtPath:file toPath:[AppDelegate dataFilePath:filename] error:&error]){
NSLog(@"File successfully copied");
} else {
[[[UIAlertView alloc]initWithTitle:NSLocalizedString(@"error", nil) message: NSLocalizedString(@"failedcopydb", nil) delegate:nil cancelButtonTitle:NSLocalizedString(@"ok", nil) otherButtonTitles:nil] show];
NSLog(@"Error description-%@ \n", [error localizedDescription]);
NSLog(@"Error reason-%@", [error localizedFailureReason]);
}
file = nil;
}
}
}
NSLocalizedString
- локализовать строки приложения.
Ответ 5
Вы регистрируете ошибку, прежде чем знаете, что есть ошибка
введите код в if-block
if(error)
{
NSLog(@"Error description-%@ \n", [error localizedDescription]);
NSLog(@"Error reason-%@", [error localizedFailureReason]);
}
Чтобы описать вашу проблему более подробно: указатель ошибок указывает в любом случае, и этот объект не распознает это сообщение. Поэтому вы получаете исключение
Ответ 6
Как общая практика программирования, всегда лучше инициализировать переменные со значением по умолчанию:
NSError *error = nil;
В Objective-C можно отправить сообщение на nil
. Таким образом, в вашем случае переменная ошибки не приведет к сбою, если она была инициализирована на nil
.
Для получения дополнительной информации о предмете проверьте Sending Messages to nil
раздел https://developer.apple.com/library/ios/#documentation/cocoa/conceptual/objectivec/Chapters/ocObjectsClasses.html