首頁 資料庫 mysql教程 oralce函数substr和mysql函数substring_index初记

oralce函数substr和mysql函数substring_index初记

Jun 07, 2016 pm 04:23 PM
mysql substr 函數

oralce函数substr和mysql函数substring_index小记 ???? 最近偶尔在mysql数据库中看到了substring_index函数,先简单介绍下mysql的substring_index函数。??? substring_index(str,delim,count) 返回字符串str中在第 count 个出现的分隔符 delim之前的子串,和ja

oralce函数substr和mysql函数substring_index小记

???? 最近偶尔在mysql数据库中看到了substring_index函数,先简单介绍下mysql的substring_index函数。???

substring_index(str,delim,count) 返回字符串str中在第 count 个出现的分隔符 delim之前的子串,和java中的str.substring函数类似。
如果count是正数时候,substring_index是从左到右的顺序检查分隔符delim所在的位置。
如果count是负数时候,substring_index是从右到左的顺序检查分隔符delim所在的位置,返回的是从字符串str左边下标1开始的下标位置。
特殊点如果count等于0,那么,不管delim是否在str中存在,你得到的将是一个空白字符串。如下所示
登入後複製

???

select substring_index('www.baidu.com', '.', 0);
select substring_index('www.baidu.com', '-', 0);
登入後複製

??? 上面说的很抽象,来个简单的例子。

???

select substring_index('www.baidu.com', '.', 2);
登入後複製

?? 2为正数,.在str中存在,返回的结果是从下标1到第2次.出现的位置之间substr(1,10-1)的子串。所以结果是'www.baidu'。

??

 select substring_index('www.baidu.com', '.', -2);
登入後複製

??? -2为负数,.在str中存在,.在str中搜索顺序是从右到左第2次出现的位置是4,用于是负数返回的是从最后到4+1的子串,也就是substr(str,4+1),所以结果是baidu.com。

??? 可以看到substring_index返回的子串是不包含分隔符 delim在str中第count次匹配的位置的,如果分隔符delim在str中出现的位置小于count次会发生什么呢?来测试下:

??

select substring_index('www.baidu.com', '.', 3);
select substring_index('www.baidu.com', '.', -3);
登入後複製

?? 运行之后可以看到返回的结果都是'www.baidu.com'。再来测试delim在str中不存在的情况:

??

select substring_index('www.baidu.com', '-', -1);
select substring_index('www.baidu.com', '-',1);
登入後複製

??? 运行之后可以看到返回的结果都是'www.baidu.com',从上面的测试可以知道分隔符delim在str中不存在或者出现次数小于指定的count时将返回str本身。上面只是测试了英文数据,中文数据测试也是一样的,测试如下:

??

select substring_index('我是中文_测试_数据','_',2) from dual;--我是中文_测试
select substring_index('我是中文_测试_数据','_',-2) from dual;--测试_数据
select substring_index('我是中文_测试_数据','_',0) from dual;--空白字符串
select substring_index('我是中文_测试_数据','_',3) from dual;--我是中文_测试_数据
登入後複製

??? 中英文混合的情况我就不测试了。

??? mysql有substring_index函数,oracle是否也要类似的函数呢?
我所知道的oracle中常见的只有substr和mysql的substring_index函数类似,有网友知道oralce其他函数的,麻烦留言告知。mysql中也有substr,关于两者的区别,后面我会简述。

??? 先简单的介绍下oracle的substr函数和instr函数,后面我会用这2个函数模拟mysql的substring_index函数

???

instr( string1, string2 [, start_position [, nth_appearance ] ] )
string1 源字符串
string2 目标字符串.
start_position 从string1 的哪个位置开始查找。默认为1. 字符串索引从1开始。为正,从左到右开始检索,为负,从右到左检索。
nth_appearance 代表要查找第几次出现的string2. 此参数可选,默认为 1.不能为负数
返回要查找的字符串string2在源字符串string1中的符合条件的开始索引
登入後複製

???

substr(string,start_position,length)
string 源字符串,即被截取的字符串.
start_position 字符截取的开始位置.start_position大于0时,从左边算起,小于0时,从右边查起
length 截取字符的个数.默认截取到最后一位.
登入後複製

??? 先来个测试:

???

select instr('www.baidu.com', '.',-1,2) from dual
登入後複製

??? .在str中存在,-1说明检索是从最后一位开始,2说明出现2次,返回的结果是4

??

select instr('www.baidu.com', '.',1,2) from dual
登入後複製

??? .在str中存在,1说明按从左到右的顺序,2说明出现2次,返回的结果是10

??? 下面来点特殊的测试,先测试nth_appearance,不是说不能为负数吗,我测试nth_appearance=0看下

???

