sql - Teradata SQL 性能调整 : STRTOK_SPLIT_TO_TABLE: EXPLAIN Failed. 3738:字符串长度超过 31000 个字符

标签 sql user-defined-functions teradata data-warehouse sqlperformance

我可能再次抱有希望,但在列表中使用 STRTOK_SPLIT_TO_TABLE 时,我的列表是 > 32K 字符,它给了我上述错误

EXPLAIN Failed. 3738:  String is longer than 31000 characters

我做了什么

拆分字符串并使用 2 个 STRTOK_SPLIT_TO_TABLE 创建 2 个不同的表 然后加入我的主表

TD 错误文档中的实际错误翻译建议使用“USING 后跟数据包”。据我了解,您不能在 UDF 中做所有这些事情 那么除了拆分字符串之外,还有其他方法吗? 我有 400K 的列表内字符串,我想避免将其加载到 VT,因为这似乎有其自身的开销成本,因此使用此函数,但它从未奏效。


    EXPLAIN 
SELECT  COL1 
FROM    DB.TB1 , ( 
SELECT  TOKEN2 
FROM    TABLE ( STRTOK_SPLIT_TO_TABLE( 1 , '123456,123456,43456,......20K Long string' ,
','  ) 
RETURNS ( OUTKEY INT , TOKENNUM INT , TOKEN2 VARCHAR ( 20 ) CHARACTER 
SET unicode ) ) DD ) D2 ,


 ( 
SELECT  TOKEN1
FROM    TABLE ( STRTOK_SPLIT_TO_TABLE( 1 , '123456,123456,43456.....20K long string' ,','  ) 
RETURNS ( OUTKEY INT , TOKENNUM INT , TOKEN1 VARCHAR ( 20 ) CHARACTER 
SET unicode ) ) DD ) D1 

WHERE   SUBSTR ( TB1.COL2 , 21 , 9 ) = d2.token2 

or  SUBSTR ( TB1.COL2 , 21 , 9 ) = d1.token1
GROUP BY 1 ;


/* the same logic can be re-wrritten 
   with a UNION  or  Where EXISTS logic  
      but focus's on that UDF */

总结一下....

  • 我必须进一步将字符串长度拆分为更小的位。两个是不够的。当字符串确实大到值得运行拆分字符串 UDF 时,您如何利用 UDF 功能? 这是我在使用较小的字符串时遇到的错误。我的 explain 将运行良好,因为 UDF 抛出了这个错误

    Query Failed. 9134:  STRTOK: Input string exceeded the max. allowed string   length. 
    
  • 它不会让我使用拉丁语。我的表是 LATIN,因此它使用导致字符集转换的翻译开销。这里有解决方法吗? (也许自己翻译 UDF 的输出列,这样优化器就不会在连接列上执行此操作。但这在改进方面并不重要,因为优化器无论如何都会选择用较少的行来翻译字符集)

顺便说一句:我在 14.1 上,如下所示

    instring
a character or CLOB argument.
If instring is NULL, the STRTOK_SPLIT_TO_TABLE returns NULL.

根据文档 - 我可以为 instring 粘贴一个 CLOB。所以即使在我的 > 32K 字符串被转换为 CLOB 之后它也会抛出

EXPLAIN Failed. 3738:  String is longer than 31000 characters

最后 ....使用 character_length('D1 中的字符串')和('ditto d2')我得到 19K。它说 varchar (32000 ) limit for instring,所以想知道为什么拒绝?

更新:我可以对 D1 和 D2 进行 CLOB 并限制为 2 个字符串的方法,从而避免出现 9134 错误。但是 2 个注意事项仍然存在 -

  • 这个 UDF 只喜欢 token 的 UNICODE。我为 token 添加了一个“翻译”,无论它值多少钱。
  • 当我用这个测试时,我不太明白 32K 逻辑在这里是如何工作的

    sel Character_length ('这里是字符串')

    返回一个 < 32K 的数字,所以我的字符串不应受 32K 上限的限制

因此在查询的较小部分中使用它可以正常工作。我可以将字符串列表转换为表格,让这些加入并提取我感兴趣的 ID 列表,但是当我将组件插入更大的报告时,它会吓坏了——“本地 AMP 中的段错误不重新提交”。 ... 所以我不得不采用另一种方法。

最佳答案

这可能是 teradata UDF 中的一个错误,尽管 instring 长度小于 32K,它仍会拒绝它。

sel char_length ('string 32K or less' ) will give o/p 
31890 

当我在这里使用它的时候

SELECT  COL1 
FROM    DB.TB1 , ( 
SELECT  TOKEN2 
FROM    TABLE ( STRTOK_SPLIT_TO_TABLE( 1 , '123456,123456,43456,......32K Long string' ,
','  ) 
RETURNS ( OUTKEY INT , TOKENNUM INT , TOKEN2 VARCHAR ( 20 ) CHARACTER 
SET unicode ) ) DD ) D2 

它会失败,失败点就在 32K 的一半左右。对于 > 16K 个字符,它将失败。 但是使用 CLOB 可以让它工作 除了错误,我无法提供其他解释

SELECT  COL1 
FROM    DB.TB1 , ( 
SELECT  TOKEN2 
FROM    TABLE ( STRTOK_SPLIT_TO_TABLE( 1 , '123456,123456,43456,......32K Long string' ( CLOB)  ,
','  ) 
RETURNS ( OUTKEY INT , TOKENNUM INT , TOKEN2 VARCHAR ( 20 ) CHARACTER 
SET unicode ) ) DD ) D2

关于sql - Teradata SQL 性能调整 : STRTOK_SPLIT_TO_TABLE: EXPLAIN Failed. 3738:字符串长度超过 31000 个字符,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33883323/

相关文章:

php - WordPress 自定义 SQL-Query,太重了?

SQL 用户定义函数,找不到 dbo 或用户定义函数

java - 插入语句的Postgresql错误代码

php - 如何保存 foreach 循环中的选择选项列表

带有可选参数的 SAS 用户定义函数

sql - 在 Teradata 中选择值

python - Teradata Python 模块游标结果集在一次迭代后耗尽

java - 如何将 Java 字符串转换为 Teradata 日期和时间戳

c# - 从表中获取 auto_increment id

matlab - Matlab 中的脚本内部函数?