INFORMIX SQL 实践与技巧(1)
如何加快sql的执行速度?
1.select 语句中使用sort,或join
如果你有排序和连接操作,你可以先select数据到一个临时表中,然后再对临时表进行处理。因为临时表是建立在内存中,所以比建立在磁盘上表操作要快的多。
如:
SELECT time_records.*, case_name
FROM time_records, OUTER cases
WHERE time_records.client = "AA1000"
AND time_records.case_no = cases.case_no
ORDER BY time_records.case_no
这个语句返回34个经过排序的记录,花费了5分钟42秒。而:
SELECT time_records.*, case_name
FROM time_records, OUTER cases
WHERE time_records.client = "AA1000"
AND time_records.case_no = cases.case_no
INTO temp foo;
SELECT * from foo ORDER BY case_no
返回34条记录,只花费了59秒。
2.使用not in 或者not exists 语句
下面的语句看上去没有任何问题,但是可能执行的非常慢:
SELECT code FROM table1
WHERE code NOT IN ( SELECT code FROM table2 )
如果使用下面的方法:
SELECT code, 0 flag
FROM table1
INTO TEMP tflag;
然后:
UPDATE tflag SET flag = 1
WHERE code IN ( SELECT code
FROM table2
WHERE tflag.code = table2.code );
然后:
SELECT * FROM
tflag
WHERE flag = 0;
看上去也许要花费更长的时间,但是你会发现不是这样。
事实上这种方式效率更快。有可能第一种方法也会很快,那是在对相关的每个字段都建立了索引的情况下,但是那显然不是一个好的注意。
3.避免使用过多的“or"
如果有可能的话,尽量避免过多地使用or:
WHERE a = "B" OR a = "C"
要比
WHERE a IN ("B","C")
慢。
有时甚至UNION会比OR要快。
4.使用索引。
如何在shell脚本中使用一个sql查询的结果?
以下的是一个运行在sh/ksh下面的脚本。在online中,如果你想要更新一个有许多表的数据库的统计信息。这个脚本不太好。因为这个脚本只能单个处理数据库中的表,而不能同时处理大量的表。
例子:
# update_em
# Run UPDATE STATISTICS on a table by table basis
#
DATABASE=$1
if [ -z "$DATABASE" ]
then
echo "usage: update_em dbname" >&2
exit 1
fi
isql $DATABASE - <
output to pipe "cat" without headings
select "update statistics for table ", tabname, ";"
from systables where tabid >= 100 order by tabname;
EOF
exit 0
也许你已经注意到exit的返回值对不同的isql不是都相同,因此这样作不是很可靠,代替通过$?来检查返回值的更好的主意是将标准错误重定向到一个文件中,然后在这个文件中grep “error"。例如:
# Generate the data
isql -qr <<!>stage.rep 2>$stage.err
database $database;
select ...
!
# Check for errors
if grep -i "error" $stage.err >/dev/null
then
...error_handler...
fi
为什么不能对一个计算产生的字段创建视图?
问题:为什么我不能创建视图:
CREATE VIEW tst AS
SELECT ship_charge - totval cout
FROM orders WHERE ship_charge > 0;
回答:你应该这样写:
CREATE VIEW tst (cout) AS
SELECT ship_charge - totval
FROM orders WHERE ship_charge > 0;
如何只select 出数据库中的部分数据(例如10%)。
问题:如果你想要得到一个select 语句正常返回的数据的一部分,例如:
SELECT firstname, lastname, city, state
FROM bigdatabase
WHERE state = "TX"
回答:
有一个方法可以返回一个近似值,只需要在where后加上:
AND rowid=(trunc(rowid/x)*x)
其中的x代表你想要返回的总的记录的1/x。需要说明的是,这种方法只能返回一个近似的值,并且表中的数据在物理上分布的连续性。
如何创建一个表结构和永久表完全一致的临时表。
例如:CREATE TEMP TABLE mytemp (prodno LIKE product.prodno
desc LIKE product.desc)
你可以使用如下的语句:
SELECT prodno, desc FROM product
WHERE ROWID = -1
INSERT INTO TEMP mytemp