select instr('www.baidu.com', '.',1,0) from dual
select instr('www.baidu.com', '.',-1,0) from dual
登入後複製

??? 你们猜结果是什么?结果是ORA-01428:参数'0'超出范围,说明nth_appearance不能为0,只能大于0。

??? 再测试start_position,令start_position=0看下

???

select instr('www.baidu.com', '.',0,1) from dual
select instr('www.baidu.com', '-',0,1) from dual
select instr('www.baidu.com', '.',0,2) from dual
select instr('www.baidu.com', '-',0,2) from dual
登入後複製

??? 测试结果都是0;如果把start_position改为>0,结果如下

???

select instr('www.baidu.com', '.',1,1) from dual--4
select instr('www.baidu.com', '-',1,1) from dual--0
select instr('www.baidu.com', '.',1,2) from dual--10
select instr('www.baidu.com', '-',1,2) from dual--0
登入後複製

??? 如果把start_position改为<0,结果如下:

???

select instr('www.baidu.com', '.',-1,1) from dual--10
select instr('www.baidu.com', '-',-1,1) from dual--0
select instr('www.baidu.com', '.',-1,2) from dual--4
select instr('www.baidu.com', '-',-1,2) from dual--0
select instr('www.baidu.com', '.',-1,3) from dual--0
select instr('www.baidu.com', '-',-1,3) from dual--0
select instr('www.baidu.com', '.',-1,3) from dual--0
select instr('www.baidu.com', '-',-1,3) from dual--0
登入後複製

??? 可以看出instr函数目标字符串在str中不存在时候或者出现次数小于给定的次数时,返回的都是0;

??? 使用中文测试也是一样:

???

select instr('我是中文_测试_数据','_',1,0) from dual;--ORA-01428:参数'0'超出范围
select instr('我是中文_测试_数据','_',1,2) from dual;--8
select instr('我是中文_测试_数据','_',1,3) from dual;--0
select instr('我是中文_测试_数据','_',-1,2) from dual;--5
select instr('我是中文_测试_数据','_',-1,3) from dual;--0
登入後複製

??? 下面测试下substr。

???

select substr('www.baidu.com',0,4) from dual
select substr('www.baidu.com',1,4) from dual
登入後複製

??? 上面2个执行结果都是'www.'。如果上面是在mysql下测试,结果如下:

???

select substr('www.baidu.com',0,4) from dual--空白字符串
select substr('www.baidu.com',1,4) from dual--www.
登入後複製

??? 从这里可以看出mysql中substr开始位置不能为0,而oracle下从0开始或者从1开始结果是一样的,下面继续oracle测试。

??

select substr('www.baidu.com',1) from dual
select substr('www.baidu.com',1,22) from dual
登入後複製

??? 返回结果是'www.baidu.com'。

??

select substr('www.baidu.com',-1) from dual
select substr('www.baidu.com',-1,4) from dual
select substr('www.baidu.com',-1,22) from dual
登入後複製

??? 返回的结果是'm',说明substr返回的是从开始位置到str最后一位。

??? 使用中文测试如下:

???

select substr('我是中文_测试_数据',1) from dual--我是中文_测试_数据
select substr('我是中文_测试_数据',1,22) from dual--我是中文_测试_数据
select substr('我是中文_测试_数据',-1) from dual--据
select substr('我是中文_测试_数据',-1,4) from dual--据
select substr('我是中文_测试_数据',-1,22) from dual--据
登入後複製

??? 从上面的测试可以看出oracle下instr函数和mysql的substring_index函数很像,instr函数已经可以得到下标了,结合substr可以截取子串返回。

??? Mysql的

???

select substring_index('www.baidu.com', '.', 2); 
登入後複製

??? Oralce可以这样做:

???

select substr('www.baidu.com', 1, instr('www.baidu.com', '.',1,2)-1) from dual
登入後複製

??? instr函数再-1是因为mysqlsubstring_index返回结果不包括.,上面返回的结果都是'www.baidu'。

???

??? Mysql的

???

select substring_index('www.baidu.com', '.', -2);
登入後複製

??? Oracle可以这样做:

???

select substr('www.baidu.com', instr('www.baidu.com', '.',-1,2) + 1, length('www.baidu.com'))from dual
select substr('www.baidu.com', instr('www.baidu.com', '.',-1,2) + 1) from dual
登入後複製

??? 上面可以不使用length函数,因为oracle的substr默认截取到str最后一位。

???? 中文测试如下:
???? Mysql下面:

????

select substring_index('我是中文_测试_数据','_',1) from dual;--我是中文
select substring_index('我是中文_测试_数据','_',2) from dual;--我是中文_测试
select substring_index('我是中文_测试_数据','_',-2) from dual;--测试_数据
select substring_index('我是中文_测试_数据','_',0) from dual;--空白字符串
select substring_index('我是中文_测试_数据','_',3) from dual;--我是中文_测试_数据
select substring_index('我是中文_测试_数据','.',3) from dual;--我是中文_测试_数据
登入後複製

