欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

mysql怎么調(diào)試函數(shù),mysql函數(shù)怎么調(diào)用

mysql udf函數(shù)怎么調(diào)用

1)基本用戶定義函數(shù)是一類代碼,對MYSQL服務器功能進行擴充,通過添加新函數(shù),性質(zhì)就象使用本地MYSQL函數(shù)abs()或concat().UDF是用C(或C++)寫的。也許還可以用BASIC,.NET或其它什么雖然還沒見過有人這么做。

公司主營業(yè)務:網(wǎng)站建設、網(wǎng)站制作、移動網(wǎng)站開發(fā)等業(yè)務。幫助企業(yè)客戶真正實現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)是一支青春激揚、勤奮敬業(yè)、活力青春激揚、勤奮敬業(yè)、活力澎湃、和諧高效的團隊。公司秉承以“開放、自由、嚴謹、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領域給我們帶來的挑戰(zhàn),讓我們激情的團隊有機會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)推出蘇仙免費做網(wǎng)站回饋大家。

2)從字面上何以知道UDF是很有用的,尤其當需要擴展MYSQL服務器功能時。下表給出了最佳解決方法的比較:

Method Speed Language Development

方法 速度 語言 開發(fā)

Stored Procedures slow SQL ~minutes (for small functions)

存儲過程 慢 SQL ~分鐘(對于小函數(shù))

UDF fast C ~hour

UDF 快 C ~小時

Native Function fast C major pain in the ***

本地函數(shù) 快 C 未知

慢的意思是和其它比較時。存儲過程和一般SQL語句比仍然是很快的。

對本地函數(shù)的一點解釋:本質(zhì)上和UDF沒太大區(qū)別。但是必須用MYSQL的資源代碼來寫然后重新編譯全部。這將是很大的工作量,必須一邊又一邊的用最新版的MYSQL來完成這項工作。

3)這部分很簡單。當完成了一個UDF,只是使用它就可以了。例如:"SELECT MyFunction(data1, data2) FROM table"

4)編寫UDF

現(xiàn)在開發(fā)寫一個UDF:

建立一個新的shared-library項目(該例中用的VC++ 6.0建立一個標準的DLL)

首先需要一些頭文件。這些頭文件是標準的頭文件和MYSQL服務器的包含目錄里的文件

#ifdef STANDARD

/* STANDARD is defined, don't use any mysql functions */

#include

#include

#include

#ifdef __WIN__

typedef unsigned __int64 ulonglong; /* Microsofts 64 bit types */

typedef __int64 longlong;

#else

typedef unsigned long long ulonglong;

typedef long long longlong;

#endif /*__WIN__*/

#else

#include

#include

#endif

#include

#include

static pthread_mutex_t LOCK_hostname;

現(xiàn)在必須決定需要哪類函數(shù)。本質(zhì)上有兩種選擇:

該函數(shù)是聚合函數(shù)嗎?(后面將學習很多關(guān)于聚合函數(shù)的內(nèi)容)

返回類型是什么?有4個選擇:

類型 描述

STRING 一個合法的字符串,轉(zhuǎn)換成char*類型

INTEGER 一個普通的整型變量,轉(zhuǎn)換成64位的整型

REAL型 一個俘點數(shù),轉(zhuǎn)換成double型

DECIAML型 這個并沒真正的結(jié)束,MYSQL將做字符串對待

現(xiàn)在開始討論關(guān)于非聚合函數(shù)。必須聲明并執(zhí)行一些MYSQL使用UDF時用到的函數(shù),但首先一些必要的結(jié)構(gòu)必須并確:

UDF_INIT:

類型 名稱 描述

my_bool maybe_null 是1如果函數(shù)能返回NULL

unsigned int decimals 針對REAL函數(shù)

unsigned long max_length 針對字符串函數(shù)

char * ptr 自由指針針對函數(shù)的數(shù)據(jù)

my_bool const_item 0如果結(jié)果是獨立的

UDF_ARGS:

類型 名稱 描述

unsigned int arg_count 成員數(shù)量

enum Item_result * arg_type 成員類型的數(shù)組

char ** args 指向成員的指針的數(shù)組

unsigned long * lengths 成員長度的數(shù)組(針對字符串)

char * maybe_null "maybe_null"標記的數(shù)組

char ** attributes 指向成員屬性的指針的數(shù)組

unsigned long * attribute_lengths 屬性長度數(shù)組

現(xiàn)在看一下該函數(shù):

De-/Initialization:

Collapseextern "C" my_bool MyTest_init(UDF_INIT *initid, UDF_ARGS *args,

char *message)

