mysql 中用正則表達式如何取一個字符串中指定的字段, mysql 正則表達式 如何截取字符串中指定格式的字符
substring_index(input,split,index):input為要截取的字符,split為分隔符,Index為要截取第index個分隔符左(index為正)或右(index為負)的字符串。
舉例:
'Provider="RiskManagement" finalScore="65" RGID="100397278"' //獲取finalScore的值
1、獲取finalScore右邊的字符
select substring_index('Provider="RiskManagement" finalScore="65" RGID="100397278"','finalScore="',-1);
2、再獲取" RGID="左邊的字符
select substring_index(substring_index('Provider="RiskManagement" finalScore="65" RGID="100397278"','finalScore="',-1),'" RGID="',1);
擴展資料
MySQL 字符串截取函數(shù):left(), right(), substring(), substring_index()。還有 mid(), substr()。其中,mid(), substr() 等價于 substring() 函數(shù),substring() 的功能非常強大和靈活。
1、字符串截取:left(str, length)
mysql> select left('sqlstudy.com', 3);
| left('sqlstudy.com', 3) |
| sql |
2、字符串截取:right(str, length)
mysql> select right('sqlstudy.com', 3);
| right('sqlstudy.com', 3) |
| com |
代碼如下:
CREATE PROCEDURE sp_str
(
IN p_str VARCHAR(50), /*原始字符串*/
IN p_begin_str VARCHAR(50), /*要匹配的起始字符串*/
IN p_end_str VARCHAR(50)) /*要匹配的結(jié)束字符串*/
OUT p_result VARCHAR(50)) /*返回結(jié)果*/
NOT DETERMINISTIC
SQL SECURITY DEFINER
COMMENT ''
BEGIN
DECLARE m_len INT DEFAULT 0;
DECLARE m_index INT DEFAULT 0;
/*計算第一個匹配字符串的索引位置*/
select locate(p_begin_str,p_str)+char_length(p_begin_str) into m_index;
/*計算第一個匹配字符串的長度*/
select locate(p_end_str,p_str,m_index) into m_len;
select SUBSTRING(p_str,m_index,m_len-m_index) INTO p_result ;
END;
執(zhí)行:
CALL sp_str('[]abcd[12345]aa[]ss','abcd[',']',@result);
返回值 @result 為12345
call sp_str('[]abcd[sdww]aa[]ss','abcd[',']',@result);
返回值 @result 為sdww
如果不用存儲過程,可以直接寫sql語句實現(xiàn):
代碼如下:
select SUBSTRING(
']abcd[12345]111[]',
locate('abcd[',']abcd[12345]111[]')+CHAR_LENGTH('abcd['),
locate(']',']abcd[12345]111[]',CHAR_LENGTH('abcd['))-
(select locate('abcd[',']abcd[12345]111[]')+CHAR_LENGTH('abcd['))
)
返回值為 12345
關(guān)于mysql的函數(shù)介紹:
CHAR_LENGTH(str)
返回字符串str的長度。
LOCATE(substr,str)
POSITION(substr IN str)
返回子串substr在字符串str第一個出現(xiàn)的位置,如果substr不是在str里面,返回0.
mysql> select LOCATE('bar', 'foobarbar');
-> 4
mysql> select LOCATE('xbar', 'foobar');
-> 0
該函數(shù)是多字節(jié)可靠的。 LOCATE(substr,str,pos)
返回子串substr在字符串str第一個出現(xiàn)的位置,從位置pos開始。如果substr不是在str里面,返回0。
mysql> select LOCATE('bar', 'foobarbar',5);
-> 7
這函數(shù)是多字節(jié)可靠的。
SUBSTRING(str,pos,len)
SUBSTRING(str FROM pos FOR len)
MID(str,pos,len)
從字符串str返回一個len個字符的子串,從位置pos開始。使用FROM的變種形式是ANSI SQL92語法。
mysql> select SUBSTRING('Quadratically',5,6);
-> 'ratica'
該函數(shù)是多字節(jié)可靠的。
SUBSTRING(str,pos)
MySQL 一直以來都支持正則匹配,不過對于正則替換則一直到MySQL 8.0 才支持。對于這類場景,以前要么在MySQL端處理,要么把數(shù)據(jù)拿出來在應(yīng)用端處理。
比如我想把表y1的列str1的出現(xiàn)第3個action的子 串替換成dble,怎么實現(xiàn)?
1. 自己寫SQL層的存儲函數(shù)。代碼如下寫死了3個,沒有優(yōu)化,僅僅作為演示,MySQL 里非常不建議寫這樣的函數(shù)。
mysql
DELIMITER $$
USE `ytt`$$
DROP FUNCTION IF EXISTS `func_instr_simple_ytt`$$
CREATE DEFINER=`root`@`localhost` FUNCTION `func_instr_simple_ytt`(
f_str VARCHAR(1000), -- Parameter 1
f_substr VARCHAR(100), -- Parameter 2
f_replace_str varchar(100),
f_times int -- times counter.only support 3.
) RETURNS varchar(1000)
BEGIN
declare v_result varchar(1000) default 'ytt'; -- result.
declare v_substr_len int default 0; -- search string length.
set f_times = 3; -- only support 3.
set v_substr_len = length(f_substr);
select instr(f_str,f_substr) into @p1; -- First real position .
select instr(substr(f_str,@p1+v_substr_len),f_substr) into @p2; Secondary virtual position.
select instr(substr(f_str,@p2+ @p1 +2*v_substr_len - 1),f_substr) into @p3; -- Third virtual position.
if @p1 > 0 && @p2 > 0 && @p3 > 0 then -- Fine.
select
concat(substr(f_str,1,@p1 + @p2 + @p3 + (f_times - 1) * v_substr_len - f_times)
,f_replace_str,
substr(f_str,@p1 + @p2 + @p3 + f_times * v_substr_len-2)) into v_result;
else
set v_result = f_str; -- Never changed.
end if;
-- Purge all session variables.
set @p1 = null;
set @p2 = null;
set @p3 = null;
return v_result;
end;
$$
DELIMITER ;
-- 調(diào)用函數(shù)來更新:
mysql> update y1 set str1 = func_instr_simple_ytt(str1,'action','dble',3);
Query OK, 20 rows affected (0.12 sec)
Rows matched: 20 Changed: 20 Warnings: 0
mysqlmysql> select * from y1 into outfile '/var/lib/mysql-files/y1.csv';Query OK, 20 rows affected (0.00 sec)
shellroot@ytt-Aspire-V5-471G:/var/lib/mysql-files# sed -i 's/action/dble/3' y1.csv
mysql
mysql> truncate y1;
Query OK, 0 rows affected (0.99 sec)
mysql> load data infile '/var/lib/mysql-files/y1.csv' into table y1;
Query OK, 20 rows affected (0.14 sec)
Records: 20 Deleted: 0 Skipped: 0 Warnings: 0
mysqlmysql> update y1 set str1 = regexp_replace(str1,'action','dble',1,3) ;Query OK, 20 rows affected (0.13 sec)Rows matched: 20 Changed: 20 Warnings: 0
mysqlmysql> set @a = 'aa bb cc ee fi lucy 1 1 1 b s 2 3 4 5 2 3 5 561 19 10 10 20 30 10 40';Query OK, 0 rows affected (0.04 sec)
mysqlmysql> select regexp_instr(@a,'[:digit:]{2,}',1,2);+--------------------------------------+| regexp_instr(@a,'[:digit:]{2,}',1,2) |+--------------------------------------+| 50 |+--------------------------------------+1 row in set (0.00 sec)
mysql
mysql> set @a = '中國 美國 俄羅斯 日本 中國 北京 上海 深圳 廣州 北京 上海 武漢 東莞 北京 青島 北京';
Query OK, 0 rows affected (0.00 sec)
mysql> select regexp_instr(@a,'北京',1,1);
+-------------------------------+
| regexp_instr(@a,'北京',1,1) |
+-------------------------------+
| 17 |
+-------------------------------+
1 row in set (0.00 sec)
mysql> select regexp_instr(@a,'北京',1,2);
+-------------------------------+
| regexp_instr(@a,'北京',1,2) |
+-------------------------------+
| 29 |
+-------------------------------+
1 row in set (0.00 sec)
mysql> select regexp_instr(@a,'北京',1,3);
+-------------------------------+
| regexp_instr(@a,'北京',1,3) |
+-------------------------------+
| 41 |
+-------------------------------+
1 row in set (0.00 sec)
2. 導(dǎo)出來用sed之類的工具替換掉在導(dǎo)入,步驟如下:(推薦使用)
1)導(dǎo)出表y1的記錄。
2)用sed替換導(dǎo)出來的數(shù)據(jù)。
3)再次導(dǎo)入處理好的數(shù)據(jù),完成。
以上兩種還是推薦導(dǎo)出來處理好了再重新導(dǎo)入,性能來的高些,而且還不用自己費勁寫函數(shù)代碼。
那MySQL 8.0 對于以上的場景實現(xiàn)就非常簡單了,一個函數(shù)就搞定了。
還有一個regexp_instr 也非常有用,特別是這種特指出現(xiàn)第幾次的場景。比如定義 SESSION 變量@a。
拿到至少兩次的數(shù)字出現(xiàn)的第二次子串的位置。
那我們在看看對多字節(jié)字符支持如何。
那總結(jié)下,這里我提到了 MySQL 8.0 的兩個最有用的正則匹配函數(shù) regexp_replace 和 regexp_instr。針對以前類似的場景算是有一個完美的解決方案。
set @str := '{"brightness_current":0,"profile_currunt":1}';
set @find_str := '"brightness_current":';
select @find_str, value from (select @start_pos := locate(@find_str, @str), @end_pos := @start_pos + length(@find_str), @tail_pos := if(locate(",", @str, @end_pos) = 0, locate("}", @str, @end_pos), locate(",", @str, @end_pos)), substring(@str, @end_pos, @tail_pos - @end_pos) as value) as t;
輸出:
+-----------------------+-------+
| @find_str | value |
+-----------------------+-------+
| "brightness_current": | 0 |
+-----------------------+-------+
1 row in set (0.01 sec)
set @str := '{"brightness_current":0,"profile_currunt":1}';
set @find_str := '"profile_currunt":';
select @find_str, value from (select @start_pos := locate(@find_str, @str), @end_pos := @start_pos + length(@find_str), @tail_pos := if(locate(",", @str, @end_pos) = 0, locate("}", @str, @end_pos), locate(",", @str, @end_pos)), substring(@str, @end_pos, @tail_pos - @end_pos) as value) as t;
輸出:
+--------------------+-------+
| @find_str | value |
+--------------------+-------+
| "profile_currunt": | 1 |
+--------------------+-------+
1 row in set (0.01 sec)
確定是在mysql環(huán)境及SQL下用正則解析嗎?你貼的這是標準的JSON字符流啊、在應(yīng)用層解析會更好啊。否則你干脆substr還直接些。
相關(guān)評說:
鄰水縣剛性: ______ 代碼如下:CREATE PROCEDURE sp_str( IN p_str VARCHAR(50), /*原始字符串*/ IN p_begin_str VARCHAR(50), /*要匹配的起始字符串*/ IN p_end_str VARCHAR(50)) /*要匹配的結(jié)束字符串*/ OUT p_result VARCHAR(50)) /*返回結(jié)果*/ NOT ...
鄰水縣剛性: ______ MS SQL不支持正則表達式,可以用CLR實現(xiàn).1、 新建一個 MSSQL的數(shù)據(jù)庫項目,配置到你的數(shù)據(jù)庫中2、 在數(shù)據(jù)庫項目中新建一個函數(shù)庫,編寫如下代碼: /// <summary> /// 驗證是否符合正則表達式 /// </summary> [SqlFunction] public static ...
鄰水縣剛性: ______ mysql 只有正則匹配,而沒有正則替換功能的. 你需要先將要替換的數(shù)據(jù)select出來,然后讓PHP進行正則替換后再update到數(shù)據(jù)庫.
鄰水縣剛性: ______ 使用Python正則表達式可以做到,data = ''.join(re.findall(u'[0-9a-zA-Z\u4e00-\u9fa5]+', str1))
鄰水縣剛性: ______ 在正則表達式加兩個單詞邊界符,就不會匹配IS_USE字段了.正則表達式 (?i)\buse\b\s+\[?\s*(\w+)\s*\]?我給你個Java語言的例子,你看看吧.import java.util.regex.Matcher; import java.util.regex.Pattern; public class PM{ public static void main(...
鄰水縣剛性: ______ 具體SQL語句:select * from a where b REGEXP '^[0-9]' 如果想查詢出非數(shù)字開頭的,也可以用正則,并且使用NOT.具體命令如下:select * from a where b not REGEXP '^[0-9]'
鄰水縣剛性: ______ ok,這個問題還不怎么難,就是稍微饒了一點點,在我做題目之前,我希望樓主能自己先看看API幫助文檔關(guān)于String類的說明:提供兩種方法,都比較簡陋,考慮不全.1、 String sql = "一月租金500元,二月租金600元,三月租金1000元"; ...
鄰水縣剛性: ______ 正則表達式語法 一個正則表達式就是由普通字符(例如字符 a 到 z)以及特殊字符(稱為元字符)組成的文字模式.該模式描述在查找文字主體時待匹配的一個或多個字符串.正則表達式作為一個模板,將某個字符模式與所搜索的字符串進行匹...
鄰水縣剛性: ______ 不知道你用的什么語言:以下為Java String str = "USE TEST";//標準 String str2 = "USE [test]";//加[] String str3 = "use [TEST];";//小寫 +[] + ; System.out.println(str.replaceAll("(?i)use\\s+","").replaceAll("[\\[\\];]","")); System....
鄰水縣剛性: ______ oracel 數(shù)據(jù)庫查詢某一個表中是否有小寫字母的sql語句用正則表達式 用 regexp_like這個函數(shù)來解決,正則表達式為:[[:punct:]]+這個正則可以找出任何標點符號. 查詢value中包含任何標點符號的記錄如下: select * from xxx where regexp_like(value,'[[:punct:]]+');