MySQL加载数据:尚未在准备好的语句协议中支持此命令

时间:2017-06-05 02:05:19

标签: mysql linux bash import parameter-passing

我正在尝试编写一个MySQL脚本,将数据导入到我的Linux服务器的表中。以下是名为update.sql的脚本:

SET @query = CONCAT("LOAD DATA LOCAL INFILE '", @spaceName, "' INTO TABLE tmp FIELDS TERMINATED BY ',' LINES TERMINATED BY '\n';"); 
PREPARE stmt FROM @query;
EXECUTE stmt;
DEALLOCATE PREPARE stmt;

另外,我写了一个名为main.sh的bash脚本:

mysql -h "localhost" -u "root" "-pmypassword" "mydb" -e "set @spaceName=\"$1\";source update.sql;"

然后我执行./main.sh France.listFrance.list是我要导入数据库的数据文件。

但是我收到了一个错误:

  

文件中第2行的错误1295(HY000):'update.sql':此命令是   尚未在准备好的声明协议中支持

我发现了这个问题:
MySQL - Load Data Infile with variable path

那么,是否意味着无法将参数传递给LOAD DATA查询?

2 个答案:

答案 0 :(得分:3)

您无法使用PREPARE来运行LOAD DATA INFILE

此页面中记录了可以使用PREPARE运行的语句列表:https://dev.mysql.com/doc/refman/5.7/en/sql-syntax-prepared-statements.html在小标题“准备语句中允许的SQL语法”下。请注意,此列表在早期版本的MySQL中可能有所不同。

因为您无法使用PREPARE,所以您无法通过设置变量和制作动态SQL语句来执行您正在使用的方法。

但您可以在不使用LOAD DATA INFILE的情况下运行PREPARE。您必须使用shell变量替换将文件名插入到语句中,然后将其作为直接SQL语句运行。

您的update.sql文件可能如下所示:

LOAD DATA LOCAL INFILE '%spacename%' INTO TABLE tmp 
FIELDS TERMINATED BY ',' LINES TERMINATED BY '\n';

然后,您可以将shell变量替换为文件并以这种方式运行结果:

sed s/%spacename%/$1/ update.sql | 
  mysql -h "localhost" -u "root" "-pmypassword" "mydb"

另一种更简单的方法是使用mysqlimport,但这要求输入文件名与表名相同。您可以重命名输入文件以匹配要加载到的表(您调用tmp),或者创建符号链接:

ln -s $1 /tmp/tmp.list
mysqlimport --local -h "localhost" -u "root" "-pmypassword" "mydb" /tmp/tmp.list
rm -f /tmp/tmp.list

mysqlimport会忽略“.list”扩展名,因此您可以使用任何文件扩展名,也可以不使用。

答案 1 :(得分:1)

如果您使用的是Java 8 +,Mysql 8.0+和Connector / J库8.0.18,则可能会遇到类似“此版本不支持命令”之类的问题。

要解决此问题,您必须添加一个属性,该属性告诉prepareStatement到所有加载的本地文件中。您必须添加配置它,我在连接URL中对其进行配置并解决该问题。

jdbc:mysql://localhost:3306/tempDB?allowLoadLocalInfile=true

我也对my.cnf文件进行了一些更改,例如:

  1. local_infile = 1