{

//非常重要的一件事是建立內(nèi)存

//需要

//需要一個很長的變量來保存檢測數(shù)

//雖然該例中不需要

longlong* i = new longlong; // 建立變量

*i = 0; // 設初值

//指針變量中保存為一個字符指針

//確認不會遇到類型問題

initid-ptr = (char*)i;

//檢測成員的格式

if (args-arg_count != 1)

{

strcpy(message,"MyTest() requires one arguments");

return 1;

}

if (args-arg_type[0] != INT_RESULT)

{

strcpy(message,"MyTest() requires an integer");

return 1;

}

return 0;

}

extern "C" void MyTest_deinit(UDF_INIT *initid)

{

//這里必須清空所分配的內(nèi)存

//引入函數(shù)

delete (longlong*)initid-ptr;

}

The actual function:

extern "C" longlong MyTest(UDF_INIT *initid, UDF_ARGS *args,

char *is_null, char *error)

{

/*最后這是實際的工作部分。該函數(shù)為每個記錄調(diào)用,返回值或指向當前值的指針保存在UDF_ARGS變量中。必須獲得值,完成計算并返回值。注意可以通過UDF_INIT變量進入MyTest_init中分配的內(nèi)存,該例中將為每個值設置為5

*/

return *((longlong*)args-args[0])+5;

}

全部完成!現(xiàn)在必須編譯連接庫,然后將其拷貝到操作系統(tǒng)可以加載的目錄下。通常在WINDOWS里是系統(tǒng)變量的定義路徑。個人使用的是MYSQL服務器的bin目錄。必須確認該目錄是其它MYSQL不能訪問的。然后確認所有MYSQL需要的函數(shù)功能。

必須告訴MYSQL,這必須直接了當:執(zhí)行以下SQL指令:

CREATE [AGGREGATE] FUNCTION MyTest

RETURNS [INTEGER|STRING|REAL|DECIMAL] SONAME the_libraries_exact_name

現(xiàn)在可以想使用其他函數(shù)一樣使用它了。

5)成員函數(shù):

現(xiàn)在說一下成員函數(shù)。當?shù)腢DF是個成員函數(shù),必須增加一些函數(shù),一些函數(shù)在不同的方式中使用。調(diào)用次序是:

調(diào)用yTest_init來分配內(nèi)存(就象一般的UDF一樣)

MYSQL將表分類是通過GROUP BY

每組里的第一行調(diào)用MyTest_clear

每組里的第一列調(diào)用 MyTest_add

在組改變后或最后一列改變后調(diào)用MyTest得到結(jié)果

重復3到5直到所有列被處理。

調(diào)用MyTest_deinit清空內(nèi)存

現(xiàn)在讓看一下新的聚合函數(shù)所需的函數(shù)。該例中將簡單的添加所有的值。(就象本地SUM函數(shù))

void MyTest_clear(UDF_INIT *initid, char *is_null, char *error)

{

/*為每個新組重新將總數(shù)設置為0,當然必須分配一個longlong類型變量在在init函數(shù)中,并分配給指針

*/

*((longlong*)initid-ptr) = 0;

}

void MyTest_add(UDF_INIT *initid, UDF_ARGS *args, char *is_null, char *error)

{

//為每列將當前值添加到總數(shù)

*((longlong*)initid-ptr) = *((longlong*)initid-ptr) +

*((longlong*)args-args[0]);

}

longlong MyTest(UDF_INIT *initid, UDF_ARGS *args, char *is_null, char *error)

{

//最后返回總值

return *((longlong*)initid-ptr);

}

6)更進一步的問題:

在寫一些復雜的UDF時需要注意幾個問題:

一個字符串函數(shù)應該返回一個指向結(jié)果的指針并且設置*result和*length作為目錄和返回值的長度值。例如:

memcpy(result, "result string", 13);

*length = 13;

MyTest建立的結(jié)果緩沖區(qū)是255字節(jié)。如果的結(jié)果保存在里面。不必擔心結(jié)果的內(nèi)存分配問題。

如果的字符串函數(shù)需要返回一個大于255字節(jié)長度的字符串。必須用malloc或新的MyTest_init或MyTest函數(shù)分配,然后用MyTest_deinit釋放它。能用UDF_INIT的指針保存分配的內(nèi)存地址,并在MyTest中重用。

在主函數(shù)中指定一個錯誤返回,設置 *error為1:如果MyTest()為任何列將*error設置為1,則函數(shù)的值是NULL針對于當前列,以及對任何的通過MyTest()被調(diào)用的聲明中并發(fā)的列請求。

