iOS中SQLite3數(shù)據(jù)庫(kù)修改字段名

關(guān)于數(shù)據(jù)庫(kù)的增阱当、刪、改糜工、查等基本操作本文不去介紹弊添,網(wǎng)上大把的詳細(xì)教程,這里只介紹實(shí)際使用中的一種情況:數(shù)據(jù)庫(kù)中修改字段名稱捌木。

在產(chǎn)品更迭的過(guò)程中我們有時(shí)候需要修改數(shù)據(jù)庫(kù)表中已存在的字段名或字段類型油坝,這個(gè)時(shí)候我們?nèi)?a target="_blank" rel="nofollow">baidu.com或jianshu.com的話找到的答案應(yīng)該是以下兩種:

  • alter table user CHANGE new1 new4 int;
  • alter table user rename column a to b;

然而這兩種方法都并不能在SQLite3中起作用,因?yàn)?a target="_blank" rel="nofollow">在 SQLite 中刨裆,除了重命名表和在已有的表中添加列澈圈,ALTER TABLE 命令不支持其他操作。也就是說(shuō)帆啃,SQLite無(wú)法使用rename/change語(yǔ)句修改字段名了瞬女。但如果我們產(chǎn)品更迭過(guò)程中不巧有這種奇葩需求了,怎么辦呢努潘?這時(shí)候只能采取迂回政策了:

假如要把tableA中的字段oldColumn修改為newColumn

解決方法:

  1. 在tableA中添加字段newColumn
ALTER TABLE tableA ADD COLUMN newColumn INTEGER
  1. 將舊字段oldColumn的值賦值給新字段newColumn
UPDATE tableA SET newColumn = oldColumn
  1. 刪除舊字段oldColumn
    ALTER TABLE tableA DROP COLUMN oldColumn
    然后到第三步你會(huì)發(fā)現(xiàn)又被SQLite坑了:SQLite無(wú)法刪除字段7掏怠坤学!

第3步解決辦法:(依舊是迂回政策)
3.1 先創(chuàng)建備份表tableA_backup將tableA數(shù)據(jù)備份

CREATE TEMPORARY TABLE tableA_backup(id INTEGER PRIMARY KEY AUTOINCREMENT, newColumn INTEGER)

3.2 將原表tableA中的數(shù)據(jù)對(duì)應(yīng)的復(fù)制到tableA_backup表中

INSERT INTO tableA_backup SELECT id, newColumn FROM tableA

不要將舊字段oldColumn復(fù)制過(guò)去,備份表tableA_backup中沒有oldColumn字段

3.3 此時(shí)實(shí)際已經(jīng)得到了改過(guò)字段名的表:tableA_backup报慕,然后將原表tableA刪掉

DROP TABLE tableA

3.4 然后創(chuàng)建表tableA

CREATE TABLE tableA(id INTEGER PRIMARY KEY AUTOINCREMENT, newColumn INTEGER)

3.5 再把備份表tableA_backup中的數(shù)據(jù)復(fù)制到tableA中

INSERT INTO tableA SELECT id, newColumn FROM tableA_backup

3.6 把備份表tableA_backup刪除深浮,Have done!

DROP TABLE tableA_backup

修改字段名及類型的思路就是上面這些步驟眠冈,下面給貼上代碼方便大家閱讀使用:

# 方法調(diào)用:
            [self alterTable:@"currentList"
             renameOldColumn:@"isvipa"
                 toNewColumn:@"vipType"
            andNewColumnType:@"INTEGER"
                withDatabase:db
   allRenamedColumnsAndTypes:@"id INTEGER PRIMARY KEY AUTOINCREMENT, isinsider INTEGER, vipType INTEGER, groupcode TEXT, groupname TEXT, grouppicpath TEXT, uuid TEXT, AESkey TEXT"
       allRenamedColumnNames:@"id, isinsider, vipType, groupcode, groupname, grouppicpath, uuid, AESkey"];

/**
 修改表中一個(gè)字段名 (sqlite3無(wú)法使用rename/change語(yǔ)句直接修改字段名)

 @param tableName 表名
 @param oldColumnName 要被修改的字段名
 @param newColumnName 字段被修改后的名稱
 @param newColumnType 字段被修改后的類型
 @param db 數(shù)據(jù)庫(kù)對(duì)象
 @param renamedColumnsAndTypes 修改后所有的字段及對(duì)應(yīng)的類型  eg: id INTEGER PRIMARY KEY AUTOINCREMENT, isinsider INTEGER, vipType INTEGER, groupcode TEXT, groupname TEXT, grouppicpath TEXT, uuid TEXT, AESkey TEXT
 @param renamedColumnNames 修改后所有的字段名  eg: id, isinsider, vipType, groupcode, groupname, grouppicpath, uuid, AESkey FROM currentList
 */
-(void)alterTable:(NSString *)tableName
  renameOldColumn:(NSString *)oldColumnName toNewColumn:(NSString *)newColumnName andNewColumnType:(NSString *)newColumnType
     withDatabase:(FMDatabase *)db
