Как сделать ДВА петли в UDF на mysql - PullRequest
0 голосов
/ 18 апреля 2020

аналогично StackOverflowPosting Я хотел бы рассчитать расстояние Левенштейна для матрицы m x n, состоящей из TITLE1 и TITLE2.

Мои функции Левенштейна работают нормально и отсюда: LD Но мой вопрос заключается в том, как я могу l oop через m x n в UDF?

Результатом должна быть таблица с m x n строками с LD, TITLE1 и TITLE2.

Я сделал это - НО Я ВСЕГДА ПОЛУЧИЛ ОШИБКУ

     1338 Cursor Declaration after Handler Declaration

Мой UDF выглядит так:

  BEGIN  
       DECLARE bDone INT;
       DECLARE bDone1 INT;

       DECLARE var2 varCHAR(255);    -- or approriate type
       DECLARE Var1 INT;
       DECLARE c1Var1 VARCHAR(250);

       DECLARE curs CURSOR FOR  SELECT recid as BIOTIrecid, replace(replace(BIOGRAPHYTITLE," [in SCOPUS]",""),"[SIMILAR]","") as bioti FROM BIO ;
       DECLARE CONTINUE HANDLER FOR NOT FOUND SET bDone = 1;
       DECLARE curs1 CURSOR FOR  SELECT trim(concat(scopus.Titel," ",scopus.Untertitel)) as scopusti FROM scopus ;
       DECLARE CONTINUE HANDLER FOR NOT FOUND SET bDone = 1;

       DROP  TABLE IF EXISTS LDResults;
        CREATE TABLE `LDResults` (
           `BIOGRAPHYTITLE` varchar(255) DEFAULT NULL,
           `recid` int(11) NOT NULL AUTO_INCREMENT,
           `BIOTIrecid` int(11) default NULL,
           `LD` varchar(255) DEFAULT NULL,
           `ScopusTI` varchar(255) DEFAULT NULL,
           PRIMARY KEY (`recid`)
        ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

        OPEN curs1;

        SET bDone1 = 0;
        #---------------- run all rows for scopusti
        REPEAT
            FETCH curs1 into c1var1;

        #-----------------------------------------
            OPEN curs;

            SET bDone = 0;
            #----- run all COLUMNs for biographytitle
            REPEAT
                FETCH curs INTO var1, var2;
                INSERT INTO LDResults (`BIOGRAPHYTITLE`, `BIOTIrecid`, `LD`, `ScopusTI`) VALUES (var2, var1, LEVENSHTEIN(var2,c1var1), c1var1);
            UNTIL bDone END REPEAT;
        #--------------------------------------------
        CLOSE curs;

        UNTIL bDone1 END REPEAT;


       CLOSE curs1;
       SELECT * FROM LDResults;
     END

Является ли мой способ решить эту проблему изощренным или это можно сделать с помощью более быстрого и лучшего решения?

Спасибо за все советы.

РЕДАКТИРОВАТЬ: Я мог сделайте это со счетчиком здесь: Любые комментарии?

     BEGIN  
        -- DECLARE bDone INT;
        -- DECLARE bDone1 INT;

        DECLARE i INT;

       DECLARE var2 varCHAR(255);    -- or approriate type
       DECLARE Var1 INT;
       DECLARE cVar1 VARCHAR(250);

       DECLARE curs1 CURSOR FOR  SELECT trim(concat(t.Titel," ",t.Untertitel)) as scopusti FROM tscopus t ;
       DECLARE curs CURSOR FOR  SELECT recid as BIOTIrecid, replace(replace(BIOGRAPHYTITLE," [in SCOPUS]",""),"[SIMILAR]","") as bioti FROM tBIO ;
       DECLARE CONTINUE HANDLER FOR NOT FOUND SET bDone = 1;
       #DECLARE curs1 CURSOR FOR  SELECT trim(concat(t.Titel," ",t.Untertitel)) as scopusti FROM tscopus t ;
       #DECLARE CONTINUE HANDLER FOR NOT FOUND SET bDone = 1;

       DROP  TABLE IF EXISTS LDResults;
        CREATE TABLE `LDResults` (
       `BIOGRAPHYTITLE` varchar(255) DEFAULT NULL,
       `recid` int(11) NOT NULL AUTO_INCREMENT,
       `BIOTIrecid` int(11) default NULL,
       `LD` varchar(255) DEFAULT NULL,
       `ScopusTI` varchar(255) DEFAULT NULL,
       PRIMARY KEY (`recid`)
        ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;


        OPEN curs1;
        SET i = 0;

        SET bDone1 = 0;
        -- ---------------- run all rows for scopusti
        REPEAT
            FETCH curs1 into cvar1;

         set i=(i+1);
        -- -----------------------------------------
            OPEN curs;

            SET bDone = 0;
            -- ----- run all COLUMNs for biographytitle
            REPEAT
                FETCH curs INTO var1, var2;
                INSERT INTO LDResults (`BIOGRAPHYTITLE`, `BIOTIrecid`, `LD`, `ScopusTI`) VALUES (var2, var1, LEVENSHTEIN(var2,cvar1), cvar1);
            UNTIL bDone END REPEAT;
        -- --------------------------------------------
        CLOSE curs;

        UNTIL (i >= 2) END REPEAT;


       CLOSE curs1;
       SELECT * FROM LDResults;
     END

1 Ответ

1 голос
/ 18 апреля 2020

Я имею в виду, что вы можете сделать это следующим образом, используя CROSS JOIN без циклов в вашем коде. CROSS JOIN по определению возвращает результат результата строк из двух таблиц. Таким образом, вы можете использовать этот результат и после некоторых манипуляций с данными вставить результат в новую таблицу, например:

DROP  TABLE IF EXISTS LDResults;
CREATE TABLE `LDResults` (
   `BIOGRAPHYTITLE` varchar(255) DEFAULT NULL,
   `recid` int(11) NOT NULL AUTO_INCREMENT,
   `BIOTIrecid` int(11) default NULL,
   `LD` varchar(255) DEFAULT NULL,
   `ScopusTI` varchar(255) DEFAULT NULL,
   PRIMARY KEY (`recid`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

INSERT INTO LDResults (`BIOGRAPHYTITLE`, `BIOTIrecid`, `LD`, `ScopusTI`)
SELECT bioti, BIOTIrecid, LEVENSHTEIN(bioti,scopusti), scopusti
FROM (
    SELECT 
        replace(replace(BIO.BIOGRAPHYTITLE," [in SCOPUS]",""),"[SIMILAR]","") as bioti,
        BIO.recid as BIOTIrecid,
        trim(concat(scopus.Titel," ",scopus.Untertitel)) as scopusti 
    FROM scopus
    CROSS JOIN BIO
) tbl;
...