想了解更多內(nèi)容看一下MYSQL在線幫助。

7)一些指導方針:

如果確實希望的UDF運行良好,這里有一些建議:)

不要在UDF中調(diào)用任何其他的程序或進程

不要保存任何的本地信息。(這些在普通的庫里已經(jīng)共享)

不要分配任何的全局或靜態(tài)的變量。

始終檢測成員的類型。就象MYSQL將所有類型都轉(zhuǎn)換為字符類型。如果將一個字符類型轉(zhuǎn)換成整型指針可能會出錯。

特別注意內(nèi)存的分配。如果有內(nèi)存泄漏問題會使服務器徹底崩潰!

8)調(diào)式UDF

調(diào)試UDF需要勇氣因為如果UDF有問題,每次都會使整個MYSQL服務器死掉。所以寫了一個命令行工具,來圍繞這個問題工作。僅僅運行它,它會模仿"SELECT"調(diào)用指令將結(jié)果保存到庫文件中,可以打印所有的結(jié)果行。所以當UDF存在一些錯誤只是該程序崩潰而不會是整個服務器。

如何在Windows下編譯或調(diào)試MySQL

用vs code 就可以了。

Visual Studio Code

Visual Studio Code(簡稱VS Code)是由微軟開發(fā)的,同時支持Windows、Linux和macOS操作系統(tǒng)的開源文本編輯器。它支持調(diào)試,內(nèi)置了Git 版本控制功能,同時也具有開發(fā)環(huán)境功能,例如代碼補全(類似于IntelliSense)、代碼片段、代碼重構(gòu)等。該編輯器支持用戶自定義配置,例如改變主題顏色、鍵盤快捷方式、編輯器屬性和其他參數(shù),還支持擴展程序并在編輯器中內(nèi)置了擴展程序管理的功能。

安裝LLDB

LLDB是LLVM編譯器的一部分,推薦使用Homebrew安裝LLVM工具集,不建議使用系統(tǒng)自帶的LLDB,安裝前必須先創(chuàng)建證書否則無法安裝,步驟如下:

創(chuàng)建完成后,開始安裝LLVM

brew install llvm --with-python@2?--with-lldb

安裝插件

VS Code自帶有debug功能,這里我推薦使用LLDB Debugger插件。

?接下來,為項目配置調(diào)試參數(shù)。

配置調(diào)試參數(shù)

使用VS Code打開MySQL源碼目錄,在側(cè)邊欄選擇debug欄目,添加配置,program輸入需要調(diào)試的程序路徑,這里選擇你編譯好的mysqld路徑,args輸入程序啟動所需的參數(shù),通常會指定mysqld的配置文件。這樣就配置好了,是不是很簡單。

啟動調(diào)試

點擊啟動按鈕,啟動后如果沒有設置斷點會mysqld會正常啟動,如果觸發(fā)了斷點會如下圖顯示。

整個調(diào)試窗口基本分為六部分,所有的調(diào)試操作都在這里完成:

1:?顯示變量信息

2:?設置重點關(guān)注的變量

3:?顯示調(diào)用棧信息

4:?設置斷點信息,在代碼行號前也可以設置斷點

5:?代碼顯示區(qū)域,上方是調(diào)試按鈕,包括 continue/stepover/step in/step out/restart/stop

6:?調(diào)試終端輸入輸出區(qū)

斷點設置

在代碼行號前點擊即可在該行為設置斷點,也可以根據(jù)條件設置斷點。以設置ConditionalBreakpoint為例,當程序啟動后會按照你設置的條件表達式判斷是否觸發(fā)斷點。

Conditional Breakpoint這種方式用在目標變量達到某條件時觸發(fā)斷點,其余則跳過繼續(xù)執(zhí)行。比如:設置變量等于目標表名時觸發(fā)斷點,其余表則跳過,相對函數(shù)名斷點省去很多手工跳過操作。

遠程調(diào)試

假如你想調(diào)試遠程Linux服務器上的MySQL上面的方法就不合適了,這時需要遠程調(diào)試。lldb和gdb都支持遠程調(diào)試,這里以lldb為例。

需要先在遠程主機上安裝lldb,使用yum安裝,源地址在這里

remote$ yum install -y llvm-toolset-7

安裝完成后,啟動lldb-server

remote$ /opt/rh/llvm-toolset-7/root/usr/bin/lldb-serverplatform --listen?"*:9191"?--server

接下來,在VS Code調(diào)試界面中新增配置項。

