不能简单地使用PostgreSQL表名("关系不存在")

Key*_*ger 157 php sql postgresql quoted-identifier

我正在尝试运行以下PHP脚本来执行简单的数据库查询:

$db_host = "localhost";
$db_name = "showfinder";
$username = "user";
$password = "password";
$dbconn = pg_connect("host=$db_host dbname=$db_name user=$username password=$password")
    or die('Could not connect: ' . pg_last_error());

$query = 'SELECT * FROM sf_bands LIMIT 10';
$result = pg_query($query) or die('Query failed: ' . pg_last_error());
Run Code Online (Sandbox Code Playgroud)

这会产生以下错误:

查询失败:错误:关系"sf_bands"不存在

在所有的例子中,我可以找到有人得到错误的地方,说明关系不存在,这是因为他们在表名中使用大写字母.我的表名没有大写字母.有没有办法查询我的表而不包括数据库名称,即showfinder.sf_bands

Bil*_*win 260

根据我的阅读,此错误意味着您没有正确引用表名.一个常见的原因是该表是使用混合大小写拼写定义的,并且您尝试使用所有小写进行查询.

换句话说,以下失败:

CREATE TABLE "SF_Bands" ( ... );

SELECT * FROM sf_bands;  -- ERROR!
Run Code Online (Sandbox Code Playgroud)

使用双引号来分隔标识符,以便在定义表时使用特定的混合大小写拼写.

SELECT * FROM "SF_Bands";
Run Code Online (Sandbox Code Playgroud)

重新评论,您可以向"search_path"添加一个模式,这样当您引用表名而不限定其模式时,查询将匹配该表名,方法是按顺序检查每个模式.就像PATH在shell或include_pathPHP等中一样.您可以检查当前的架构搜索路径:

SHOW search_path
  "$user",public
Run Code Online (Sandbox Code Playgroud)

您可以更改架构搜索路径:

SET search_path TO showfinder,public;
Run Code Online (Sandbox Code Playgroud)

另见http://www.postgresql.org/docs/8.3/static/ddl-schemas.html

  • 看来,即使你输入`SELECT*FROM SF_Bands`,这仍然会失败,因为Postgres决定为你小写那个表名.奇怪的... (12认同)
  • @Andy,当您编写自己的 SQL 数据库时,请随意以其他方式实现不区分大小写的标识符。:) (4认同)
  • @romkyns:是的,这在RDBMS品牌中非常普遍,无限制的标识符被宣传为"不区分大小写".但是他们并没有真正区分大小写,因为他们实施的方式是强制小写.仅当您在定义表时允许表名小写时,才匹配表的名称.如果在CREATE TABLE时使用双引号分隔符,则在查询中引用分隔符时必须使用分隔符. (3认同)
  • 如果表名不在引号中,Postgres 会自动将它们小写吗?真是够蠢的... (2认同)
  • 在你对 PostgreSQL 进行错误的谩骂之前,请先听听 SQL 标准是怎么说的: *<SQL 语言标识符> 相当于 <SQL 语言标识符>,其中每个小写字母都被替换为相应的一个或多个大写字母。这种处理包括确定等价性、信息和定义模式中的表示、诊断区域中的表示以及类似用途。*因此,虽然 PostgreSQL 不遵循标准,因为它将所有内容折叠为小写,但大小写不敏感是 PostgreSQL 所要求的。标准。 (2认同)

Mit*_*tzi 69

我有这个问题,这是故事(悲伤但真实):

  1. 如果您的表名全是小写,例如:您可以使用的帐户:select * from AcCounTs它可以正常工作

  2. 如果您的表名全部小写如下:accounts 以下将失败: select * from "AcCounTs"

  3. 如果您的表名称混合大小写如下:Accounts 以下将失败: select * from accounts

  4. 如果您的表名称是混合大小写的情况:Accounts 以下将正常工作: select * from "Accounts"

我不喜欢记住像这样无用的东西,但你必须;)

  • 5.像`Accounts`这样的混合大小写将失败,来自Accounts的`select*;`我发现最奇怪的部分:同一个案例不相同. (8认同)
  • 所有这一切:postgres查询中的所有名称都是小写的,除非您使用引号. (6认同)
  • 感谢您安排所有互动!:) (2认同)

Ugu*_*tun 15

Postgres进程查询与其他RDMS不同.在您的表名之前将模式名称放在双引号中,例如"SCHEMA_NAME"."SF_Bands"

  • 你的回答是什么增加了之前接受的答案,22次投票并且有很多细节? (6认同)

小智 14

将dbname参数放在连接字符串中.一切都失败了,它对我有用.

另外,在进行选择时,请指定your_schema.your_table像这样:

select * from my_schema.your_table
Run Code Online (Sandbox Code Playgroud)

  • 非常感谢你!它确实可以帮助我解决问题!但是有办法我可以省略方案名称吗? (2认同)

小智 8

我在OSX上遇到了类似的问题,但尝试使用双引号和单引号。对于您的情况,您可以尝试这样的事情

$query = 'SELECT * FROM "sf_bands"'; // NOTE: double quotes on "sf_Bands"
Run Code Online (Sandbox Code Playgroud)


小智 8

您必须在引号中写入架构名称和表名称。如下:

select * from "schemaName"."tableName";
Run Code Online (Sandbox Code Playgroud)


Ale*_*kin 7

这真的很有帮助

SET search_path TO schema,public;
Run Code Online (Sandbox Code Playgroud)

我更多地挖掘了这个问题,并发现了如何通过默认为当前数据库中的新用户设置这个“search_path”。

打开数据库属性,然后打开工作表“变量”,只需为您的用户添加具有实际值的变量。

所以现在您的用户将默认获得这个 schema_name,您可以使用 tableName 而不使用 schemaName。


Ste*_*way 6

如果表名包含下划线或大写,则需要用双引号将其括起来。

SELECT * from "Table_Name";
Run Code Online (Sandbox Code Playgroud)