3

希望有人能帮我解决这个问题!

我有一个如下所示的 sql 文件:

CREATE TABLE IF NOT EXISTS users(
    id INT UNSIGNED NOT NULL AUTO_INCREMENT,
    firstname VARCHAR(30) NOT NULL,
    lastname VARCHAR(30) NOT NULL,

    PRIMARY KEY (id),
    CONSTRAINT UNIQUE (firstname,lastname)
)
ENGINE=InnoDB
;

INSERT IGNORE INTO users (firstname,lastname) VALUES ('x','y');
/*
INSERT IGNORE INTO users (firstname,lastname) VALUES ('a','b');
*/

我有一个 web 应用程序,它在启动时使用这个函数初始化一个 mysql 数据库:

public static void initDatabase(ConnectionPool pool, File sqlFile){
    Connection con = null;
    Statement st = null;
    String mySb=null;
    try{
        con = pool.getConnection();
        mySb=IOUtils.copyToString(sqlFile);

        // We use ";" as a delimiter for each request then we are sure to have well formed statements
        String[] inst = mySb.split(";");

        st = con.createStatement();

        for(int i = 0; i<inst.length; i++){
            // we ensure that there is no spaces before or after the request string
            // in order not to execute empty statements
            if(!inst[i].trim().isEmpty()){
                st.executeUpdate(inst[i]);
            }
        }
        st.close();
    }catch(IOException e){
        throw new RuntimeException(e);
    }catch(SQLException e){
        throw new RuntimeException(e);
    }finally{
        SQLUtils.safeClose(st);
        pool.close(con);
    }
}

(这个功能是在网上找到的。作者,请原谅我没有引用你的名字,我丢了!!)

只要没有 SQL 注释块,它就可以完美运行。

copyToString()功能基本上按照它所说的去做。我现在想要的是构建一个正则表达式,它将从字符串中删除块注释。我在文件中只有块注释/* */,没有--.

到目前为止我已经尝试过:

mySb = mySb.replaceAll("/\\*.*\\*/", "");

不幸的是,我不太擅长正则表达式......

我遇到了“匹配的字符串看起来像/* comment */ real statement /* another comment*/”等等的所有麻烦......

4

4 回答 4

10

尝试

mySb = mySb.replaceAll("/\\*.*?\\*/", "");

(注意?代表“懒惰”)。

编辑:要涵盖多行注释,请使用以下方法:

Pattern commentPattern = Pattern.compile("/\\*.*?\\*/", Pattern.DOTALL);
mySb = commentPattern.matcher(mySb).replaceAll("");

希望这对你有用。

于 2012-04-19T10:19:49.110 回答
2

您需要使用这样的勉强限定词:

public class Main {

    public static void main(String[] args) {
        String s = "The matched string look something like /* comment */ real statement /* another comment*/";
        System.err.println(s.replaceAll("/\\*.*?\\*/", ""));
    }
}
于 2012-04-19T10:19:49.350 回答
2

尝试以下方法:

String s = "/* comment */ select * from XYZ; /* comment */";
System.out.println(s.replaceAll("/\\*.*?\\*/", ""));

输出:

 select * from XYZ; 

.*?代表使用懒惰而不是贪婪(这意味着默认情况下匹配最大的字符串,即贪婪=>您必须使用表达式中的 将.*其配置为非贪婪)。?.*?

于 2012-04-19T10:26:02.530 回答
1

它不会 100% 工作

注释可以是 SQL 中指定的有效字符串的一部分,在这种情况下,它们需要保留...

我只是在研究一个解决方案......似乎很复杂

到目前为止,我有:

\G(?:[^']*?|'(?:[^']|'')*?'(?!'))*?\/\*.*?\*\/

但它匹配所有内容,而我只需要匹配评论......并且刚刚发现它可能会在单行评论之前失败......该死

于 2013-08-14T15:03:25.850 回答