{

"type":?"lldb",

"request":?"attach",

"name":?"Remote attach",

"program":?"~/mysql5626/usr/local/mysql/bin/mysqld",

"pid":"target_pid",

"initCommands": [

"platform select remote-linux",

"platform connect connect://remote_host:9191"

],

"sourceMap": {

"/export/home/pb2/build/sb_0-15908961-1436910670.17/mysql-5.6.26":?"/Users/hongbin/workbench/mysql-server"

}

},

program:?本機也要拷貝一份目標程序,加載

pid:?填寫遠程主機的mysqld進程id

sourceMap:?填寫mysqld編譯的代碼路徑與本機代碼庫路徑的映射,這樣調(diào)試時代碼才可以和程序關(guān)聯(lián)在一起看

注意:記得調(diào)試前將代碼切換到與目標程序版本一致的branch

五、MYSQL存儲過程和函數(shù)

? create procedure用來創(chuàng)建 存儲過程 ,create function用來創(chuàng)建 函數(shù)

? Delimiter命令是改變語句的結(jié)束符 ,MySQL默認的結(jié)束符為;號,由于procedure和function中的;號并不代表創(chuàng)建的結(jié)束,所以要替換成另外的結(jié)束符以便表示創(chuàng)建的結(jié)束

? rontine_body子句可以包含一個簡單的SQL語句,也可以包含多個SQL語句, 通過begin…end將這多個SQL語句 包含在一起

? MySQL存儲過程和函數(shù)中也可以包含類似create和drop等DDL語句

? comment子句用來寫入對存儲過程和函數(shù)的注釋

? Language子句用來表示此存儲過程和函數(shù)的創(chuàng)建語言

? 存儲過程和函數(shù)被標注為deterministic表明當輸入相同的參數(shù)是會返回相同的結(jié)果,反之如果是not deterministic則表示相同參數(shù)不會是相同結(jié)果,默認是not deterministic

? 相關(guān)屬性短語只有咨詢含義,并不是強制性的約束

? Drop procedure/function語句用來 刪除指定名稱的存儲過程或函數(shù)

? Begin…end語句通常出現(xiàn)在存儲過程、函數(shù)和觸發(fā)器中,其中 可以包含一個或多個語句 ,每個語句用;號隔開

? 標簽label可以加在begin…end語句以及l(fā)oop, repeat和while語句

? 語句中通過iterate和leave來控制流程,iterate表示返回指定標簽位置,leave表示跳出標簽

? Declare語句通常用來聲明本地變量、游標、條件或者handler

? Declare語句只允許出現(xiàn)在begin … end語句中而且必須出現(xiàn)在第一行

? Declare的順序也有要求,通常是先聲明本地變量,再是游標,然后是條件和handler

? 本地變量可以通過declare語句進行聲明

? 聲明后的變量可以通過select … into var_list進行賦值,或者通過set語句賦值,或者通過定義游標并使用fetch … into var_list賦值

? 通過declare聲明變量方法:

? MySQL支持if,case,iterate,leave,loop,while,repeat語句作為存儲過程和函數(shù)中的 流程控制語句 ,另外return語句也是函數(shù)中的特定流程控制語句

? Case語句在存儲過程或函數(shù)中表明了 復雜的條件選擇語句

? IF語句在存儲過程或函數(shù)中表明了 基礎的條件選擇語句

其中在 function 里面,只有 DETERMINISTIC, NO SQL 和 READS SQL DATA 被支持。如果我們開啟了 bin-log, 我們就必須為我們的 function 指定一個參數(shù)。

在 MySQL 中創(chuàng)建函數(shù)時出現(xiàn)這種錯誤的解決方法:

set global log_bin_trust_function_creators=TRUE;

? Iterate語句 僅出現(xiàn)在loop,repeat,while循環(huán)語句中,其含義表示重新開始此循環(huán)

? Leave語句表明 退出指定標簽的流程控制語句塊

? 通常會用在begin…end,以及l(fā)oop,repeat,while的循環(huán)語句中

? Loop語句是存儲過程或函數(shù)中表達 循環(huán)執(zhí)行 的一種方式

? repeat語句是存儲過程或函數(shù)中表達 循環(huán)執(zhí)行 的一種方式

? while語句是存儲過程或函數(shù)中表達 循環(huán)執(zhí)行 的一種方式

? Return語句用在 函數(shù)中,用來終結(jié)函數(shù)的執(zhí)行并將指定值返回給調(diào)用者

? Cursor游標用來 聲明一個數(shù)據(jù)集

? 游標的聲明必須在變量和條件聲明之后,在handler聲明之前

? Cursor close語句用來 關(guān)閉之前打開的游標

