mysql多次調(diào)用存儲(chǔ)過程的問題
更新時(shí)間:2011年05月07日 23:59:15 作者:
這個(gè)問題也困擾了我很長(zhǎng)時(shí)間,準(zhǔn)確的說正是因?yàn)樗拇鎯?chǔ)過程無法在同一連接中2次或者多次執(zhí)行,我大幅修該了程序架構(gòu),全部題換成了sql,但是畢竟sql無法執(zhí)行有相當(dāng)邏輯的代碼,最總讓我從新測(cè)試以求尋找解決之道。
問題是這樣的,在直接使用mysql c api構(gòu)建應(yīng)用的時(shí)候,一個(gè)連接只能執(zhí)行一次存儲(chǔ)過程,不管怎樣free再次利用這個(gè)連接的時(shí)候就會(huì)出現(xiàn)不能執(zhí)行qurry的提示,我想,這么大型的軟件 不會(huì)存在這樣的基礎(chǔ)性問題吧,畢竟大多數(shù)的adodb都是基于c api的,難道所有的軟件都無法進(jìn)行連接緩存?
別說,接觸到這類問題的人還真不多,大多數(shù)的連接池都寫好了,誰會(huì)去直接構(gòu)建連接進(jìn)行操作呢,所以baidu google基本都沒有結(jié)果,今天回頭來解決這個(gè)問題,突然就有了突破,找到了一篇文章《對(duì)Mysql的C API調(diào)用存儲(chǔ)過程的問題及解決方法分析》,正是我的問題。
作者分析的挺復(fù)雜了,其實(shí)我們一般只用mysql_query、mysql_use_result等幾個(gè)函數(shù),問題的癥結(jié)在于當(dāng)執(zhí)行一個(gè)存儲(chǔ)過程的時(shí)候, 數(shù)據(jù)庫返回的是多個(gè)數(shù)據(jù)集合,即使只有一個(gè)數(shù)據(jù)集合,他也會(huì)有一個(gè)空集合用于結(jié)束一次回話,作者罵他變態(tài),其實(shí)不然,可能作者考慮到的只是一般的請(qǐng) 求,mysql是給所有用戶使用的,說不能真有變態(tài)的人把圖片文件等等直接保存在mysql字段里面,那么回復(fù)就不可能一次完成,需要多此網(wǎng)絡(luò)交互,那么 所有的交互肯定需要一個(gè)結(jié)束符號(hào),并且存儲(chǔ)過程本來就可以返回多個(gè)數(shù)據(jù)集合,如果他在c pai中只做一個(gè)結(jié)果己處理就允許下一次全新的請(qǐng)求,那么對(duì)于同一連接,在mysql服務(wù)斷其實(shí)還有沒有發(fā)送完成的數(shù)據(jù),這個(gè)時(shí)候他安全的做法就是不接受 任何新的請(qǐng)求,直到數(shù)據(jù)發(fā)送完全,或者連接關(guān)閉,不然,mysql協(xié)議解析就會(huì)出現(xiàn)問題,下次發(fā)送就會(huì)出現(xiàn)黏包或者丟包,所以他的做法是完全正確的,即使 只有一個(gè)結(jié)果集合,也需要當(dāng)前會(huì)話內(nèi)的通訊(比如mysql_next_result)確認(rèn)完畢,然后結(jié)束本次請(qǐng)求,這個(gè)時(shí)候服務(wù)端其實(shí)沒有數(shù)據(jù)了,但是 這個(gè)過程是不能省略的,然后在不關(guān)閉連接的情況下就可以進(jìn)行全新的請(qǐng)求了。
所以總結(jié)起來就是:使用存儲(chǔ)過程的時(shí)候一定要循環(huán)執(zhí)行,把所有的結(jié)果集合都取到,直到為空,這個(gè)時(shí)候當(dāng)前數(shù)據(jù)庫連接才可以安全歸還回去,下面是demo。
MYSQL_RES *conn;
MYSQL_RES *res;
MYSQL_ROW row;
conn = 連接池.Get("xxxx");
mysql_query(conn, "call qt()");
res = mysql_use_result(conn);
while ((row = mysql_fetch_row(res))) {
do main thins; //這次是對(duì)于第一個(gè)數(shù)據(jù)集的處理
}
mysql_free_result(res);
while ((res = mysql_next_result(conn)) != NULL) {
do some thing; //循環(huán)處理其他的數(shù)據(jù)集
mysql_free_result(res);
}
連接池.Free(conn)
經(jīng)過上面的循環(huán)讀取就能夠保證一條連接會(huì)干凈地歸還到連接池,當(dāng)然有的連接池可能已經(jīng)把后面一個(gè)循環(huán)放在連接池中處理了,解釋說把剩余的數(shù)據(jù)集全部取過來然后釋放,這樣比如在php等里面使用mysql的連接池的時(shí)候就沒有感受到上面說的問題。
別說,接觸到這類問題的人還真不多,大多數(shù)的連接池都寫好了,誰會(huì)去直接構(gòu)建連接進(jìn)行操作呢,所以baidu google基本都沒有結(jié)果,今天回頭來解決這個(gè)問題,突然就有了突破,找到了一篇文章《對(duì)Mysql的C API調(diào)用存儲(chǔ)過程的問題及解決方法分析》,正是我的問題。
作者分析的挺復(fù)雜了,其實(shí)我們一般只用mysql_query、mysql_use_result等幾個(gè)函數(shù),問題的癥結(jié)在于當(dāng)執(zhí)行一個(gè)存儲(chǔ)過程的時(shí)候, 數(shù)據(jù)庫返回的是多個(gè)數(shù)據(jù)集合,即使只有一個(gè)數(shù)據(jù)集合,他也會(huì)有一個(gè)空集合用于結(jié)束一次回話,作者罵他變態(tài),其實(shí)不然,可能作者考慮到的只是一般的請(qǐng) 求,mysql是給所有用戶使用的,說不能真有變態(tài)的人把圖片文件等等直接保存在mysql字段里面,那么回復(fù)就不可能一次完成,需要多此網(wǎng)絡(luò)交互,那么 所有的交互肯定需要一個(gè)結(jié)束符號(hào),并且存儲(chǔ)過程本來就可以返回多個(gè)數(shù)據(jù)集合,如果他在c pai中只做一個(gè)結(jié)果己處理就允許下一次全新的請(qǐng)求,那么對(duì)于同一連接,在mysql服務(wù)斷其實(shí)還有沒有發(fā)送完成的數(shù)據(jù),這個(gè)時(shí)候他安全的做法就是不接受 任何新的請(qǐng)求,直到數(shù)據(jù)發(fā)送完全,或者連接關(guān)閉,不然,mysql協(xié)議解析就會(huì)出現(xiàn)問題,下次發(fā)送就會(huì)出現(xiàn)黏包或者丟包,所以他的做法是完全正確的,即使 只有一個(gè)結(jié)果集合,也需要當(dāng)前會(huì)話內(nèi)的通訊(比如mysql_next_result)確認(rèn)完畢,然后結(jié)束本次請(qǐng)求,這個(gè)時(shí)候服務(wù)端其實(shí)沒有數(shù)據(jù)了,但是 這個(gè)過程是不能省略的,然后在不關(guān)閉連接的情況下就可以進(jìn)行全新的請(qǐng)求了。
所以總結(jié)起來就是:使用存儲(chǔ)過程的時(shí)候一定要循環(huán)執(zhí)行,把所有的結(jié)果集合都取到,直到為空,這個(gè)時(shí)候當(dāng)前數(shù)據(jù)庫連接才可以安全歸還回去,下面是demo。
復(fù)制代碼 代碼如下:
MYSQL_RES *conn;
MYSQL_RES *res;
MYSQL_ROW row;
conn = 連接池.Get("xxxx");
mysql_query(conn, "call qt()");
res = mysql_use_result(conn);
while ((row = mysql_fetch_row(res))) {
do main thins; //這次是對(duì)于第一個(gè)數(shù)據(jù)集的處理
}
mysql_free_result(res);
while ((res = mysql_next_result(conn)) != NULL) {
do some thing; //循環(huán)處理其他的數(shù)據(jù)集
mysql_free_result(res);
}
連接池.Free(conn)
經(jīng)過上面的循環(huán)讀取就能夠保證一條連接會(huì)干凈地歸還到連接池,當(dāng)然有的連接池可能已經(jīng)把后面一個(gè)循環(huán)放在連接池中處理了,解釋說把剩余的數(shù)據(jù)集全部取過來然后釋放,這樣比如在php等里面使用mysql的連接池的時(shí)候就沒有感受到上面說的問題。
您可能感興趣的文章:
- mysql 存儲(chǔ)過程中變量的定義與賦值操作
- mysql存儲(chǔ)過程 游標(biāo) 循環(huán)使用介紹
- MySQL存儲(chǔ)過程例子(包含事務(wù),輸出參數(shù),嵌套調(diào)用)
- MySQL存儲(chǔ)過程使用實(shí)例詳解
- MySQL 有輸入輸出參數(shù)的存儲(chǔ)過程實(shí)例
- mysql存儲(chǔ)過程 在動(dòng)態(tài)SQL內(nèi)獲取返回值的方法詳解
- MySQL 存儲(chǔ)過程中執(zhí)行動(dòng)態(tài)SQL語句的方法
- mysql存儲(chǔ)過程中使用游標(biāo)的實(shí)例
- 深入mysql創(chuàng)建自定義函數(shù)與存儲(chǔ)過程的詳解
- MySQL 存儲(chǔ)過程的基本用法介紹
- MySQL實(shí)現(xiàn)創(chuàng)建存儲(chǔ)過程并循環(huán)添加記錄的方法
相關(guān)文章
設(shè)置MySQL中的數(shù)據(jù)類型來優(yōu)化運(yùn)行速度的實(shí)例
這篇文章主要介紹了設(shè)置MySQL中索引的數(shù)據(jù)類型來優(yōu)化運(yùn)行速度的實(shí)例,主要是適當(dāng)使用短字節(jié)的數(shù)據(jù)類型來處理短索引,需要的朋友可以參考下2015-05-05
mysql安全啟動(dòng)腳本mysqld_safe詳細(xì)介紹
這篇文章主要介紹了mysql安全啟動(dòng)腳本mysqld_safe詳細(xì)介紹,mysqld_safe增加了一些安全特性,需要的朋友可以參考下2014-07-07
MySQL5.6 Replication主從復(fù)制(讀寫分離) 配置完整版
這篇文章主要介紹了MySQL5.6 Replication主從復(fù)制(讀寫分離) 配置完整版,需要的朋友可以參考下2016-04-04
達(dá)夢(mèng)數(shù)據(jù)庫獲取SQL實(shí)際執(zhí)行計(jì)劃方法詳細(xì)介紹
在達(dá)夢(mèng)數(shù)據(jù)庫中,使用EXPLAIN語句可以查看sql的執(zhí)行計(jì)劃,但EXPLAIN只生成執(zhí)行計(jì)劃,并不會(huì)真正執(zhí)行SQL語句,因此產(chǎn)生的執(zhí)行計(jì)劃有可能不準(zhǔn)。本章將帶領(lǐng)大家了解多種獲取SQL實(shí)際的執(zhí)行計(jì)劃的方法2022-10-10
MySQL 5.7并發(fā)復(fù)制隱式bug實(shí)例分析
這篇文章主要給大家介紹了關(guān)于MySQL 5.7并發(fā)復(fù)制隱式bug的相關(guān)資料,文中介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用mysql5.7具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-11-11
Mysql數(shù)據(jù)庫錯(cuò)誤代碼中文詳細(xì)說明
在mysql開發(fā)中出現(xiàn)錯(cuò)誤代碼各種各樣,下面我來給大家收集常用見的mysql使用過程中出錯(cuò)代碼的中文說明,希望些文章對(duì)各位朋友有所幫助了
2013-08-08 