??? Oracle下面:

???

select substr('我是中文_测试_数据', 1, instr('我是中文_测试_数据', '_',1,1)-1) from dual--我是中文
select substr('我是中文_测试_数据', 1, instr('我是中文_测试_数据', '_',1,2)-1) from dual--我是中文_测试
select substr('我是中文_测试_数据', instr('我是中文_测试_数据', '_',-1,2) + 1) from dual--测试_数据
select substr('我是中文_测试_数据', 1, instr('我是中文_测试_数据', '_',1,0)-1) from dual--ORA-01428:参数'0'超出范围
select substr('我是中文_测试_数据', 1, instr('我是中文_测试_数据', '_',1,3)-1) from dual--空白
select substr('我是中文_测试_数据', 1, instr('我是中文_测试_数据', '.',1,3)-1) from dual--空白
登入後複製

??? 可以看到我写的这个有点问题,如果想在nth_appearance(>0)大于目标串在str中出现的次数时和mysql一样返回整个字符串,可以这样写:

???

 select substr('我是中文_测试_数据',
              1,
              (select (case
                        when instr('我是中文_测试_数据', '_', 1, 3) <= 1 then
                         (select length('我是中文_测试_数据') from dual)
                        else
                         (instr('我是中文_测试_数据', '_', 1, 3) - 1)
                      end)
                 from dual))
  from dual
登入後複製

??? 这个在nth_appearance=0时用不了,进一步,考虑到nth_appearance=0时候返回NULL可以这样做,返回''这样的字符串使用substr做不了。

???

select substr('我是中文_测试_数据',
                1,
                (select (case
                          when 0 = 0 then
                           -1
                          else
                           case
                          when instr('我是中文_测试_数据', '_', 1, 3) <= 1 then
                           (select length('我是中文_测试_数据') from dual)
                          else
                           (instr('我是中文_测试_数据', '_', 1, 3) - 1)
                        end end)
                   from dual))
    from dual
登入後複製

??? 上面的sql是不是很丑陋,,我们可以定义一个存储过程封装一下。

????

create or replace procedure my_substring_index(str      in varchar2,
                                               delim    in varchar2,
                                               in_count in int,
                                               out_str  out varchar2) as
  v_time  int;
  v_delim varchar2(20);
begin
  if str is null then
    out_str := '空白字符串';
  elsif length(str) = 0 then
    out_str := '空白字符串';
  end if;

  if delim is null then
    v_delim := ' ';
  elsif length(delim) = 0 then
    v_delim := ' ';
  else
    v_delim := delim;
  end if;

  if in_count = 0 then
    out_str := '空白字符串';
  elsif in_count>0 then
   v_time := instr(str, delim, 1, in_count);
  else
   v_time := instr(str, delim, -1, (-1)*in_count);
  end if;
  

  if v_time = 0 then
    out_str := str;
  end if;
  
  if in_count>0  then
    if v_time = 1 then
      out_str := '空白字符串';
    elsif v_time > 1 then
      select substr(str, 1, v_time - 1) into out_str from dual;
    end if;
  elsif in_count<0 then
    if v_time >= 1 then
     select substr(str, v_time + 1) into out_str from dual;
     end if;
  end if;
end my_substring_index;
登入後複製

?? Oracle上面测试过程如下:

??? 在My Object下面的Procedures下找到存储过程my_substring_index,点击右键选择Test.出来Oracle的测试脚本:

???

begin
  -- Call the procedure
  my_substring_index(str => :str,
                     delim => :delim,
                     in_count => :in_count,
                     out_str => :out_str);
end;
登入後複製

?? 在测试脚本页面下面填上具体值,按F8,不出意外的化可以看到返回值。

?? 如果想自己写脚本可以这样,新建一个sql文件,内容如下:

??

declare
   v_str varchar2(50);
   v_delim varchar2(20);
   v_in_count number;
   v_out_str varchar2(20);
begin
  v_str:='&str';
  v_delim:='&delim';
  v_in_count:=&in_count;
  my_substring_index(str => v_str,
                     delim => v_delim,
                     in_count => v_in_count,
                     out_str => v_out_str);
  dbms_output.put_line('result='||v_out_str);
end;
登入後複製

??? 在命令行下运行:

???

SQL> set serveroutput on;
SQL> start f:/saveFile/test.sql
 16  /
 
result=www.baidu
PL/SQL procedure successfully completed
登入後複製

??? 如果想直接在cmd window下测试,可以这样:

???