? Cursor declare語句用來聲明一個游標和指定游標對應的數(shù)據(jù)集合, 通常數(shù)據(jù)集合是一個select語句

? Cursor fetch語句用來獲取游標指定數(shù)據(jù)集的 下一行數(shù)據(jù) 并將各個字段值賦予后面的變量

? Open cursor語句用來打開一個之前已經(jīng) 聲明好的游標

? Declare condition語句命名 特定的錯誤條件 ,而該特定錯誤可以在declare…h(huán)andler中指定 處理方法

? 比如在MySQL中1051error code表示的是unknown table的錯誤,如果要對這

個錯誤做特殊處理,可以用三種方法:

? Declare handler語句用來聲明一個handler來處理一個或多個特殊條件,當其中的某個條件滿足時則觸發(fā)其中的statement語句執(zhí)行

? Statement可以是一個簡單SQL語句,也可以是begin…end組成的多個語句

? Handler_action子句聲明當執(zhí)行完statement語句之后應該怎么辦

Condition_value的值有以下幾種:

? 當condition發(fā)生但沒有聲明handler時,則存儲過程和函數(shù)依照如下規(guī)則處理

? create trigger語句用來創(chuàng)建一個觸發(fā)器,觸發(fā)器的作用是當表上有對應SQL語句發(fā)生時,則觸發(fā)執(zhí)行

? 觸發(fā)器創(chuàng)建時需要 指定對應的表名 tbl_name

? Definer關(guān)鍵詞用來指定trigger的安全環(huán)境

? Trigger_time指定觸發(fā)器的執(zhí)行時間,BEFORE和AFTER指定觸發(fā)器在表中的 每行數(shù)據(jù)修改前或者后 執(zhí)行

? Trigger_event指定觸發(fā)該觸發(fā)器的具體 事件

? INSERT當新的一行數(shù)據(jù)插入表中時觸發(fā),比如通過執(zhí)行insert,load data,replace語句插入新數(shù)據(jù)

? UPDATE當表的一行數(shù)據(jù)被修改時觸發(fā),比如執(zhí)行update語句時

? DELETE當表的一行數(shù)據(jù)被刪除時觸發(fā),比如執(zhí)行delete,replace語句時

? 當執(zhí)行insert into … on duplicate key update語句時,當碰到重復行執(zhí)行update時,則觸發(fā)update下的觸發(fā)器

? 從5.7.2版本開始,可以創(chuàng)建具有相同trigger_time和trigger_event的同一個表上的多個觸發(fā)器,默認情況下按照創(chuàng)建的時間依次執(zhí)行,通過 指定FOLLOWS/PRECEDES改變執(zhí)行順序 ,即FOLLOWS時表示新創(chuàng)建的觸發(fā)器后執(zhí)行,PRECEDES則表示新觸發(fā)器先執(zhí)行

? Trigger_body表示觸發(fā)器觸發(fā)之后要執(zhí)行的一個或多個語句,在內(nèi)部可以引用涉及表的字段, OLD.col_name表示行數(shù)據(jù)被修改或刪除之前的字段數(shù)據(jù),NEW.col_name表示行數(shù)據(jù)被插入或修改之后的字段數(shù)據(jù)

? Drop trigger語句用來 刪除一個觸發(fā)器

? If exists短語用來避免刪除不存在的觸發(fā)器時引發(fā)報錯

? 當你執(zhí)行drop table時,表上的觸發(fā)器也被drop掉了

在mysql中自定義的函數(shù)怎么調(diào)用?

調(diào)用如:select 函數(shù)名(參數(shù)列表)。

mysql中的UDF(自定義函數(shù)),可以寫好一些方法或?函數(shù),然后進行調(diào)用,而且是在SQL語句中可以進行調(diào)用。?

DROP FUNCTION CalculateAmount?

CREATE FUNCTION CalculateAmount(userid INT) RETURNS float(10,2)?

BEGIN?

DECLARE totalCredits FLOAT;?

SELECT SUM(amount) INTO totalAmount FROM credit_user WHERE id =userid;?

RETURN totalAmount;?

END?

要注意的是,在UDF中,不要定義與數(shù)據(jù)表中重名的列。而在SQL中,?則可以像SELECT CalculateAmount(1);那樣去調(diào)用了。

網(wǎng)頁名稱:mysql怎么調(diào)試函數(shù),mysql函數(shù)怎么調(diào)用
本文URL:http://chinadenli.net/article33/dseiess.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供響應式網(wǎng)站關(guān)鍵詞優(yōu)化App開發(fā)動態(tài)網(wǎng)站網(wǎng)站營銷做網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

網(wǎng)站托管運營