Copying data to the Application Data folder on the iPhone

后端 未结 3 1923
Happy的楠姐
Happy的楠姐 2020-12-29 17:29

I am in the process of (finally) loading my App onto an iPhone device for testing. So far I have only tested the App in the simulator. The application is data-centric and us

相关标签:
3条回答
  • 2020-12-29 18:04

    As Alex said, you have to add your database file as a resource to your project. This will instruct Xcode to bundle your database file in your .app. That file however is read-only, it's up to you to copy it over to your apps document folder (on the device or simulator, same thing), where it becomes writable basically.

    Here's the code that I use for this kind of thing below. The nice thing about this code is that it will automatically refresh the writable copy in your apps document folder whenever you change the "main" copy bundled in the .app. Use 'pathLocal' to open your (writable) database...

    The function below returns YES when the operation was successful (whether a copy was needed or not). You're free to change that with whatever suits you :)

    NSString *pathLocal, *pathBundle;
    
    // Automatically copy DB from .app bundle to device document folder if needed
    - (BOOL)automaticallyCopyDatabase {
        NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory , NSUserDomainMask, YES);
        NSString *documentsDir = [paths objectAtIndex:0];
        pathLocal = [documentsDir stringByAppendingPathComponent:@"mydb.sqlite"];
        pathBundle = [[[NSBundle mainBundle] resourcePath] stringByAppendingPathComponent:@"mydb.sqlite"];
    
        NSFileManager *fileManager = [NSFileManager defaultManager];
        NSDictionary *localAttr = [fileManager fileAttributesAtPath:pathLocal traverseLink:YES];
        BOOL needsCopy = NO;
        if (localAttr == nil) {
            needsCopy = YES;
        } else {
            NSDate *localDate;
            NSDate *appDBDate;
            if (localDate = [localAttr objectForKey:NSFileModificationDate]) {
                NSDictionary *appDBAttr = [fileManager fileAttributesAtPath:pathBundle traverseLink:YES];
                appDBDate = [appDBAttr objectForKey:NSFileModificationDate];
                needsCopy = [appDBDate compare:localDate] == NSOrderedDescending;
            } else {
                needsCopy = YES;
            }
        }
        if (needsCopy) {
            NSError *error;
            BOOL success;
            if (localAttr != nil) {
                success = [fileManager removeItemAtPath:pathLocal error:&error];
            }
            success = [fileManager copyItemAtPath:pathBundle toPath:pathLocal error:&error];
            return success;
        }
        return YES;
    }
    
    0 讨论(0)
  • 2020-12-29 18:05

    here how i done this create a class called DBManagement and in m file implement these methods.also put this line in implementation file above @implementation line

    static sqlite3 *CHManagement = nil;

    -(NSString *) getDBPath
    {
     NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask,YES);
     NSString *documentsDirectory = [paths objectAtIndex:0];
     NSString *MyDatabasePath = [documentsDirectory stringByAppendingString:@"/CHManagement.sqlite"];
      return MyDatabasePath;
    
     }
    
    -(void) checkDataBase
    {
    NSLog(@"CheckDatabase Called");
    NSFileManager *fileManager=[NSFileManager defaultManager];
    NSError *error=[[[NSError alloc]init] autorelease]; 
    NSString *dbPath = [self getDBPath];
    BOOL success=[fileManager fileExistsAtPath:dbPath];
    if(!success)
    {
     NSString *defaultDBPath=nil;
     defaultDBPath=[[[NSBundle mainBundle]resourcePath]stringByAppendingPathComponent:@"CHManagement.sqlite"];
     success=[fileManager copyItemAtPath:defaultDBPath  toPath:dbPath error:&error];
     if(!success)
     {
       NSAssert1(0,@"failed to create database with message '%@'.",[error localizedDescription]); 
      }
     }
     else
     {
     sqlite3 *MyDatabase;
     NSInteger VersionNumber=0;
     sqlite3_stmt *CompiledStatement=nil;
     NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask,YES);
     NSString *documentsDirectory = [paths objectAtIndex:0];
     NSString *MyDatabasePath = [documentsDirectory stringByAppendingString:@"/CHManagement.sqlite"];
     if(sqlite3_open([MyDatabasePath UTF8String],&MyDatabase) == SQLITE_OK)
     {
      sqlite3_prepare_v2(MyDatabase, "select appVersionNo from VersionInfo", -1, &CompiledStatement, NULL);
      while(sqlite3_step(CompiledStatement) == SQLITE_ROW)
      {
       VersionNumber=sqlite3_column_int(CompiledStatement,0);
      }
      sqlite3_reset(CompiledStatement);
      sqlite3_finalize(CompiledStatement);
      sqlite3_close(MyDatabase);
     }
     if (VersionNumber!=1)
     {
      [self deleteDatabase];
      NSString *defaultDBPath=[[[NSBundle mainBundle]resourcePath]stringByAppendingPathComponent:@"CHManagement.sqlite"];
      success=[fileManager copyItemAtPath:defaultDBPath  toPath:dbPath error:&error];
      if(!success)
      {
       NSAssert1(0,@"failed to create database with message '%@'.",[error localizedDescription]); 
       }
      }
     }
     }
    
    - (void)deleteDatabase
    {
    NSLog(@"Delete Database Called");
    NSError **error=nil;
    NSFileManager *FileManager = [NSFileManager defaultManager];
    NSString *databasepath = [self getDBPath];
      BOOL success = [FileManager fileExistsAtPath:databasepath];
    if(success) {
    [FileManager removeItemAtPath:databasepath error:error];
    }
     }
    

    with these methods you can easily load database into iphone document folder of the app. Sorry about the formatting. and appVersionNo is a field in a tableCalled versionInfo the default value is 1

    0 讨论(0)
  • 2020-12-29 18:06

    Why not add it to your application bundle? Right-click on the Resources folder in your project window in Xcode, then select Add > Existing Files... to add your sqlite database.

    You would then load the file from the application bundle:

    NSString *databasePath = [[NSBundle mainBundle] pathForResource:@"mySQLiteDatabaseFile" ofType:@"sqlite3"];
    NSData *databaseData = [NSData dataWithContentsOfFile:databasePath];
    // ...
    
    0 讨论(0)
提交回复
热议问题