SQL> set serveroutput on;
SQL> var v_str varchar2(50);
SQL> var v_delim varchar2(20);
SQL> var v_in_count number;
SQL> var v_out_str varchar2(50);
SQL> exec :v_str:='我是中文_测试_数据';
SQL> exec :v_delim:='_';
SQL> exec :v_in_count:=2;
SQL> exec my_substring_index(:v_str,:v_delim,:v_in_count,:v_out_str);
SQL> print v_out_str;
登入後複製

????? 以上是我写的oralce函数substr和mysql函数substring_index简单介绍,文章有点简单,但毕竟是原创,转载请注明出处。
????? 全文完,写的不好的地方请见谅,大神请绕过。

?

?

?

?

?

?

?

?

?

?

?

?

??

????

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1665
14
CakePHP 教程
1424
52
Laravel 教程
1322
25
PHP教程
1270
29
C# 教程
1249
24
laravel入門實例 laravel入門實例 Apr 18, 2025 pm 12:45 PM

Laravel 是一款 PHP 框架,用於輕鬆構建 Web 應用程序。它提供一系列強大的功能,包括:安裝: 使用 Composer 全局安裝 Laravel CLI,並在項目目錄中創建應用程序。路由: 在 routes/web.php 中定義 URL 和處理函數之間的關係。視圖: 在 resources/views 中創建視圖以呈現應用程序的界面。數據庫集成: 提供與 MySQL 等數據庫的開箱即用集成,並使用遷移來創建和修改表。模型和控制器: 模型表示數據庫實體,控制器處理 HTTP 請求。

MySQL和PhpMyAdmin:核心功能和功能 MySQL和PhpMyAdmin:核心功能和功能 Apr 22, 2025 am 12:12 AM

MySQL和phpMyAdmin是強大的數據庫管理工具。 1)MySQL用於創建數據庫和表、執行DML和SQL查詢。 2)phpMyAdmin提供直觀界面進行數據庫管理、表結構管理、數據操作和用戶權限管理。

MySQL與其他編程語言:一種比較 MySQL與其他編程語言:一種比較 Apr 19, 2025 am 12:22 AM

MySQL与其他编程语言相比,主要用于存储和管理数据,而其他语言如Python、Java、C 则用于逻辑处理和应用开发。MySQL以其高性能、可扩展性和跨平台支持著称,适合数据管理需求,而其他语言在各自领域如数据分析、企业应用和系统编程中各有优势。

解決數據庫連接問題:使用minii/db庫的實際案例 解決數據庫連接問題:使用minii/db庫的實際案例 Apr 18, 2025 am 07:09 AM

在開發一個小型應用時,我遇到了一個棘手的問題:需要快速集成一個輕量級的數據庫操作庫。嘗試了多個庫後,我發現它們要么功能過多,要么兼容性不佳。最終,我找到了minii/db,這是一個基於Yii2的簡化版本,完美地解決了我的問題。

laravel框架安裝方法 laravel框架安裝方法 Apr 18, 2025 pm 12:54 PM

文章摘要:本文提供了詳細分步說明,指導讀者如何輕鬆安裝 Laravel 框架。 Laravel 是一個功能強大的 PHP 框架,它 упростил 和加快了 web 應用程序的開發過程。本教程涵蓋了從系統要求到配置數據庫和設置路由等各個方面的安裝過程。通過遵循這些步驟,讀者可以快速高效地為他們的 Laravel 項目打下堅實的基礎。

解決MySQL模式問題:TheliaMySQLModesChecker模塊的使用體驗 解決MySQL模式問題:TheliaMySQLModesChecker模塊的使用體驗 Apr 18, 2025 am 08:42 AM

在使用Thelia開發電商網站時,我遇到了一個棘手的問題:MySQL模式設置不當,導致某些功能無法正常運行。經過一番探索,我找到了一個名為TheliaMySQLModesChecker的模塊,它能夠自動修復Thelia所需的MySQL模式,徹底解決了我的困擾。

MySQL:結構化數據和關係數據庫 MySQL:結構化數據和關係數據庫 Apr 18, 2025 am 12:22 AM

MySQL通過表結構和SQL查詢高效管理結構化數據,並通過外鍵實現表間關係。 1.創建表時定義數據格式和類型。 2.使用外鍵建立表間關係。 3.通過索引和查詢優化提高性能。 4.定期備份和監控數據庫確保數據安全和性能優化。

MySQL:解釋的關鍵功能和功能 MySQL:解釋的關鍵功能和功能 Apr 18, 2025 am 12:17 AM

MySQL是一個開源的關係型數據庫管理系統,廣泛應用於Web開發。它的關鍵特性包括:1.支持多種存儲引擎,如InnoDB和MyISAM,適用於不同場景;2.提供主從復制功能,利於負載均衡和數據備份;3.通過查詢優化和索引使用提高查詢效率。

See all articles