今天在用sql模糊查詢包含字母d的時(shí)候,發(fā)現(xiàn)一些不包含此字母的也被查詢出來了:
SELECT * FROM custom WHERE custom_realname LIKE ' %d% '
查詢了一下,發(fā)現(xiàn)以下說明:
(最后修改為:
SELECT * FROM custom WHERE custom_realname LIKE BINARY ' %d% '
)
?
今天在做mysql的一個(gè)搜索的時(shí)候發(fā)現(xiàn)我用 select name from contact where name like '%a%'的時(shí)候出來的結(jié)果除了包含a的名字外連包含中文“新”的名字也出現(xiàn)在搜索結(jié)果里面,這令我想弄清楚mysql的匹配模式和規(guī)則到底是怎么樣的,所以決定查查資料了解了解,另外在匹配的時(shí)候正則表達(dá)式也很常用!所以準(zhǔn)備在這里記錄我學(xué)習(xí)這兩個(gè)玩意的收獲!
出現(xiàn)這個(gè)問題的原因是:MySQL在查詢字符串時(shí)是大小寫不敏感的,在編繹MySQL時(shí)一般以ISO-8859字符集作為默認(rèn)的字符集,因此在比較過程中中文編碼字符大小寫轉(zhuǎn)換造成了這種現(xiàn)象。
解決辦法:
1
.在建表的時(shí)候?qū)τ诎形牡淖侄渭由稀癇INARY”屬性,使之進(jìn)行二進(jìn)制比較,例如講"name char(10)"改成"name char(10) BINARY"。但是這樣你對該表的該字段進(jìn)行匹配的時(shí)候是區(qū)分大小寫的。
2
.如果使用源碼編譯MySQL,可以在編譯的時(shí)候使用--with--charset=gbk參數(shù),這樣mysql就直接支持中文查找和排序。
3
.使用mysql的locate函數(shù)來判斷。如:
SELECT * FROM table WHERE locate(substr,str)>0 ;
locate()有兩個(gè)形式:
LOCATE(
,?
substr
,
str
)
LOCATE(
substr
,
str
,
pos
)。
返回substr在str中的位置,如果str不包含substr返回0。這個(gè)函數(shù)也是不區(qū)分大小寫的。
4
.這樣使用sql語句:SELECT * FROM TABLE WHERE FIELDS LIKE BINARY '%FIND%',但是這和1一樣是區(qū)分大小寫的如果你想進(jìn)行不區(qū)分大小寫的查詢的時(shí)候就要使用upper或者lower進(jìn)行轉(zhuǎn)換。
正則表達(dá)式是為復(fù)雜搜索指定模式的強(qiáng)大方式。
5
.使用binary和ucase函數(shù)及concat函數(shù)。ucase是講英文全部轉(zhuǎn)換大寫,concat對字符串進(jìn)行連接。新的sql語句如下:
select id,title,name from achech_com.news where?binary ucase(title)?like?concat('%',ucase('a'),'%')?
也可以寫為select id,title,name from achech_com.news where binary ucase(title) like ucase('%a%')?
檢索的結(jié)果還算滿意吧,不過速度可能會(huì)因此而慢N毫秒喔。 因?yàn)槭褂胠ike和%進(jìn)行匹配的話對效率會(huì)有一定的影響。
正則表達(dá)式:
^?
所匹配的字符串以后面的字符串開頭?
mysql> select "fonfo" REGEXP "^fo$"; -> 0(表示不匹配)?
mysql> select "fofo" REGEXP "^fo"; -> 1(表示匹配)?
$?
所匹配的字符串以前面的字符串結(jié)尾?
mysql> select "fono" REGEXP "^fono$"; -> 1(表示匹配)?
mysql> select "fono" REGEXP "^fo$"; -> 0(表示不匹配)?
.
匹配任何字符(包括新行)?
mysql> select "fofo" REGEXP "^f.*"; -> 1(表示匹配)?
mysql> select "fonfo" REGEXP "^f.*"; -> 1(表示匹配)?
a*?
匹配任意多個(gè)a(包括空串)?
mysql> select "Ban" REGEXP "^Ba*n"; -> 1(表示匹配)?
mysql> select "Baaan" REGEXP "^Ba*n"; -> 1(表示匹配)?
mysql> select "Bn" REGEXP "^Ba*n"; -> 1(表示匹配)
?
a+
匹配1個(gè)或多個(gè)a字符的任何序列。
mysql> select "Ban" REGEXP "^Ba+n"; -> 1(表示匹配)?
mysql> select "Bn" REGEXP "^Ba+n"; -> 0(表示不匹配)
a??
匹配一個(gè)或零個(gè)a?
mysql> select "Bn" REGEXP "^Ba?n"; -> 1(表示匹配)?
mysql> select "Ban" REGEXP "^Ba?n"; -> 1(表示匹配)?
mysql> select "Baan" REGEXP "^Ba?n"; -> 0(表示不匹配)
de|abc?
匹配de或abc?
mysql> select "pi" REGEXP "pi|apa"; -> 1(表示匹配)?
mysql> select "axe" REGEXP "pi|apa"; -> 0(表示不匹配)?
mysql> select "apa" REGEXP "pi|apa"; -> 1(表示匹配)?
mysql> select "apa" REGEXP "^(pi|apa)$"; -> 1(表示匹配)?
mysql> select "pi" REGEXP "^(pi|apa)$"; -> 1(表示匹配)?
mysql> select "pix" REGEXP "^(pi|apa)$"; -> 0(表示不匹配)
(abc)*?
匹配任意多個(gè)abc(包括空串)?
mysql> select "pi" REGEXP "^(pi)*$"; -> 1(表示匹配)?
mysql> select "pip" REGEXP "^(pi)*$"; -> 0(表示不匹配)?
mysql> select "pipi" REGEXP "^(pi)*$"; -> 1(表示匹配)
{1} {2,3}?
這是一個(gè)更全面的方法,它可以實(shí)現(xiàn)前面好幾種保留字的功能?
a*?
可以寫成a{0,}?
a?
可以寫成a{1,}?
a??
可以寫成a{0,1}?
在{}內(nèi)只有一個(gè)整型參數(shù)i,表示字符只能出現(xiàn)i次;在{}內(nèi)有一個(gè)整型參數(shù)i,?
后面跟一個(gè)“,”,表示字符可以出現(xiàn)i次或i次以上;在{}內(nèi)只有一個(gè)整型參數(shù)i,?
后面跟一個(gè)“,”,再跟一個(gè)整型參數(shù)j,表示字符只能出現(xiàn)i次以上,j次以下?
(包括i次和j次)。其中的整型參數(shù)必須大于等于0,小于等于 RE_DUP_MAX(默認(rèn)是25?
5)。 如果同時(shí)給定了m和n,m必須小于或等于n.
[a-dX],?[^a-dX]
匹配任何是(或不是,如果使用^的話)a、b、c、d或X的字符。兩個(gè)其他字符之間的“-”字符構(gòu)成一個(gè)范圍,與從第1個(gè)字符開始到第2個(gè)字符之間的所有字符匹配。例如,[0-9]匹配任何十進(jìn)制數(shù)字 。要想包含文字字符“]”,它必須緊跟在開括號(hào)“[”之后。要想包含文字字符“-”,它必須首先或最后寫入。對于[]對內(nèi)未定義任何特殊含義的任何字符,僅與其本身匹配。
mysql> select "aXbc" REGEXP "[a-dXYZ]"; -> 1(表示匹配)?
mysql> select "aXbc" REGEXP "^[a-dXYZ]$"; -> 0(表示不匹配)?
mysql> select "aXbc" REGEXP "^[a-dXYZ] $"; -> 1(表示匹配)?
mysql> select "aXbc" REGEXP "^[^a-dXYZ] $"; -> 0(表示不匹配)?
mysql> select "gheis" REGEXP "^[^a-dXYZ] $"; -> 1(表示匹配)?
mysql> select "gheisa" REGEXP "^[^a-dXYZ] $"; -> 0(表示不匹配)
[[.characters.]]?
表示比較元素的順序。在括號(hào)內(nèi)的字符順序是唯一的。但是括號(hào)中可以包含通配符,?
所以他能匹配更多的字符。舉例來說:正則表達(dá)式[[.ch.]]*c匹配chchcc的前五個(gè)字符?
。
[=character_class=]?
表示相等的類,可以代替類中其他相等的元素,包括它自己。例如,如果o和( )是?
一個(gè)相等的類的成員,那么[[=o=]]、[[=( )=]]和[o( )]是完全等價(jià)的。?
[:character_class:]?
在括號(hào)里面,在[:和:]中間是字符類的名字,可以代表屬于這個(gè)類的所有字符。?
字符類的名字有: alnum、digit、punct、alpha、graph、space、blank、lower、uppe?
r、cntrl、print和xdigit?
mysql> select "justalnums" REGEXP "[[:alnum:]] "; -> 1(表示匹配)?
mysql> select "!!" REGEXP "[[:alnum:]] "; -> 0(表示不匹配)?
alnum |
文字?jǐn)?shù)字字符 |
alpha |
文字字符 |
blank |
空白字符 |
cntrl |
控制字符 |
digit |
數(shù)字字符 |
graph |
圖形字符 |
lower |
小寫文字字符 |
|
圖形或空格字符 |
punct |
標(biāo)點(diǎn)字符 |
space |
空格、制表符、新行、和回車 |
upper |
大寫文字字符 |
xdigit |
十六進(jìn)制數(shù)字字符 |
[[:<:]]?
[[:>:]]?
分別匹配一個(gè)單詞開頭和結(jié)尾的空的字符串,這個(gè)單詞開頭和結(jié)尾都不是包含在alnum中?
的字符也不能是下劃線。?
mysql> select "a word a" REGEXP "[[:<:]]word[[:>:]]"; -> 1(表示匹配)?
mysql> select "a xword a" REGEXP "[[:<:]]word[[:>:]]"; -> 0(表示不匹配)?
mysql> select "weeknights" REGEXP "^(wee|week)(knights|nights)$"; -> 1(表示?
匹配)?
?
?
要想在正則表達(dá)式中使用特殊字符的文字實(shí)例,應(yīng)在其前面加上2個(gè)反斜杠“/”字符。MySQL解析程序負(fù)責(zé)解釋其中一個(gè),正則表達(dá)式庫負(fù)責(zé)解釋另一個(gè)。例如,要想與包含特殊字符“+”的字符串“1+2”匹配,在下面的正則表達(dá)式中,只有最后一個(gè)是正確的:
mysql> SELECT '1+2' REGEXP '1+2'; -> 0
mysql> SELECT '1+2' REGEXP '1/+2'; -> 0
mysql> SELECT '1+2' REGEXP '1//+2'; -> 1
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
