本文目錄一覽:
- 1、mysql udf函數怎麼調用
- 2、怎樣用PHP中的MySQL操作函數
- 3、請教如何調試mysql 的存儲過程和函數
- 4、如何在Windows下編譯或調試MySQL
- 5、在mysql里怎樣自定義函數,如何調用和執行
mysql udf函數怎麼調用
1)基本用戶定義函數是一類代碼,對MYSQL伺服器功能進行擴充,通過添加新函數,性質就象使用本地MYSQL函數abs()或concat().UDF是用C(或C++)寫的。也許還可以用BASIC,.NET或其它什麼雖然還沒見過有人這麼做。
2)從字面上何以知道UDF是很有用的,尤其當需要擴展MYSQL伺服器功能時。下表給出了最佳解決方法的比較:
Method Speed Language Development
方法 速度 語言 開發
Stored Procedures slow SQL ~minutes (for small functions)
存儲過程 慢 SQL ~分鐘(對於小函數)
UDF fast C ~hour
UDF 快 C ~小時
Native Function fast C major pain in the ***
本地函數 快 C 未知
慢的意思是和其它比較時。存儲過程和一般SQL語句比仍然是很快的。
對本地函數的一點解釋:本質上和UDF沒太大區別。但是必須用MYSQL的資源代碼來寫然後重新編譯全部。這將是很大的工作量,必須一邊又一邊的用最新版的MYSQL來完成這項工作。
3)這部分很簡單。當完成了一個UDF,只是使用它就可以了。例如:”SELECT MyFunction(data1, data2) FROM table”
4)編寫UDF
現在開發寫一個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;
現在必須決定需要哪類函數。本質上有兩種選擇:
該函數是聚合函數嗎?(後面將學習很多關於聚合函數的內容)
返回類型是什麼?有4個選擇:
類型 描述
STRING 一個合法的字元串,轉換成char*類型
INTEGER 一個普通的整型變數,轉換成64位的整型
REAL型 一個俘點數,轉換成double型
DECIAML型 這個並沒真正的結束,MYSQL將做字元串對待
現在開始討論關於非聚合函數。必須聲明並執行一些MYSQL使用UDF時用到的函數,但首先一些必要的結構必須並確:
UDF_INIT:
類型 名稱 描述
my_bool maybe_null 是1如果函數能返回NULL
unsigned int decimals 針對REAL函數
unsigned long max_length 針對字元串函數
char * ptr 自由指針針對函數的數據
my_bool const_item 0如果結果是獨立的
UDF_ARGS:
類型 名稱 描述
unsigned int arg_count 成員數量
enum Item_result * arg_type 成員類型的數組
char ** args 指向成員的指針的數組
unsigned long * lengths 成員長度的數組(針對字元串)
char * maybe_null “maybe_null”標記的數組
char ** attributes 指向成員屬性的指針的數組
unsigned long * attribute_lengths 屬性長度數組
現在看一下該函數:
De-/Initialization:
Collapseextern “C” my_bool MyTest_init(UDF_INIT *initid, UDF_ARGS *args,
char *message)
{
//非常重要的一件事是建立內存
//需要
//需要一個很長的變數來保存檢測數
//雖然該例中不需要
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)
{
//這裡必須清空所分配的內存
//引入函數
delete (longlong*)initid-ptr;
}
The actual function:
extern “C” longlong MyTest(UDF_INIT *initid, UDF_ARGS *args,
char *is_null, char *error)
{
/*最後這是實際的工作部分。該函數為每個記錄調用,返回值或指向當前值的指針保存在UDF_ARGS變數中。必須獲得值,完成計算並返回值。注意可以通過UDF_INIT變數進入MyTest_init中分配的內存,該例中將為每個值設置為5
*/
return *((longlong*)args-args[0])+5;
}
全部完成!現在必須編譯連接庫,然後將其拷貝到操作系統可以載入的目錄下。通常在WINDOWS里是系統變數的定義路徑。個人使用的是MYSQL伺服器的bin目錄。必須確認該目錄是其它MYSQL不能訪問的。然後確認所有MYSQL需要的函數功能。
必須告訴MYSQL,這必須直接了當:執行以下SQL指令:
CREATE [AGGREGATE] FUNCTION MyTest
RETURNS [INTEGER|STRING|REAL|DECIMAL] SONAME the_libraries_exact_name
現在可以想使用其他函數一樣使用它了。
5)成員函數:
現在說一下成員函數。當的UDF是個成員函數,必須增加一些函數,一些函數在不同的方式中使用。調用次序是:
調用yTest_init來分配內存(就象一般的UDF一樣)
MYSQL將表分類是通過GROUP BY
每組裡的第一行調用MyTest_clear
每組裡的第一列調用 MyTest_add
在組改變後或最後一列改變後調用MyTest得到結果
重複3到5直到所有列被處理。
調用MyTest_deinit清空內存
現在讓看一下新的聚合函數所需的函數。該例中將簡單的添加所有的值。(就象本地SUM函數)
void MyTest_clear(UDF_INIT *initid, char *is_null, char *error)
{
/*為每個新組重新將總數設置為0,當然必須分配一個longlong類型變數在在init函數中,並分配給指針
*/
*((longlong*)initid-ptr) = 0;
}
void MyTest_add(UDF_INIT *initid, UDF_ARGS *args, char *is_null, char *error)
{
//為每列將當前值添加到總數
*((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時需要注意幾個問題:
一個字元串函數應該返回一個指向結果的指針並且設置*result和*length作為目錄和返回值的長度值。例如:
memcpy(result, “result string”, 13);
*length = 13;
MyTest建立的結果緩衝區是255位元組。如果的結果保存在裡面。不必擔心結果的內存分配問題。
如果的字元串函數需要返回一個大於255位元組長度的字元串。必須用malloc或新的MyTest_init或MyTest函數分配,然後用MyTest_deinit釋放它。能用UDF_INIT的指針保存分配的內存地址,並在MyTest中重用。
在主函數中指定一個錯誤返回,設置 *error為1:如果MyTest()為任何列將*error設置為1,則函數的值是NULL針對於當前列,以及對任何的通過MyTest()被調用的聲明中並發的列請求。
想了解更多內容看一下MYSQL在線幫助。
7)一些指導方針:
如果確實希望的UDF運行良好,這裡有一些建議:)
不要在UDF中調用任何其他的程序或進程
不要保存任何的本地信息。(這些在普通的庫里已經共享)
不要分配任何的全局或靜態的變數。
始終檢測成員的類型。就象MYSQL將所有類型都轉換為字元類型。如果將一個字元類型轉換成整型指針可能會出錯。
特別注意內存的分配。如果有內存泄漏問題會使伺服器徹底崩潰!
8)調式UDF
調試UDF需要勇氣因為如果UDF有問題,每次都會使整個MYSQL伺服器死掉。所以寫了一個命令行工具,來圍繞這個問題工作。僅僅運行它,它會模仿”SELECT”調用指令將結果保存到庫文件中,可以列印所有的結果行。所以當UDF存在一些錯誤只是該程序崩潰而不會是整個伺服器。
怎樣用PHP中的MySQL操作函數
1. 建立MySQL 連接:mysql_connect
resource mysql_connect([string $server[, string $username [, string $password [, bool $new_link [, int $client flags]]]]])
打開或重複使用一個到MySQL伺服器的連接。其中,server代表MySQL伺服器,可以包括埠號,例如「hostname:port」。username和password分別代表登錄MySQL伺服器使用的用戶名和密碼。
2. 選擇 MySQL 資料庫:mysql_select_db
bool mysql_select_db(string $database_name[, resource $link_identifier])
設定與指定的連接標識符所關聯的伺服器上的當前資料庫。如果沒有指定連接標識符,則使用上一個打開的連接。如果沒有打開的連接,本函數將無參數地調用mysql_connect()來嘗試打開一個資料庫連接。如果成功則返回true,失敗則返回false。每個其後的mysgl_query()調用都會作用於活動資料庫。
3. 執行查詢:mysql_query
resource mysql_query(string $query[, resource $link_identifier])
向與指定的連接標識的關聯的伺服器中的當前活動資料庫發送一條查詢。如果沒有指定 link_identifier,則使用上一個打開的連接。如果沒有打開的連接,本函數會嘗試無參數地調用mysql_connect()函數來建立一個連接,查詢結果會被緩存。mysql_query()僅對SELECT、SHOW、EXPLAIN或DESCRIBE語句返回一個資源標識符,如果查詢執行不正確則返回false。對於其他類型的SQL語句,mysgl_query()在執行成功時返回true,出錯時返回false。非false的返回值意味著查詢是合法的並能夠被伺服器執行但是並不說明任何影響到的或返回的行數。因為一條查詢執行成功了但並未影響到或並未返回任何行的情況是可能發生的。
4. 從結果集中取得一行作為枚舉數組:mysql_fetch_row
array mysql_fetch_row(resource $result)
從和指定的結果標識關聯的結果集中取得一行數據並作為數組返回。每個結果的列儲存在一個索引數組的單元中,偏移量從0開始。依次調用mysql_fetch_row()將返回結果集中的下一行,如果沒有更多行則返回false。
5. 從結果集中取得一行作為關聯數組,或數字數組,或二者兼有:mysql_fetch_array
array mysql_fetch_array(resource $result[, int $result_type])
mysql_fetch_row()的擴展版本。除了將數據以數字索引方式儲存在數組中之外,還可以將數據作為關聯索引儲存,用欄位名作為鍵名。如果結果中出現欄位名重名的現象,最後一列將優先。要訪問同名的其他列,必須用該列的數字索引或給該列起個別名。對有別名的列,用別名來訪問其內容。
6. 關閉 MySQL 連接:mysql_close
bool mysql_close([resource $link_identifier])
mysql_close()關閉指定的連接標識所關聯的到MySQL伺服器的非持久連接。如果沒有指定link_identifier,則關閉上一個打開的連接。
例PHP中使用MySQL資料庫(mysql.php)
?php//連接MySQL資料庫
mysql_connect(‘localhost’, ‘root’, ‘123’);
//選擇當前資料庫
//等效為執行USE test
mysql_select_db(‘test’);
//在當前資料庫執行SQL語句
$query = “SELECT * FROM students”;
$result = mysql_query($query);
//操作上次查詢返回的結果集,注意$result變數一般是需要的
while($student = mysql_fetch_array($result)) {
echo ‘pre’;
print_r($student);
echo ‘/pre’;
}
?
請教如何調試mysql 的存儲過程和函數
終於找到了一個好東東: Core Lab MyDeveloper Studio ,幾乎我想要的功能都有,特別是代碼提示和調試功能!可惜找不到註冊機,可以試用60天,強烈推薦個
如何在Windows下編譯或調試MySQL
用vs code 就可以了。
Visual Studio Code
Visual Studio Code(簡稱VS Code)是由微軟開發的,同時支持Windows、Linux和macOS操作系統的開源文本編輯器。它支持調試,內置了Git 版本控制功能,同時也具有開發環境功能,例如代碼補全(類似於IntelliSense)、代碼片段、代碼重構等。該編輯器支持用戶自定義配置,例如改變主題顏色、鍵盤快捷方式、編輯器屬性和其他參數,還支持擴展程序並在編輯器中內置了擴展程序管理的功能。
安裝LLDB
LLDB是LLVM編譯器的一部分,推薦使用Homebrew安裝LLVM工具集,不建議使用系統自帶的LLDB,安裝前必須先創建證書否則無法安裝,步驟如下:
創建完成後,開始安裝LLVM
brew install llvm –with-python@2 –with-lldb
安裝插件
VS Code自帶有debug功能,這裡我推薦使用LLDB Debugger插件。
接下來,為項目配置調試參數。
配置調試參數
使用VS Code打開MySQL源碼目錄,在側邊欄選擇debug欄目,添加配置,program輸入需要調試的程序路徑,這裡選擇你編譯好的mysqld路徑,args輸入程序啟動所需的參數,通常會指定mysqld的配置文件。這樣就配置好了,是不是很簡單。
啟動調試
點擊啟動按鈕,啟動後如果沒有設置斷點會mysqld會正常啟動,如果觸發了斷點會如下圖顯示。
整個調試窗口基本分為六部分,所有的調試操作都在這裡完成:
1: 顯示變數信息
2: 設置重點關注的變數
3: 顯示調用棧信息
4: 設置斷點信息,在代碼行號前也可以設置斷點
5: 代碼顯示區域,上方是調試按鈕,包括 continue/stepover/step in/step out/restart/stop
6: 調試終端輸入輸出區
斷點設置
在代碼行號前點擊即可在該行為設置斷點,也可以根據條件設置斷點。以設置ConditionalBreakpoint為例,當程序啟動後會按照你設置的條件表達式判斷是否觸發斷點。
Conditional Breakpoint這種方式用在目標變數達到某條件時觸發斷點,其餘則跳過繼續執行。比如:設置變數等於目標表名時觸發斷點,其餘表則跳過,相對函數名斷點省去很多手工跳過操作。
遠程調試
假如你想調試遠程Linux伺服器上的MySQL上面的方法就不合適了,這時需要遠程調試。lldb和gdb都支持遠程調試,這裡以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調試界面中新增配置項。
{
“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編譯的代碼路徑與本機代碼庫路徑的映射,這樣調試時代碼才可以和程序關聯在一起看
注意:記得調試前將代碼切換到與目標程序版本一致的branch
在mysql里怎樣自定義函數,如何調用和執行
create function 函數名稱(參數列表)
reurns 返回值類型
函數體
執行的方法很簡單:
載入
** create function 函數名稱 returns {string|real|integer}
** soname 你定義的動態庫位置
釋放!
** drop function 函數名稱
原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/278837.html