我是objective-c和sqlite的新手。我已成功使用sqlite将数据插入到表中。我的项目包含2页
添加提醒页面:用户输入数据并单击保存,这是右侧导航栏
查看提醒页面:这是用户在表格视图的单元格中查看已保存的提醒的地方,
即。第1单元的第1次提醒,第2单元的第2次提醒等等......
我使用以下代码更轻松地指定了部分数量,这意味着已保存提醒的数量等于查看提醒表中的部分数量
-(NSInteger)numberOfSectionsInTableView:(UITableView *)view
{
numTableRecords = -1;
sqlite3_stmt *statement;
if (sqlite3_open([self.databasePath UTF8String], &remindersDB) == SQLITE_OK)
{
NSString *sqlStatement = [NSString stringWithFormat: @"select count(*) from reminders"];
const char *sql = [sqlStatement cStringUsingEncoding:NSUTF8StringEncoding];
if(sqlite3_prepare_v2(remindersDB, sql, -1, &statement, NULL) == SQLITE_OK)
{
while(sqlite3_step(statement) == SQLITE_ROW)
{
numTableRecords = sqlite3_column_int(statement, 0);
}
}
else
{
printf("could not prepare statement: %s\n", sqlite3_errmsg(remindersDB));
}
}
else
{
NSLog(@"Error in Opening Database File");
}
sqlite3_close(remindersDB);
NSLog(@"Number of records = %d",numTableRecords);
return numTableRecords;
}
根据以下链接中的建议,与我的要求非常相似:
Populate the table view sections with rows of table in sqlite database in an order
我接受了一个数组并将检索到的数据作为对象插入到数组中,如下所示:
-(void)viewWillAppear:(BOOL)animated
{
//Retrieve the values of database
const char *dbpath = [self.databasePath UTF8String];
sqlite3_stmt *statement;
if (sqlite3_open(dbpath, &remindersDB) == SQLITE_OK)
{
NSString *querySQL = [NSString stringWithFormat:@"SELECT * FROM reminders"];
NSLog(@"Data = %@",querySQL);
const char *query_stmt = [querySQL UTF8String];
if (sqlite3_prepare_v2(self.remindersDB ,query_stmt , -1, &statement, NULL) == SQLITE_OK)
{
if (sqlite3_step(statement) == SQLITE_ROW)
{
ID = [[[NSString alloc] initWithUTF8String:(const char *) sqlite3_column_text(statement, 0)]autorelease];
nameField = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 1)]autorelease];
eventField = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 2)]autorelease];
dateField = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 3)]autorelease];
}
sqlite3_finalize(statement);
}
sqlite3_close(self.remindersDB);
}
[array addObject:ID];
[array addObject:nameField];
[array addObject:eventField];
[array addObject:dateField];
[self.theTable reloadData];
[super viewWillAppear:YES];
}
现在在(UITableViewCell) - 索引路径的行的单元格中,我获取了4个标签,并将它们添加为子视图,并将[array objectAtIndex:indexPath.row]分配给label.text,即:
- (UITableViewCell *)tableView:(UITableView *)view cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
//NSString *CellId = [NSString stringWithFormat:@"S%1dR%1d",indexPath.section,indexPath.row];
UITableViewCell *cell = (UITableViewCell *)[view dequeueReusableCellWithIdentifier:nil];
if (cell == nil)
{
cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:nil] autorelease];
view.backgroundColor = [UIColor clearColor];
cell.backgroundColor = [[UIColor alloc]initWithPatternImage:[UIImage imageNamed:@"reminderbutton.png"]];
label1 = [[[UILabel alloc]initWithFrame:CGRectMake(26, 3, 30, 40)]autorelease];
label1.backgroundColor = [UIColor clearColor];
label1.textColor = [UIColor whiteColor];
label2 = [[[UILabel alloc]initWithFrame:CGRectMake(45, 3, 100, 40)]autorelease];
label2.backgroundColor = [UIColor clearColor];
label2.textColor = [UIColor whiteColor];
label3 = [[[UILabel alloc]initWithFrame:CGRectMake(119, 3, 100, 40)]autorelease];
label3.backgroundColor = [UIColor clearColor];
label3.textColor = [UIColor whiteColor];
label4 = [[[UILabel alloc]initWithFrame:CGRectMake(198, 3, 120, 40)]autorelease];
label4.backgroundColor = [UIColor clearColor];
label4.textColor = [UIColor whiteColor];
}
label1.text = [array objectAtIndex:indexPath.row];
label2.text = [array objectAtIndex:indexPath.row];
label3.text = [array objectAtIndex:indexPath.row];
label4.text = [array objectAtIndex:indexPath.row];
[cell addSubview:label1];
[cell addSubview:label2];
[cell addSubview:label3];
[cell addSubview:label4];
return cell;
}
在细胞上没有显示任何内容,请帮助我解决这个问题。对此我非常不好,因为我是Objective-c的新手,如果有人提出详细的解释和解决方案,我会很高兴.Sorry如果它听起来很愚蠢的问题
提前感谢所有人:)
答案 0 :(得分:7)
我明白你现在想做什么。
我建议创建一个类来保存每个提醒的数据,然后你就可以得到一个提醒数据数组。
所以我会在你的项目中添加一个新类,称之为Reminder,然后为每个字段添加一个属性。
您的提醒课程如下:
Reminder.h
#import <Foundation/Foundation.h>
@interface Reminder : NSObject
@property (nonatomic, copy) NSString *reminderId;
@property (nonatomic, copy) NSString *name;
@property (nonatomic, copy) NSString *event;
@property (nonatomic, copy) NSString *date;
@end
并且您的reminder.m文件将是:
#import "Reminder.h"
@implementation Reminder
@synthesize reminderId, name, event, date;
-(void)dealloc
{
self.reminderId = nil;
self.name = nil;
self.event = nil;
self.date = nil;
}
@end
因此,此类只会保存每个提醒的数据。
然后在显示提醒列表的视图控制器中,您需要添加一个数组属性来保存提醒对象。
因此,请将以下属性添加到ViewController,以显示提醒列表。
@property (nonatomic, retain) NSMutableArray *reminders;
确保你也在视图控制器.m文件中合成它,为此,在视图控制器.m文件的@implementation行下添加这行代码:
@synthesize reminders;
还记得在dealloc方法中将reminders属性设置为nil以确保正确的内存清理:
self.reminders = nil;
我们将使用我们在视图控制器中创建的Reminder类,因此您还需要导入您的类,因此在视图控制器.m文件中,在文件顶部附近添加一个import语句,如下所示:
#import "Reminder.h"
现在您的视图控制器已设置好并可以使用我们创建的提醒类。
然后,您需要在视图控制器中添加一个方法,该方法将用于从数据库加载数据并填充提醒对象和提醒数组。
向视图控制器添加一个新方法,称之为loadReminders
此方法的实现将是:
-(void)loadReminders
{
// setup the reminders array
self.reminders = nil;
self.reminders = [[[NSMutableArray alloc] init] autorelease];
//Retrieve the values of database
const char *dbpath = [self.databasePath UTF8String];
sqlite3_stmt *statement;
if (sqlite3_open(dbpath, &remindersDB) == SQLITE_OK)
{
NSString *querySQL = [NSString stringWithFormat:@"SELECT * FROM reminders"];
NSLog(@"Data = %@",querySQL);
const char *query_stmt = [querySQL UTF8String];
if (sqlite3_prepare_v2(self.remindersDB ,query_stmt , -1, &statement, NULL) == SQLITE_OK)
{
while (sqlite3_step(statement) == SQLITE_ROW)
{
Reminder *loadedReminder = [[Reminder alloc] init];
loadedReminder.reminderId = [[[NSString alloc] initWithUTF8String:(const char *) sqlite3_column_text(statement, 0)]autorelease];
loadedReminder.name = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 1)]autorelease];
loadedReminder.event = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 2)]autorelease];
loadedReminder.date = [[[NSString alloc]initWithUTF8String:(const char *) sqlite3_column_text(statement, 3)]autorelease];
[self.reminders addObject:loadedReminder];
[loadedReminder release];
}
sqlite3_finalize(statement);
}
sqlite3_close(self.remindersDB);
}
}
所以这个方法正在做的是从数据库加载行,对于数据库中的每一行,它创建一个新的Reminder对象,使用行中的数据设置属性,然后将此提醒对象添加到提醒阵列。
注意我从原始代码中所做的更改:
while (sqlite3_step(statement) == SQLITE_ROW)
while语句将循环并从数据库中获取每一行。
每当你想要刷新数据列表时,你都会调用这个新方法(loadReminders),所以你当前正在重新加载表的代码中的任何地方,在重新加载表之前调用这个方法,这将确保你的数组提醒已准备好供重新加载时使用的表。因此,在上面的示例中,我将viewWillAppear方法替换为:
-(void)viewWillAppear:(BOOL)animated
{
[self loadReminders];
[self.theTable reloadData];
}
现在所有数据都已设置好并可以使用。使用提醒对象应该使tableView中的代码更容易阅读。
因此,要将数据放入表中,请使用以下内容替换当前的tableview方法。
-(NSInteger)numberOfSectionsInTableView:(UITableView *)view
{
return [self.reminders count];
}
这将告诉表格为提醒数组中的每个提醒创建一个部分。
现在,根据您的描述(和链接),您正在做的是在每个部分创建1个单元格。因此,添加以下方法告诉tableView我们希望每个部分有1行
-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return 1;
}
现在对于cellForRowAtIndexPath,请使用以下内容替换当前方法:
- (UITableViewCell *)tableView:(UITableView *)view cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
Reminder *reminderToDisplay;
reminderToDisplay = [self.reminders objectAtIndex:indexPath.section];
// Now create the cell to display the reminder data:
cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:nil] autorelease];
view.backgroundColor = [UIColor clearColor];
cell.backgroundColor = [[UIColor alloc]initWithPatternImage:[UIImage imageNamed:@"reminderbutton.png"]];
label1 = [[[UILabel alloc]initWithFrame:CGRectMake(26, 3, 30, 40)]autorelease];
label1.backgroundColor = [UIColor clearColor];
label1.textColor = [UIColor whiteColor];
label2 = [[[UILabel alloc]initWithFrame:CGRectMake(45, 3, 100, 40)]autorelease];
label2.backgroundColor = [UIColor clearColor];
label2.textColor = [UIColor whiteColor];
label3 = [[[UILabel alloc]initWithFrame:CGRectMake(119, 3, 100, 40)]autorelease];
label3.backgroundColor = [UIColor clearColor];
label3.textColor = [UIColor whiteColor];
label4 = [[[UILabel alloc]initWithFrame:CGRectMake(198, 3, 120, 40)]autorelease];
label4.backgroundColor = [UIColor clearColor];
label4.textColor = [UIColor whiteColor];
// Now set the labels using our reminder object
label1.text = reminderToDisplay.reminderId;
label2.text = reminderToDisplay.name;
label3.text = reminderToDisplay.event;
label4.text = reminderToDisplay.date;
// now add the labels to the cell
cell.contentView = [[UIView alloc] init] autorelease]
[cell.contentView addSubview:label1];
[cell.contentView addSubview:label2];
[cell.contentView addSubview:label3];
[cell.contentView addSubview:label4];
return cell;
}
所以在这个方法中发生了什么,首先我们得到提醒数组中当前部分的提示:
Reminder *reminderToDisplay;
reminderToDisplay = [self.reminders objectAtIndex:indexPath.section];
接下来的几行应该是熟悉的,我们正在创建单元格和标签。然后我们使用提醒对象来设置标签文本。
您现有解决方案无法正常工作的原因是因为您使用的是indexPath.Row,并且因为您为每个提醒使用了一个部分,所以indexPath.Row将始终为0,因为每个部分中只有1行。
我知道这篇文章很长,但希望能帮助你了解正在发生的事情。任何问题然后让我知道
答案 1 :(得分:1)
你应该真正创建一个自定义UITableViewCell。
但是对于快速修复答案,您可以使用以下
替换底部[cell addSubview]行cell.contentView = [[UIView alloc] init] autorelease]
[cell.contentView addSubview:label1];
[cell.contentView addSubview:label2];
[cell.contentView addSubview:label3];
[cell.contentView addSubview:label4];