创建Query
CREATE QUERY queryname(parameter_list) [characteristics] [ LANGUAGE SQL ] BEGIN code_body ; END CREATE QUERY queryname(parameter_list) [characteristics] LANGUAGE OBJECTSCRIPT { code_body }
queryname
- 要在存储过程类中创建的查询的名称。queryname
必须是有效的标识符。过程名可以是限定的(schema.procname
),也可以是非限定的(procname
)。非限定过程名接受默认模式名。即使没有指定参数,queryname
也必须后跟括号。parameter_list
- 可选-传递给查询的参数列表。参数列表用圆括号括起来,列表中的参数用逗号分隔。即使没有指定参数,括号也是必须的。characteristics
- 可选-指定查询特征的一个或多个关键字。允许的关键字有结果、容器ID
、FOR
、FINAL
、PROCEDURE
、SELECTMODE
。多个特征由空白(空格或换行符)分隔。特性可以以任何顺序指定。LANGUAGE OBJECTSCRIPT
,LANGUAGE SQL
- 可选—指定用于code_body
的编程语言的关键字子句。指定语言对象脚本或语言SQL。如果省略了LANGUAGE
子句,则默认为SQL
。code_body
- 查询的程序代码。SQL程序代码以BEGIN
关键字开头,以END
关键字结尾。查询的code_body
只包含一个完整的SQL
语句(一个SELECT
语句)。该SELECT
语句以分号(;
)结束。ObjectScript
程序代码用花括号括起来。ObjectScript
代码行必须缩进。CREATE QUERY
语句在类中创建一个查询。
默认情况下,名为MySelect
的查询将被存储为User.queryMySelect
或SQLUser.queryMySelect
。
CREATE QUERY
创建的查询可能作为存储过程公开,也可能不作为存储过程公开。
要创建公开为存储过程的查询,必须指定procedure
关键字作为其特征之一。
还可以使用CREATE PROCEDURE
语句创建作为存储过程公开的查询。
为了创建查询,必须拥有%CREATE_QUERY
管理权限,如GRANT
命令所指定的。如果试图为已定义所有者的现有类创建查询,则必须以该类的所有者身份登录。否则,操作将失败,并出现SQLCODE -99
错误。
如果类定义是已部署的类,则不能在类中创建查询。此操作失败,出现SQLCODE -400
错误,出现%msgUnable to execute DDL that modifies a deployed class: 'classname'
。
要创建为存储过程的查询的名称。此名称可以是非限定名称(StoreName
)并采用默认架构名称,也可以通过指定架构名称(Patient.StoreName
)进行限定。可以使用$SYSTEM.SQL.Schema.Default()
方法来确定当前系统范围内的默认架构名称。系统范围内的初始默认模式名是SQLUser
,它对应于类包名User
。
注意,FOR
特征(将在下面描述)覆盖queryname
中指定的类名。
如果已经存在具有此名称的方法,则操作将失败,并出现SQLCODE -361
错误。
生成的类的名称是对应于架构名称的包名,后跟一个点,后跟“query”
,后跟指定的queryname
。例如,如果非限定查询名RandomLetter
采用初始默认模式SQLUser
,则得到的类名将是:User.queryRandomLetter
。
SQL
不允许指定只以字母大小写不同的查询名。
指定一个与现有查询名称仅在字母大小写上不同的查询名称将导致SQLCODE -400
错误。
如果指定的queryname
已经存在于当前命名空间中,系统将生成SQLCODE -361
错误。
用于将值传递给查询的参数的参数声明列表。
形参列表用圆括号括起来,列表中的形参声明用逗号分隔。
括号是必须的,即使没有指定参数。
列表中的每个参数声明由(按顺序)组成:
IN
(输入值)、OUT
(输出值)还是INOUT
(修改值)。IN
。DEFAULT
关键字后跟一个默认值;DEFAULT
关键字是可选的。如果没有指定默认值,则假定默认值为NULL
。下面的示例创建了一个公开为存储过程的查询,该存储过程具有两个输入参数,这两个参数都具有默认值。
topnum
输入参数指定可选的DEFAULT
关键字;
minage
输入参数忽略了这个关键字:
CREATE QUERY AgeQuery(IN topnum INT DEFAULT 10,IN minage INT 20) PROCEDURE BEGIN SELECT TOP :topnum Name,Age FROM Sample.Person WHERE Age > :minage ; END
以下是该查询的所有有效CALL语句:Call AgeQuery(6,65);Call AgeQuery(6);Call AgeQuery(,65);Call AgeQuery()
。
CALL AgeQuery(6,65); CALL AgeQuery(6); CALL AgeQuery(,65); CALL AgeQuery()
可用的特征关键字如下:
CONTAINID integer
- 指定返回ID的字段(如果有)。将CONTAINID
设置为返回ID的列的编号,如果没有列返回ID,则设置为0。 IRIS不验证命名字段是否确实包含ID,因此此处的用户错误会导致数据不一致。FOR className
- 指定要在其中创建方法的类的名称。如果该类不存在,则会创建它。还可以通过限定方法名称来指定类名。在FOR
子句中指定的类名将覆盖通过限定方法名指定的类名。FINAL
- 指定子类不能重写该方法。默认情况下,方法不是最终的。Final
关键字由子类继承。PROCEDURE
- 指定查询为SQL存储过程。存储过程由子类继承。(此关键字可以缩写为proc
。)RESULTS (result_set)
- 按查询返回数据字段的顺序指定数据字段。如果指定RESULTS
子句,则必须将查询返回的所有字段作为逗号分隔的列表列出,并将其括在圆括号中。指定比查询返回的字段少或多的字段会导致SQLCODE-76
基数不匹配错误。为每个字段指定列名(将用作列标题)和数据类型。如果使用SQL语言,则可以省略RESULTS
子句。如果省略RESULTS
子句,则会在类编译期间自动生成ROWSPEC
。SELECTMODE mode
- 指定用于编译查询的模式。可能的值有Logical
、ODBC
、Runtime
和Display
。默认值为运行时。如果指定的方法关键字(如PRIVATE
或RETURNS
)对查询无效,系统将生成SQLCODE-47
错误。指定重复特征会导致SQLCODE-44
错误。
SELECTMODE
子句指定返回数据的模式。如果模式值是逻辑值,则返回逻辑值(内部存储)。例如,日期以$HOROLOG
格式返回。如果模式值为ODBC
,则应用逻辑到ODBC
的转换,并返回ODBC
格式值。如果模式值为DISPLAY
,则应用逻辑到显示的转换,并返回显示格式值。如果模式值为RUNTIME
,则可以通过设置%SQL.Statement
类%SelectMode
属性在执行时设置模式(设置为LOGICAL
、ODBC
或DISPLAY
),运行时模式的值为Logical
。为SELECTMODE
指定的值将添加到ObjectScript
类方法代码的开头:#SQLCompile select=mode
。
RESULTS
子句指定查询的结果。RESULTS
子句中的SQL
数据类型参数被转换为查询的ROWSPEC
中相应的 IRIS数据类型参数。例如,RESULTS
子句RESULTS(Code VARCHAR(15))
生成ROWSPEC
规范ROWSPEC=“Code:%Library.String(MAXLEN=15)”
。
指定CODE_BODY
使用的语言的关键字子句。允许的子句是Language OBJECTSCRIPT
或Language SQL
。如果省略LANGUAGE
子句,则默认为SQL
。
如果语言是SQL
,则会生成%Library.SQLQuery
类型的类查询。如果语言是OBJECTSCRIPT
,则会生成%Library.Query
类型的类查询。
要创建的查询的程序代码。可以在SQL或ObjectScript中指定此代码。使用的语言必须与LANGUAGE
子句匹配。但是,在ObjectScript中指定的代码可以包含嵌入式SQL
。
如果指定的代码是SQL
,则它必须由单个SELECT
语句组成。SQL
中查询的程序代码以BEGIN
关键字开头,后跟程序代码(SELECT
语句)。在程序代码的末尾,指定分号(;
),然后指定END
关键字。
如果指定的代码是OBJECTSCRIPT
,则它必须包含对 IRIS提供的%Library.Query
类的Execute()
和Fetch()
类方法的调用,并且可以包含Close()
、FetchRows()
和GetInfo()
方法调用。ObjectScript代码用大括号括起来。如果EXECUTE()
或FETCH()
丢失,则编译时会生成SQLCODE-46
错误。
如果ObjectScript
代码块将数据提取到局部变量(例如,Row
)中,则必须以行set Row=""
结束代码块,以指示数据结束条件。
如果查询公开为存储过程(通过在Characteristic
中指定PROCEDURE
关键字),则它使用过程上下文处理程序在过程及其调用方之间来回传递过程上下文。
调用存储过程时,%Library.SQLProcContext
类的对象在%sqlcontext
变量中实例化。这用于在过程及其调用者(例如,ODBC
服务器)之间来回传递过程上下文。
%sqlcontext
由几个属性组成,包括错误对象、SQLCODE
错误状态、SQL行数和错误消息。下面的示例显示了用于设置其中几个值的值:
SET %sqlcontext.%SQLCODE=SQLCODE SET %sqlcontext.%ROWCOUNT=%ROWCOUNT SET %sqlcontext.%Message=%msg
SQLCODE
和%ROWCOUNT
的值由SQL
语句的执行自动设置。每次执行前都会重置%sqlcontext
对象。
或者,可以通过实例化%SYSTEM.Error
对象并将其设置为%sqlcontext.Error
来建立错误上下文。
IRIS使用提供的代码生成查询的实际代码。
下面的嵌入式SQL示例创建名为DocTestPersonState
的查询。它不声明任何参数,设置SELECTMODE
特征,并采用默认语言(SQL):
ClassMethod CreateQuery() { &sql( CREATE QUERY DocTestPersonState() SELECTMODE RUNTIME BEGIN SELECT Name,Home_State FROM Sample.Person ; END ) if SQLCODE=0 { w !,"创建查询" } elseif SQLCODE=-361 { w !,"查询存在: ",%msg } else { w !,"创建 QUERY 错误 ",SQLCODE } }
可以转到管理门户,选择Classes选项,然后选择Samples
命名空间。将在那里找到由上面的示例创建的查询:User.queryDocTestPersonState.cls
。在重新运行上面的程序示例之前,您可以从该显示中删除此查询。当然,可以使用DROP QUERY
删除创建的查询。
Class User.queryDocTestPersonState Extends %Library.RegisteredObject [ ClassType = "", DdlAllowed, Owner = {yx}, Not ProcedureBlock ] { Query DocTestPersonState() As %Library.SQLQuery(SELECTMODE = "RUNTIME") { SELECT Name,Home_State FROM Sample.Person } }
下面的嵌入式SQL示例创建一个名为DocTestSQLCODEList
的基于方法的查询,该查询获取SQLCODE
及其说明的列表。它设置结果结果集特征,将语言设置为ObjectScript
,并调用Execute()
、Fetch()
和Close()
方法:
ClassMethod CreateQuery1() { &sql( CREATE QUERY DocTestSQLCODEList() RESULTS ( SQLCODE SMALLINT, Description VARCHAR(100) ) PROCEDURE LANGUAGE OBJECTSCRIPT Execute(INOUT QHandle BINARY(255)) { s QHandle=1, %i(QHandle)="" q ##lit($$$OK) } Fetch(INOUT QHandle BINARY(255), INOUT Row %List, INOUT AtEnd INT) { s AtEnd = 0, Row = "" s %i(QHandle) = $o(^%qCacheSQL("SQLCODE", %i(QHandle))) if %i(QHandle) = "" { s AtEnd = 1 q ##lit($$$OK) } s Row = $lb(%i(QHandle), ^%qCacheSQL("SQLCODE", %i(QHandle), 1, 1)) q ##lit($$$OK) } Close(INOUT QHandle BINARY(255)) { k %i(QHandle) q ##lit($$$OK) } ) if SQLCODE=0 { w !,"创建查询" } elseif SQLCODE=-361 { w !,"查询存在: ",%msg } else { w !,"创建 QUERY 错误 ",SQLCODE _ " "_%msg } }
可以转到管理门户,选择Classes选项,然后选择Samples
命名空间。将在那里找到由上面的示例创建的查询:User.queryDocTestSQLCODEList.cls
。在重新运行上面的程序示例之前,可以从该显示中删除此查询。当然,可以使用DROP QUERY
删除创建的查询。
下面的动态SQL示例创建名为DocTest的查询,然后使用%SQL.Statement
类的%PrepareClassQuery()
方法执行此查询:
ClassMethod CreateQuery2() { s SQLCODE = 0 /* 创建 Query */ s myquery=4 s myquery(1) = "CREATE QUERY DocTest() SELECTMODE RUNTIME " s myquery(2) = "BEGIN " s myquery(3) = "SELECT TOP 5 Name,Home_State FROM Sample.Person ; " s myquery(4) = "END" s tStatement = ##class(%SQL.Statement).%New() s qStatus = tStatement.%Prepare(.myquery) if qStatus '= 1 { w "%Prepare 失败:" DO $System.Status.DisplayError(qStatus) q } s rset = tStatement.%Execute() if SQLCODE = 0 { w !,"创建查询" } elseif SQLCODE=-361 { w !,"查询存在: ",%msg } else { w !,"创建 QUERY 错误 ",SQLCODE _ " "_%msg } /* 调用 Query */ w !,"调用 Query",! s cqStatus = tStatement.%PrepareClassQuery("User.queryDocTest","DocTest") if cqStatus'=1 { w "%PrepareClassQuery 失败:" d $System.Status.DisplayError(cqStatus) } s rset = tStatement.%Execute() w "Query 数据",!,! while rset.%Next() { d rset.%Print() } w !,"结束结束" /* 删除 Query */ &sql(DROP QUERY DocTest) if SQLCODE = 0 { w !,"删除 Query" } }