allRenamedColumnsAndTypes:(NSString *)renamedColumnsAndTypes allRenamedColumnNames:(NSString *)renamedColumnNames
{
    if (![db columnExists:oldColumnName inTableWithName:tableName])
        return;
    
    DLog(@"%@表中存在%@字段,修改%@字段名稱為%@", tableName, oldColumnName, oldColumnName, newColumnName);
    // 1.添加新字段
    if (![db executeUpdate:[NSString stringWithFormat:@"ALTER TABLE %@ ADD COLUMN %@ %@", tableName, newColumnName, newColumnType]]) {
        DLog(@"%@修改字段:添加新字段%@失敗", tableName, newColumnName);
    }

    // 2.將舊字段賦值給新字段
    if (![db executeUpdate:[NSString stringWithFormat:@"UPDATE %@ SET %@ = %@", tableName, newColumnName, oldColumnName]]) {
        DLog(@"%@修改字段:%@賦值%@字段失敗", tableName, oldColumnName, newColumnName);
        return;
    }
    
    // 3.刪除舊字段(sqlite3無(wú)法使用DROP刪除字段略号,先復(fù)制備份,再創(chuàng)建新的表)
    if (![db executeUpdate:@"BEGIN TRANSACTION"]) {
        DLog(@"BEGIN TRANSACTION失敗");
    }
    NSString *table_backup = [NSString stringWithFormat:@"%@_backup", tableName];
    NSString *sql = [NSString stringWithFormat:@"CREATE TEMPORARY TABLE %@(%@)", table_backup, renamedColumnsAndTypes];
    if (![db executeUpdate:sql]) {
        DLog(@"CREATE TEMPORARY TABLE %@ 失敗", table_backup);
    }
    if (![db executeUpdate:[NSString stringWithFormat:@"INSERT INTO %@ SELECT %@ FROM %@", table_backup, renamedColumnNames, tableName]]) {
        DLog(@"INSERT INTO %@ 失敗", table_backup);
    }
    if (![db executeUpdate:[NSString stringWithFormat:@"DROP TABLE %@", tableName]]) {
        DLog(@"DROP TABLE %@ 失敗", tableName);
    }
    if (![db executeUpdate:[NSString stringWithFormat:@"CREATE TABLE %@(%@)", tableName, renamedColumnsAndTypes]]) {
        DLog(@"CREATE TABLE %@ 失敗", tableName);
    }
    if (![db executeUpdate:[NSString stringWithFormat:@"INSERT INTO %@ SELECT %@ FROM %@", tableName, renamedColumnNames, table_backup]]) {
        DLog(@"INSERT INTO %@ SELECT FROM %@ 失敗", tableName, table_backup);
    }
    if (![db executeUpdate:[NSString stringWithFormat:@"DROP TABLE %@", table_backup]]) {
        DLog(@"DROP TABLE %@ 失敗", table_backup);
    }
    if (![db executeUpdate:@"COMMIT"]) {
        DLog(@"COMMIT失敗");
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末洋闽,一起剝皮案震驚了整個(gè)濱河市玄柠,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌诫舅,老刑警劉巖羽利,帶你破解...
    沈念sama閱讀 219,490評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異刊懈,居然都是意外死亡这弧,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門虚汛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)匾浪,“玉大人,你說(shuō)我怎么就攤上這事卷哩〉氨玻” “怎么了?”我有些...
    開封第一講書人閱讀 165,830評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵将谊,是天一觀的道長(zhǎng)冷溶。 經(jīng)常有香客問(wèn)我,道長(zhǎng)尊浓,這世上最難降的妖魔是什么逞频? 我笑而不...
    開封第一講書人閱讀 58,957評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮栋齿,結(jié)果婚禮上苗胀,老公的妹妹穿的比我還像新娘。我一直安慰自己瓦堵,他們只是感情好基协,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評(píng)論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著谷丸,像睡著了一般堡掏。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上刨疼,一...
    開封第一講書人閱讀 51,754評(píng)論 1 307
  • 那天泉唁,我揣著相機(jī)與錄音鹅龄,去河邊找鬼。 笑死亭畜,一個(gè)胖子當(dāng)著我的面吹牛扮休,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播拴鸵,決...
    沈念sama閱讀 40,464評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼玷坠,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了劲藐?” 一聲冷哼從身側(cè)響起八堡,我...
    開封第一講書人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎聘芜,沒想到半個(gè)月后兄渺,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,847評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡汰现,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評(píng)論 3 338
  • 正文 我和宋清朗相戀三年挂谍,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瞎饲。...
    茶點(diǎn)故事閱讀 40,137評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡口叙,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出嗅战,到底是詐尸還是另有隱情妄田,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評(píng)論 5 346
  • 正文 年R本政府宣布仗哨,位于F島的核電站形庭,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏厌漂。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評(píng)論 3 331
  • 文/蒙蒙 一斟珊、第九天 我趴在偏房一處隱蔽的房頂上張望苇倡。 院中可真熱鬧,春花似錦囤踩、人聲如沸旨椒。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)综慎。三九已至,卻和暖如春勤庐,著一層夾襖步出監(jiān)牢的瞬間示惊,已是汗流浹背好港。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留米罚,地道東北人钧汹。 一個(gè)月前我還...
    沈念sama閱讀 48,409評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像录择,于是被迫代替她去往敵國(guó)和親拔莱。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容