有兩種方法,一種方法使用mysql的check table和repair table 的sql語句,另一種方法是使用MySQL提供的多個myisamchk, isamchk數據檢測恢復工具。前者使用起來比較簡便。推薦使用。
創新互聯建站提供高防服務器租用、云服務器、香港服務器、內江服務器托管等
1. check table 和 repair table
登陸mysql 終端:
mysql -uxxxxx -p dbname
check table tabTest;
如果出現的結果說Status是OK,則不用修復,如果有Error,可以用:
repair table tabTest;
進行修復,修復之后可以在用check table命令來進行檢查。在新版本的phpMyAdmin里面也可以使用check/repair的功能。
2. myisamchk, isamchk
其中myisamchk適用于MYISAM類型的數據表,而isamchk適用于ISAM類型的數據表。這兩條命令的主要參數相同,一般新的系統都使用MYISAM作為缺省的數據表類型,這里以myisamchk為例子進行說明。當發現某個數據表出現問題時可以使用:
myisamchk tablename.MYI
進行檢測,如果需要修復的話,可以使用:
myisamchk -of tablename.MYI
關于myisamchk的詳細參數說明,可以參見它的使用幫助。需要注意的時在進行修改時必須確保MySQL服務器沒有訪問這個數據表,保險的情況下是最好在進行檢測時把MySQL服務器Shutdown掉。
MySQL 存儲過程中,使用游標查詢,返回的是結果集時,如何查看調用存儲過程輸出結果呢?
解決方案:存儲過程不返回數據,但它能創建和填充另一個表。所以在存儲過程運行中創建臨時表。該臨時表將保存存儲過程中生成的結果集,在遍歷游標時,用insert保存每條數據到臨時表中。后續調用時可以用select語句查詢臨時表中的存儲過程運行結果。
以下有 三種方式 使用游標創建一個存儲過程,統計某一部門下的員工信息
方法一:Loop循環
調用存儲過程:
方法二:While 循環
調用存儲過程:
方法三:REPEAT 循環
調用存儲過程:
上述三種實現方法在測試過程中遇到下述問題。
調用存儲過程查詢臨時表輸出結果時,會發現多循環了一次,像這樣:
解決方法:
在遍歷游標查詢結果時,先判斷游標的結束標志(done) 是否是為1,如果不是1,則向臨時表中插入數據。
CREATE? PROCEDURE `gk_test`()
COMMENT '測試數據'? -- 函數注釋
BEGIN
DECLARE done INT DEFAULT FALSE;?-- 定義遍歷數據結束標志。非游標變量,需要定義在游標變量前
DECLARE m_id CHAR(32);?-- 定義接收游標數據中id變量
DECLARE m_or VARCHAR(255);?--?定義接收游標數據中orderno變量
DECLARE m_c INT(11);??--?定義接收游標數據中count變量
DECLARE m_soult CURSOR FOR SELECT `id`, `orderno`, count(*) t FROM hb_shipper_order GROUP BY orderno HAVING t 1;???-- 定義游標變量,即接收查詢結果集
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;??-- 將結束標志綁定到游標
OPEN m_soult;? ??-- 打開游標
read_loop:LOOP? ?-- 開始循環。這個是mysql 3種循環中的1種。
FETCH m_soult INTO m_id,m_or,m_c;???-- 提取游標數據
IF done THEN? ?-- 判斷,當為true時
LEAVE read_loop;? ?-- 跳出循環
END IF;
-- 處理事 自己想處理的事
UPDATE `hb_shipper_waybill` SET
ordernos = REPLACE (`ordernos`, m_or, CONCAT(m_or,'-2')),
goodsname = REPLACE (`goodsname`, m_or, CONCAT(m_or,'-2'))
WHERE `id` IN ( SELECT `waybillid` FROM `hb_shipper_order_waybill_relation` WHERE orderid = m_id);
UPDATE `hb_shipper_order` SET orderno = CONCAT(m_or,'-2') WHERE id = m_id;
END LOOP;? -- 結束循環
-- 關閉游標
CLOSE m_soult;
END
上圖:
DEMO: 2? ? // 通過 存儲過程,處理? copy 數據并處理對應數據關系
delimiter $$
DROP PROCEDURE IF EXISTS gk;
CREATE DEFINER =? PROCEDURE `gk`(IN pname varchar(255))
COMMENT '測試'
BEGIN
DECLARE new_id CHAR(32);
DECLARE m_id CHAR(32);
DECLARE new_noids VARCHAR(255);
DECLARE done INT DEFAULT 0;
DECLARE m_result CURSOR FOR select id from `xxx`;
DECLARE m_result2 CURSOR FOR select id from `xxx`;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = 1;
DROP TABLE IF EXISTS `xxx`;
CREATE TABLE `xxx` (
`id` char(32) NOT NULL COMMENT 'ID',
`subsystem` char(32) NOT NULL COMMENT '',
`name` varchar(20) NOT NULL COMMENT '',
`type` varchar(15) NOT NULL COMMENT '',
`url` varchar(200) DEFAULT NULL COMMENT '',
`icon` varchar(20) DEFAULT NULL COMMENT '',
`target` varchar(10) NOT NULL DEFAULT '_self' COMMENT '',
`method` mediumtext COMMENT '',
`orderby` smallint(3) NOT NULL DEFAULT '0' COMMENT '',
`parentid` char(32) NOT NULL DEFAULT '0' COMMENT '',
`relateid` char(32) NOT NULL COMMENT '',
`isopen` tinyint(1) NOT NULL DEFAULT '1' COMMENT '',
`nodeids` varchar(255) DEFAULT NULL COMMENT '',
`nodenum` smallint(3) NOT NULL DEFAULT '0' COMMENT '',
`createtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '',
`createuser` char(32) NOT NULL DEFAULT '0' COMMENT '',
`updatetime` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00' COMMENT '',
`udateuser` char(32) NOT NULL DEFAULT '0' COMMENT '',
`deleted` tinyint(1) DEFAULT '0' COMMENT '',
`status` tinyint(1) NOT NULL DEFAULT '0' COMMENT '',
`isdefault` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '',
? ? ? ? ? ? `isdisable` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '',
`projecttype` varchar(255) NOT NULL COMMENT '',
? ? ????PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=COMPACT;
INSERT INTO `xxxx` SELECT * FROM `aaa` WHERE `projecttype` = 'T';
UPDATE `xxx` SET `projecttype` = pname;
OPEN m_result;
REPEAT
FETCH m_result INTO m_id;
SET new_id = UPPER(REPLACE(UUID(),'-',''));
UPDATE `xxx` SET `id` = new_id WHERE `id` = m_id;
UPDATE `xxx` SET `parentid` = new_id WHERE `parentid` = m_id;
UNTIL done END REPEAT;
CLOSE m_result;
SET done = 0;
OPEN m_result2;
REPEAT
FETCH m_result2 INTO m_id;
SELECT CONCAT(',', GROUP_CONCAT(`id`)) INTO new_noids FROM `xxx` WHERE `parentid` = m_id;
UPDATE `xxx` SET `nodeids` = new_noids WHERE `id` = m_id;
UNTIL done END REPEAT;????
CLOSE m_result2;
INSERT INTO `aaa` SELECT * FROM `xxx`;
DROP TABLE IF EXISTS xxx;
END $$
delimiter ;
從字面可以這么理解什么是游標,游標就像是水面上漂浮的一個標記,這個標記可以來回游動,一會游到這里一會游到那里,這里的河水可以理解為是數據的集合,這個標記就是在這些數據間來回游動。
為什么 MySQL 會有游標這個概念,由于 SQL 語言是面向集合的語句,它每次查詢出來都是一堆數據的集合,沒有辦法對其中一條記錄進行單獨的處理。如果要對每條記錄進行單獨處理就需要游標。
游標其實就像是編程語言中的 for/foreach 循環,把一個數組(數據的集合)中每條數據一條一條地循環出來,然后你在 for/foreach 循環中使用判斷語句對你感興趣的數據進行處理。
哪里可以使用游標呢,函數,存儲過程,觸發器中都可以使用。
說完概念,就來看下游標的固定寫法。不管概念是否理解,記住下面的固定模式也可以完成搬磚任務。
1、聲明游標
SELECT 語句就是正常的查詢語句,例如:SELECT id,age FROM table;
2、打開游標
在打開游標之前,游標定義的 SQL 語句是不執行的。
3、取出記錄
將當前的記錄數據存入變量。
當 FETCH 沒有找到記錄時會拋出異常,異常的定義需要下面的 HANDLER FOR 語句。
聲明游標語句中的 SELECT 如果有多個字段,INTO 后面需要多個變量進行接收。
4、設置結束條件
這個語句的作用是指定一個條件,告訴程序所有數據已經循環完畢,可以結束了。由于游標是使用 WHILE 循環進行每條數據的讀取,就需要給 WHILE 一個結束條件。
處理種類:可以是, EXIT 立即結束。CONTINUE 繼續下面的處理。
異常的類型:一般指定為 NOT FOUND ,意思是沒有找到任何數據。
異常發生時的處理:當異常發生時需要做的事情,這里一般改變一個變量的值來記錄異常已經發生了,如如 SET flat = 1 詳細用法查看下面的例子。
5、關閉游標
實戰代碼:
完畢,看懂沒,如果沒看懂沒關系,游標處理是一套固定的格式,按照上面例子中固定的格式套入到你的程序就可以了。
在數據庫中,游標是一個十分重要的概念。游標提供了一種對從表中檢索出的數據進行操作的靈活手段,就本質而言,游標實際上是一種能從包括多條數據記錄的結果集中每次提取一條記錄的機制。游標總是與一條SQL??選擇語句相關聯因為游標由結果集(可以是零條、一條或由相關的選擇語句檢索出的多條記錄)和結果集中指向特定記錄的游標位置組成。當決定對結果集進行處理時,必須聲明一個指向該結果集的游標。如果曾經用?C?語言寫過對文件進行處理的程序,那么游標就像您打開文件所得到的文件句柄一樣,只要文件打開成功,該文件句柄就可代表該文件。對于游標而言,其道理是相同的。可見游標能夠實現按與傳統程序讀取平面文件類似的方式處理來自基礎表的結果集,從而把表中數據以平面文件的形式呈現給程序。
我們知道關系數據庫管理系統實質是面向集合的,在MS?SQL?SERVER?中并沒有一種描述表中單一記錄的表達形式,除非使用where??子句來限制只有一條記錄被選中。因此我們必須借助于游標來進行面向單條記錄的數據處理。由此可見,游標允許應用程序對查詢語句select??返回的行結果集中每一行進行相同或不同的操作,而不是一次對整個結果集進行同一種操作;它還提供對基于游標位置而對表中數據進行刪除或更新的能力;而且,正是游標把作為面向集合的數據庫管理系統和面向行的程序設計兩者聯系起來,使兩個數據處理方式能夠進行溝通。
在數據庫開發過程中,當你檢索的數據只是一條記錄時,你所編寫的事務語句代碼往往使用SELECT?INSERT?語句。但是我們常常會遇到這樣情況,即從某一結果集中逐一地讀取一條記錄。那么如何解決這種問題呢?游標為我們提供了一種極為優秀的解決方案——那就是使用游標。
先聲明一個游標,語法:
DECLARE v_1 VARCHAR(16);
DECLARE v_2 VARCHAR(16);
DECLARE c_XXX CURSOR FOR
SELECT c_1, c_2
FROM t_1;
同時聲明一個布爾型的變量FOUND,當循環的條件不成立時,結束循環用的。
DECLARE FOUND BOOLEAN DEFAULT TRUE;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET FOUND = FALSE;
在使用游標的時候,要先打開游標,語法:OPEN c_XXX;
使用的時候,要先打開游標,取出第一條數據,語法: FETCH c_XXX INTO v_1, v_2;
然后才開始執行循環,語法:WHILE FOUND DO
執行需要執行的語句;
取下一條數據放到當前游標中,FETCH c_XXX INTO v_1, v_2;
結束循環,語法:END WHILE;
結束游標,語法:CLOSE c_cargo。有什么問題我們再溝通啊。不知道你具體要問題的是什么。
本文名稱:mysql怎么使用游標,java mysql 游標
路徑分享:http://vcdvsql.cn/article0/dsiigio.html
成都網站建設公司_創新互聯,為您提供搜索引擎優化、服務器托管、網站維護、品牌網站制作、網站策劃、品牌網站建設
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