Oracle 常用FAQ

Oracle 常用FAQ
  
第一部分、 SQL&PL/SQL
1.        [Q] 怎么样查询特殊字符,如通配符 % _
[A]select * from table where name like 'A/_%' escape '/'

2.        [Q] 如何插入单引号到数据库表中
[A]
可以用 ASCII 码处理,其它特殊字符如 & 也一样,如
insert into t values('i'||chr(39)||'m'); -- chr(39)
代表字符 '
或者用两个单引号表示一个
or insert into t values('I''m'); --
两个 '' 可以表示一个 '

3.        [Q] 怎样设置事务一致性
[A]set transaction [isolation level] read committed;
默认语句级一致性
set transaction [isolation level] serializable;
read only;
事务级一致性

4.        [Q] 怎么样利用游标更新数据
[A]cursor c1 is
select * from tablename
where name is null for update [of column]
……
update tablename set column = ……
where current of c1;

5.        [Q] 怎样自定义异常
[A] pragma_exception_init(exception_name,error_number);
如果立即抛出异常
raise_application_error(error_number,error_msg,true|false);
其中 number -20000 -20999 ,错误信息最大 2048B
异常变量
SQLCODE
错误代码
SQLERRM
错误信息

6.        [Q] 十进制与十六进制的转换
[A]8i
以上版本:
to_char(100,'XX')
to_number('4D','XX')
8i
以下的进制之间的转换参考如下脚本
create or replace function to_base( p_dec in number, p_base in number )
return varchar2
is
l_str varchar2(255) default NULL;
l_num number default p_dec;
l_hex varchar2(16) default '0123456789ABCDEF';
begin
if ( p_dec is null or p_base is null ) then
return null;
end if;
if ( trunc(p_dec) <> p_dec OR p_dec < 0 ) then
raise PROGRAM_ERROR;
end if;
loop
l_str := substr( l_hex, mod(l_num,p_base)+1, 1 ) || l_str;
l_num := trunc( l_num/p_base );
exit when ( l_num = 0 );
end loop;
return l_str;
end to_base;
/
create or replace function to_dec
( p_str in varchar2,
p_from_base in number default 16 ) return number
is
l_num number default 0;
l_hex varchar2(16) default '0123456789ABCDEF';
begin
if ( p_str is null or p_from_base is null ) then
return null;
end if;
for i in 1 .. length(p_str) loop
l_num := l_num * p_from_base + instr(l_hex,upper(substr(p_str,i,1)))-1;
end loop;
return l_num;
end to_dec;
/

7.        [Q] 能不能介绍 SYS_CONTEXT 的详细用法
[A]
利用以下的查询,你就明白了
select
SYS_CONTEXT('USERENV','TERMINAL') terminal,
SYS_CONTEXT('USERENV','LANGUAGE') language,
SYS_CONTEXT('USERENV','SESSIONID') sessionid,
SYS_CONTEXT('USERENV','INSTANCE') instance,
SYS_CONTEXT('USERENV','ENTRYID') entryid,
SYS_CONTEXT('USERENV','ISDBA') isdba,
SYS_CONTEXT('USERENV','NLS_TERRITORY') nls_territory,
SYS_CONTEXT('USERENV','NLS_CURRENCY') nls_currency,
SYS_CONTEXT('USERENV','NLS_CALENDAR') nls_calendar,
SYS_CONTEXT('USERENV','NLS_DATE_FORMAT') nls_date_format,
SYS_CONTEXT('USERENV','NLS_DATE_LANGUAGE') nls_date_language,
SYS_CONTEXT('USERENV','NLS_SORT') nls_sort,
SYS_CONTEXT('USERENV','CURRENT_USER') current_user,
SYS_CONTEXT('USERENV','CURRENT_USERID') current_userid,
SYS_CONTEXT('USERENV','SESSION_USER') session_user,
SYS_CONTEXT('USERENV','SESSION_USERID') session_userid,
SYS_CONTEXT('USERENV','PROXY_USER') proxy_user,
SYS_CONTEXT('USERENV','PROXY_USERID') proxy_userid,
SYS_CONTEXT('USERENV','DB_DOMAIN') db_domain,
SYS_CONTEXT('USERENV','DB_NAME') db_name,
SYS_CONTEXT('USERENV','HOST') host,
SYS_CONTEXT('USERENV','OS_USER') os_user,
SYS_CONTEXT('USERENV','EXTERNAL_NAME') external_name,
SYS_CONTEXT('USERENV','IP_ADDRESS') ip_address,
SYS_CONTEXT('USERENV','NETWORK_PROTOCOL') network_protocol,
SYS_CONTEXT('USERENV','BG_JOB_ID') bg_job_id,
SYS_CONTEXT('USERENV','FG_JOB_ID') fg_job_id,
SYS_CONTEXT('USERENV','AUTHENTICATION_TYPE') authentication_type,
SYS_CONTEXT('USERENV','AUTHENTICATION_DATA') authentication_data
from dual

8.        [Q] 怎么获得今天是星期几,还关于其它日期函数用法
[A]
可以用 to_char 来解决,如
select to_char(to_date('2002-08-26','yyyy-mm-dd'),'day') from dual;
在获取之前可以设置日期语言,如
ALTER SESSION SET NLS_DATE_LANGUAGE='AMERICAN';
还可以在函数中指定
select to_char(to_date('2002-08-26','yyyy-mm-dd'),'day','NLS_DATE_LANGUAGE = American') from dual;
其它更多用法,可以参考 to_char to_date 函数
如获得完整的时间格式
select to_char(sysdate,'yyyy-mm-dd hh24:mi:ss') from dual;
随便介绍几个其它函数的用法:
本月的天数
SELECT to_char(last_day(SYSDATE),'dd') days FROM dual
今年的天数
select add_months(trunc(sysdate,'year'), 12) - trunc(sysdate,'year') from dual
下个星期一的日期
SELECT Next_day(SYSDATE,'monday') FROM dual

9.        [Q] 随机抽取前 N 条记录的问题
[A]8i
以上版本
select * from (select * from tablename order by sys_guid()) where rownum < N;
select * from (select * from tablename order by dbms_random.value) where rownum< N;
注: dbms_random 包需要 手工安装,位于 $ORACLE_HOME/rdbms/admin/dbmsrand.sql
dbms_random.value(100,200)
可以产生 100 200 范围的随机数

10.    [Q] 抽取从 N 行到 M 行的记录,如从 20 行到 30 行的记录
[A]select * from (select rownum id,t.* from table where ……
and rownum <= 30) where id > 20;

11.    [Q] 怎么样抽取重复记录
[A]select * from table t1 where where t1.rowed !=
(select max(rowed) from table t2
where t1.id=t2.id and t1.name=t2.name)
或者
select count(*), t.col_a,t.col_b from table t
group by col_a,col_b
having count(*)>1
如果想删除重复记录,可以把第一个语句的 select 替换为 delete

12.    [Q] 怎么样设置自治事务
[A]8i
以上版本,不影响主事务
pragma autonomous_transaction;
……
commit|rollback;

13.    [Q] 怎么样在过程中暂停指定时间
[A]DBMS_LOCK
包的 sleep 过程
如: dbms_lock.sleep(5); 表示暂停 5 秒。

14.    [Q] 怎么样快速计算事务的时间与日志量
[A]
可以采用类似如下的脚本
DECLARE
start_time NUMBER;
end_time NUMBER;
start_redo_size NUMBER;
end_redo_size NUMBER;
BEGIN
start_time := dbms_utility.get_time;
SELECT VALUE INTO start_redo_size FROM v$mystat m,v$statname s
WHERE m.STATISTIC#=s.STATISTIC#
AND s.NAME='redo size';
--transaction start
INSERT INTO t1
SELECT * FROM All_Objects;
--other dml statement
COMMIT;
end_time := dbms_utility.get_time;
SELECT VALUE INTO end_redo_size FROM v$mystat m,v$statname s
WHERE m.STATISTIC#=s.STATISTIC#
AND s.NAME='redo size';
dbms_output.put_line('Escape Time:'||to_char(end_time-start_time)||' centiseconds');
dbms_output.put_line('Redo Size:'||to_char(end_redo_size-start_redo_size)||' bytes');
END;

15.    [Q] 怎样创建临时表
[A]8i
以上版本
create global temporary tablename(column list)
on commit preserve rows; --
提交保留数据 会话临时表
on commit delete rows; --
提交删除数据 事务临时表
临时表是相对于会话的,别的会话看不到该会话的数据。

16.    [Q] 怎么样在 PL/SQL 中执行 DDL 语句
[A]1
8i 以下版本 dbms_sql
2
8i 以上版本还可以用
execute immediate sql;
dbms_utility.exec_ddl_statement('sql');

17.    [Q] 怎么样获取 IP 地址
[A]
服务器 (817 以上 ) utl_inaddr.get_host_address
客户端: sys_context('userenv','ip_address')

18.    [Q] 怎么样加密存储过程
[A]
wrap 命令,如(假定你的存储过程保存为 a.sql
wrap iname=a.sql
PL/SQL Wrapper: Release 8.1.7.0.0 - Production on Tue Nov 27 22:26:48 2001
Copyright (c) Oracle Corporation 1993, 2000. All Rights Reserved.
Processing a.sql to a.plb
提示 a.sql 转换为 a.plb ,这就是加密了的脚本,执行 a.plb 即可生成加密了的存储过程

19.    [Q] 怎么样在 ORACLE 中定时运行存储过程
[A]
可以利用 dbms_job 包来定时运行作业,如执行存储过程,一个简单的例子,提交一个作业:
VARIABLE jobno number;
BEGIN
DBMS_JOB.SUBMIT(:jobno, 'ur_procedure;',SYSDATE,'SYSDATE + 1');
commit;
END;
之后,就可以用以下语句查询已经提交的作业
select * from user_jobs;

20.    [Q] 怎么样从数据库中获得毫秒
[A]9i
以上版本,有一个 timestamp 类型获得毫秒,如
SQL>select to_char(systimestamp,'yyyy-mm-dd hh24:mi:ssxff') time1,
to_char(current_timestamp) time2 from dual;

TIME1 TIME2
----------------------------- ----------------------------------------------------------------
2003-10-24 10:48:45.656000 24-OCT-03 10.48.45.656000 AM +08:00
可以看到,毫秒在 to_char 中对应的是 FF
8i
以上版本可以创建一个如下的 java 函数
SQL>create or replace and compile
java source
named "MyTimestamp"
as
import java.lang.String;
import java.sql.Timestamp;

public class MyTimestamp
{
public static String getTimestamp()
{
return(new Timestamp(System.currentTimeMillis())).toString();
}
};
SQL>java created.
注:注意 java 的语法,注意大小写
SQL>create or replace function my_timestamp return varchar2
as language java
name 'MyTimestamp.getTimestamp() return java.lang.String';
/
SQL>function created.
SQL>select my_timestamp,to_char(sysdate,'yyyy-mm-dd hh24:mi:ss') ORACLE_TIME from dual;
MY_TIMESTAMP ORACLE_TIME
------------------------ -------------------
2003-03-17 19:15:59.688 2003-03-17 19:15:59
如果只想获得 1/100 (hsecs) ,还可以利用 dbms_utility.get_time

21.    [Q] 如果存在就更新,不存在就插入可以用一个语句实现吗
[A]9i
已经支持了,是 Merge ,但是只支持 select 子查询,
如果是单条数据记录,可以写作 select …… from dual 的子查询。
语法为:
MERGE INTO table
USING data_source
ON (condition)
WHEN MATCHED THEN update_clause
WHEN NOT MATCHED THEN insert_clause;

MERGE INTO course c
USING (SELECT course_name, period,
course_hours
FROM course_updates) cu
ON (c.course_name = cu.course_name
AND c.period = cu.period)
WHEN MATCHED THEN
UPDATE
SET c.course_hours = cu.course_hours
WHEN NOT MATCHED THEN
INSERT (c.course_name, c.period,
c.course_hours)
VALUES (cu.course_name, cu.period,
cu.course_hours);

22.    [Q] 怎么实现左联,右联与外联
[A]
9i 以前可以这么写 :
左联:
select a.id,a.name,b.address from a,b
where a.id=b.id(+)
右联 :
select a.id,a.name,b.address from a,b
where a.id(+)=b.id
外联
SELECT a.id,a.name,b.address
FROM a,b
WHERE a.id = b.id(+)
UNION
SELECT b.id,'' name,b.address
FROM b
WHERE NOT EXISTS (
SELECT * FROM a
WHERE a.id = b.id);
9i 以上,已经开始支持 SQL99 标准,所以,以上语句可以写成:
默认内部联结:
select a.id,a.name,b.address,c.subject
from (a inner join b on a.id=b.id)
inner join c on b.name = c.name
where other_clause
左联
select a.id,a.name,b.address
from a left outer join b on a.id=b.id
where other_clause
右联
select a.id,a.name,b.address
from a right outer join b on a.id=b.id
where other_clause
外联
select a.id,a.name,b.address
from a full outer join b on a.id=b.id
where other_clause
or
select a.id,a.name,b.address
from a full outer join b using (id)
where other_clause

23.    [Q] 怎么实现一条记录根据条件多表插入
[A]9i
以上可以通过 Insert all 语句完成,仅仅是一个语句,如:
INSERT ALL
WHEN (id=1) THEN
INTO table_1 (id, name)
values(id,name)
WHEN (id=2) THEN
INTO table_2 (id, name)
values(id,name)
ELSE
INTO table_other (id, name)
values(id, name)
SELECT id,name
FROM a;
如果没有条件的话,则完成每个表的插入,如
INSERT ALL
INTO table_1 (id, name)
values(id,name)
INTO table_2 (id, name)
values(id,name)
INTO table_other (id, name)
values(id, name)
SELECT id,name
FROM a;

24.    [Q] 如何实现行列转换
[A]1
、固定列数的行列转换

student subject grade
---------------------------
student1
语文 80
student1
数学 70
student1
英语 60
student2
语文 90
student2
数学 80
student2
英语 100
……
转换为
语文 数学 英语
student1 80 70 60
student2 90 80 100
……
语句如下:
select student,sum(decode(subject,'
语文 ', grade,null)) " 语文 ",
sum(decode(subject,'
数学 ', grade,null)) " 数学 ",
sum(decode(subject,'
英语 ', grade,null)) " 英语 "
from table
group by student

2
、不定列行列转换

c1 c2
--------------
1

1

1

2

2

3

……
转换为
1
我是谁
2
知道
3

这一类型的转换必须借助于 PL/SQL 来完成,这里给一个例子
CREATE OR REPLACE FUNCTION get_c2(tmp_c1 NUMBER)
RETURN VARCHAR2
IS
Col_c2 VARCHAR2(4000);
BEGIN
FOR cur IN (SELECT c2 FROM t WHERE c1=tmp_c1) LOOP
Col_c2 := Col_c2||cur.c2;
END LOOP;
Col_c2 := rtrim(Col_c2,1);
RETURN Col_c2;
END;
/
SQL> select distinct c1 ,get_c2(c1) cc2 from table;
即可

25.    [Q] 怎么样实现分组取前 N 条记录
[A]8i
以上版本,利用分析函数
如获取每个部门薪水前三名的员工或每个班成绩前三名的学生。
Select * from
(select depno,ename,sal,row_number() over (partition by depno
order by sal desc) rn
from emp)
where rn<=3

26.    [Q] 怎么样把相邻记录合并到一条记录
[A]8i
以上版本,分析函数 lag lead 可以提取后一条或前一天记录到本记录。
Select deptno,ename,hiredate,lag(hiredate,1,null) over
(partition by deptno order by hiredate,ename) last_hire
from emp
order by depno,hiredate

27.    [Q] 如何取得一列中第 N 大的值?
[A]select * from
(select t.*,dense_rank() over (order by t2 desc) rank from t)
where rank = &N;

28.    [Q] 怎么样把查询内容输出到文本
[A]
spool
sqlplus –s " / as sysdba" <<EOF
set heading off
set feedback off
spool temp.txt
  select * from tab;
dbms_output.put_line(‘test’);
spool off
exit
EOF

29.    [Q] 如何在 SQL*PLUS 环境中执行 OS 命令?
[A]
比如进入了 SQLPLUS ,启动了数据库,忽然想起监听还没有启动,此时不用退出 SQLPLUS ,也不用另外起一个命令行窗口,直接输入:
SQL> host lsntctl start
或者 unix/linux 平台下
SQL>!<OS command>
windows
平台下
SQL>$<OS command>
总结: HOST <OS command> 可以直接执行 OS 命令。
备注: cd 命令无法正确执行。

30.    [Q] 怎么设置存储过程的调用者权限
[A]
普通存储过程都是所有者权限,如果想设置调用者权限,请参考如下语句
create or replace
procedure ……()
AUTHID CURRENT_USER
As
begin
……
end;

31.    [Q] 怎么快速获得用户下每个表或表分区的记录数
[A]
可以分析该用户,然后查询 user_tables 字典,或者采用如下脚本即可
SET SERVEROUTPUT ON SIZE 20000
DECLARE
miCount INTEGER;
BEGIN
FOR c_tab IN (SELECT table_name FROM user_tables) LOOP
EXECUTE IMMEDIATE 'select count(*) from "' || c_tab.table_name || '"' into miCount;
dbms_output.put_line(rpad(c_tab.table_name,30,'.') || lpad(miCount,10,'.'));
--if it is partition table
SELECT COUNT(*) INTO miCount FROM User_Part_Tables WHERE table_name = c_tab.table_name;
IF miCount >0 THEN
FOR c_part IN (SELECT partition_name FROM user_tab_partitions WHERE table_name = c_tab.table_name) LOOP
EXECUTE IMMEDIATE 'select count(*) from ' || c_tab.table_name || ' partition (' || c_part.partition_name || ')'

INTO miCount;
dbms_output.put_line(' '||rpad(c_part.partition_name,30,'.') || lpad(miCount, 10,'.'));
END LOOP;
END IF;
END LOOP;
END;
 
第二部分、 ORACLE构架体系
1.          [Q]ORACLE 的有那些数据类型
[A]
常见的数据类型有
CHAR
固定长度字符域,最大长度可达 2000 个字节
NCHAR
多字节字符集的固定长度字符域,长度随字符集而定,最多为 2000 个字符或 2000 个字节
VARCHAR2
可变长度字符域,最大长度可达 4000 个字符
NVARCHAR2
多字节字符集的可变长度字符域,长度随字符集而定,最多为 4000 个字符或 4000 个字节
DATE
用于存储全部日期的固定长度 (7 个字节 ) 字符域,时间作为日期的一部分存储其中。除非
通过设置 init.ora 文件的 NLS_DATE_FORMAT 参数来取代日期格式,否则查询时,日期以
DD-MON-YY
格式表示,如 13-APR-99 表示 1999.4.13
NUMBER
可变长度数值列,允许值为 0 、正数和负数。 NUMBER 值通常以 4 个字节或更少的字节存储,最多 21 字节
LONG
可变长度字符域,最大长度可到 2GB
RAW
表示二进制数据的可变长度字符域,最长为 2000 个字节
LONGRAW
表示二进制数据的可变长度字符域,最长为 2GB
MLSLABEL
只用于 TrustedOracle ,这个数据类型每行使用 2 5 个字节
BLOB
二进制大对象,最大长度为 4GB
CLOB
字符大对象,最大长度为 4GB
NCLOB
多字节字符集的 CLOB 数据类型,最大长度为 4GB
BFILE
外部二进制文件,大小由操作系统决定
ROWID
表示 RowID 的二进制数据, Oracle8RowID 的数值为 10 个字节,在 Oracle7 中使用的限定
RowID
格式为 6 个字节
UROWID
用于数据寻址的二进制数据,最大长度为 4000 个字节

2.          [Q]Oracle 有哪些常见关键字,不能被用于对象名
[A]
8i 版本为例,一般保留关键字不能用做对象名
ACCESS ADD ALL ALTER AND ANY AS ASC AUDIT BETWEEN BY CHAR CHECK CLUSTER COLUMN COMMENT COMPRESS CONNECT CREATE CURRENT DATE DECIMAL DEFAULT DELETE DESC DISTINCT DROP ELSE EXCLUSIVE EXISTS FILE FLOAT FOR FROM GRANT GROUP HAVING IDENTIFIED IMMEDIATE IN INCREMENT INDEX INITIAL INSERT INTEGER INTERSECT INTO IS LEVEL LIKE LOCK LONG MAXEXTENTS MINUS MLSLABEL MODE MODIFY NOAUDIT NOCOMPRESS NOT NOWAIT NULL NUMBER OF OFFLINE ON ONLINE OPTION OR ORDER PCTFREE PRIOR PRIVILEGES PUBLIC RAW RENAME RESOURCE REVOKE ROW ROWID ROWNUM ROWS SELECT SESSION SET SHARE SIZE SMALLINT START SUCCESSFUL SYNONYM SYSDATE TABLE THEN TO TRIGGER UID UNION UNIQUE UPDATE USER VALIDATE VALUES VARCHAR VARCHAR2 VIEW WHENEVER WHERE WITH
详细信息可以查看 v$reserved_words 视图

3.          [Q] 怎么查看数据库版本
[A]select * from v$version
包含版本信息,核心版本信息,位数信息 (32 位或 64 )
至于位数信息,在 linux/unix 平台上,可以通过 file 查看,如
file $ORACLE_HOME/bin/oracle

4.          [Q] 怎么查看数据库参数
[A]show parameter
参数名
如通过 show parameter spfile 可以查看 9i 是否使用 spfile 文件
或者 select * from v$parameter
除了这部分参数, Oracle 还有大量隐含参数,可以通过如下语句查看 :
SELECT NAME
,VALUE
,decode(isdefault, 'TRUE','Y','N') as "Default"
,decode(ISEM,'TRUE','Y','N') as SesMod
,decode(ISYM,'IMMEDIATE', 'I',
'DEFERRED', 'D',
'FALSE', 'N') as SysMod
,decode(IMOD,'MODIFIED','U',
'SYS_MODIFIED','S','N') as Modified
,decode(IADJ,'TRUE','Y','N') as Adjusted
,description
FROM ( --GV$SYSTEM_PARAMETER
SELECT x.inst_id as instance
,x.indx+1
,ksppinm as NAME
,ksppity
,ksppstvl as VALUE
,ksppstdf as isdefault
,decode(bitand(ksppiflg/256,1),1,'TRUE','FALSE') as ISEM
,decode(bitand(ksppiflg/65536,3),
1,'IMMEDIATE',2,'DEFERRED','FALSE') as ISYM
,decode(bitand(ksppstvf,7),1,'MODIFIED','FALSE') as IMOD
,decode(bitand(ksppstvf,2),2,'TRUE','FALSE') as IADJ
,ksppdesc as DESCRIPTION
FROM x$ksppi x
,x$ksppsv y
WHERE x.indx = y.indx
AND substr(ksppinm,1,1) = '_'
AND x.inst_id = USERENV('Instance')
)
ORDER BY NAME

5.          [Q] 怎么样查看数据库字符集
[A]
数据库服务器字符集 select * from nls_database_parameters ,其来源于 props$ ,是表示数据库的字符集。
客户端字符集环境 select * from nls_instance_parameters, 其来源于 v$parameter
表示客户端的字符集的设置,可能是参数文件,环境变量或者是注册表
会话字符集环境 select * from nls_session_parameters ,其来源于 v$nls_parameters ,表示会话自己的设置,可能是会话的环境变量或者是 alter session 完成,如果会话没有特殊的设置,将与 nls_instance_parameters 一致。
客户端的字符集要求与服务器一致,才能正确显示数据库的非 Ascii 字符。如果多个设置存在的时候, alter session> 环境变量 > 注册表 > 参数文件
字符集要求一致,但是语言设置却可以不同,语言设置建议用英文。如字符集是 zhs16gbk ,则 nls_lang 可以是 American_America.zhs16gbk

6.          [Q] 怎么样修改字符集
[A]8i
以上版本可以通过 alter database 来修改字符集,但也只限于子集到超集,不建议修改 props$ 表,将可能导致严重错误。
Startup nomount;
Alter database mount exclusive;
Alter system enable restricted session;
Alter system set job_queue_process=0;
Alter database open;
Alter database character set zhs16gbk;

7.          [Q] 怎样建立基于函数索引
[A]8i
以上版本,确保
Query_rewrite_enabled=true
Query_rewrite_integrity=trusted
Compatible=8.1.0
以上
Create index indexname on table (function(field));

8.          [Q] 怎么样移动表或表分区
[A]
移动表的语法
Alter table tablename move
[Tablespace new_name
Storage(initial 50M next 50M
pctincrease 0 pctfree 10 pctused 50 initrans 2) nologging]
移动分区的语法
alter table tablename move (partition partname)
[update global indexes]
之后之后 必须重建索引
Alter index indexname rebuild
如果表有 Lob 段,那么正常的 Alter 不能移动 Lob 段到别的表空间,而仅仅是移动了表段,可以采用如下的方法移动 Lob
alter table tablename move
lob(lobsegname) store as (tablespace newts);

9.          [Q] 怎么获得当前的 SCN
[A]9i
以下版本
select max(ktuxescnw*power(2,32)+ktuxescnb) from x$ktuxe;
如果是 9i 以上版本,还可以通过以下语句获取
select dbms_flashback.get_system_change_number from dual;

10.       [Q]ROWID 的结构与组成
[A]8
以上版本的 ROWID 组成
OOOOOOFFFBBBBBBRRR
8
以下 ROWID 组成(也叫受限 Rowid
BBBBBBBB.RRRR.FFFF
其中, O 是对象 ID F 是文件 ID B 是块 ID R 是行 ID
如果我们查询一个表的 ROWID ,根据其中块的信息,可以知道该表确切占用了多少个块,进而知道占用了多少数据空间(此数据空间不等于表的分配空间)

11.       [Q] 怎么样获取对象的 DDL 语句
[A]
第三方工具就不说了主要说一下 9i 以上版本的 dbms_metadata
1
、获得单个对象的 DDL 语句
set heading off
set echo off
set feedback off
set pages off
set long 90000
select dbms_metadata.get_ddl('TABLE','TABLENAME','SCAME') from dual;
如果获取整个用户的脚本,可以用如下语句
select dbms_metadata.get_ddl('TABLE',u.table_name) from user_tables u;
当然,如果是索引,则需要修改相关 table index

12.       [Q] 如何创建约束的索引在别的表空间上
[A]1
、先创建索引,再创建约束
2
、利用如下语句创建
create table test
(c1 number constraint pk_c1_id primary key
using index tablespace useridex,
c2 varchar2(10)
) tablespace userdate;

13.       [Q] 怎么知道那些表没有建立主键
[A]
一般的情况下,表的主键是必要的,没有主键的表可以说是不符合设计规范的。
SELECT table_name
FROM User_tables t
WHERE NOT EXISTS
(SELECT table_name
FROM User_constraints c
WHERE constraint_type = 'P'
AND t.table_name=c.table_name)
其它相关数据字典解释
user_tables

user_tab_columns
表的列
user_constraints
约束
user_cons_columns
约束与列的关系
user_indexes
索引
user_ind_columns
索引与列的关系

14.       [Q]dbms_output 提示缓冲区不够,怎么增加
[A]dbms_output.enable(20000);
另外,如果 dbms_output 的信息不能显示,
需要设置
set serveroutput on

15.       [Q] 怎么样修改表的列名
[A]9i
以上版本可以采用 rname 命令
ALTER TABLE UserName.TabName
RENAME COLUMN SourceColumn TO DestColumn
9i
以下版本可以采用 create table …… as select * from SourceTable 的方式。
另外, 8i 以上可以支持删除列了
ALTER TABLE UserName.TabName
SET UNUSED (ColumnName) CASCADE CONSTRAINTS
ALTER TABLE UserName.TabName
DROP (ColumnName) CASCADE CONSTRAINTS

16.       [Q] 怎么样给 sqlplus 安装帮助
[A]SQLPLUS
的帮助必须手工安装, shell 脚本为 $ORACLE_HOME/bin/helpins
在安装之前,必须先设置 SYSTEM_PASS 环境变量,如:
$ setenv SYSTEM_PASS SYSTEM/MANAGER
$ helpins
如果不设置该环境变量,将在运行脚本的时候提示输入环境变量
当然,除了 shell 脚本,还可以利用 sql 脚本安装,那就不用设置环境变量了,但是,我们必须以 system 登录。
$ sqlplus system/manager
SQL> @?/sqlplus/admin/help/helpbld.sql helpus.sql
安装之后,你就可以象如下的方法使用帮助了
SQL> help index

17.       [Q] 怎么样快速下载 Oracle 补丁
[A]
我们先获得下载服务器地址,在 http 页面上有
ftp://updates.oracle.com
然后用 ftp 登录,用户名与密码是 metalink 的用户名与密码
如我们知道了补丁号 3095277 (9204 的补丁集 ) ,则
ftp> cd 3095277
250 Changed directory OK.
ftp> ls
200 PORT command OK.
150 Opening data connection for file listing.
p3095277_9204_AIX64-5L.zip
p3095277_9204_AIX64.zip
……
p3095277_9204_WINNT.zip
226 Listing complete. Data connection has been closed.
ftp: 208 bytes received in 0.02Seconds 13.00Kbytes/sec.
ftp>
知道了这个信息,我们用用 flashget ,网络蚂蚁就可以下载了。
添加如下连接
ftp://updates.oracle.com/3095277/p3...04_AIX64-5L.zip
或替换后面的部分为所需要的内容
注意,如果是 flashget ,网络蚂蚁请输入认证用户名及密码,就是你的 metalink 的用户名与密码!

18.       [Q] 如何移动数据文件
[A]1
、关闭数据库,利用 os 拷贝
a.shutdown immediate
关闭数据库
b.
os 下拷贝数据文件到新的地点
c.Startup mount
启动到 mount
d.Alter database rename datafile '
老文件 ' to ' 新文件 ';
e.Alter database open;
打开数据库
2
、利用 Rman 联机操作
RMAN> sql "alter database datafile ''file name'' offline";
RMAN> run {
2> copy datafile 'old file location'
3> to 'new file location';
4> switch datafile ' old file location'
5> to datafilecopy ' new file location';
6> }
RMAN> sql "alter database datafile ''file name'' online";
说明:利用 OS 拷贝也可以联机操作,不关闭数据库,与 rman 的步骤一样,利用 rman 与利用 os 拷贝的原理一样,在 rman copy 是拷贝数据文件,相当于 OS cp ,而 switch 则相当于 alter database rename ,用来更新控制文件。

19.       [Q] 如果管理联机日志组与成员
[A]
以下是常见操作,如果在 OPA/RAC 下注意线程号
增加一个日志文件组
Alter database add logfile [group n] '
文件全名 ' size 10M;
在这个组上增加一个成员
Alter database add logfile member '
文件全名 ' to group n;
在这个组上删除一个日志成员
Alter database drop logfile member '
文件全名 ';
删除整个日志组
Alter database drop logfile group n;

20.       [Q] 怎么样计算 REDO BLOCK 的大小
[A]
计算方法为 (redo size + redo wastage) / redo blocks written + 16
具体见如下例子
SQL> select name ,value from v$sysstat where name like '%redo%';
NAME VALUE
---------------------------------------------------------------- ----------
redo synch writes 2
redo synch time 0
redo entries 76
redo size 19412
redo buffer allocation retries 0
redo wastage 5884
redo writer latching time 0
redo writes 22
redo blocks written 51
redo write time 0
redo log space requests 0
redo log space wait time 0
redo log switch interrupts 0
redo ordering marks 0
SQL> select (19412+5884)/51 + 16 '"Redo black(byte)" from dual;
Redo black(byte)
------------------
512

21.       [Q] 控制文件包含哪些基本内容
[A]
控制文件主要包含如下条目,可以通过 dump 控制文件内容看到
DATABASE ENTRY
CHECKPOINT PROGRESS RECORDS
REDO THREAD RECORDS
LOG FILE RECORDS
DATA FILE RECORDS
TEMP FILE RECORDS
TABLESPACE RECORDS
LOG FILE HISTORY RECORDS
OFFLINE RANGE RECORDS
ARCHIVED LOG RECORDS
BACKUP SET RECORDS
BACKUP PIECE RECORDS
BACKUP DATAFILE RECORDS
BACKUP LOG RECORDS
DATAFILE COPY RECORDS
BACKUP DATAFILE CORRUPTION RECORDS
DATAFILE COPY CORRUPTION RECORDS
DELETION RECORDS
PROXY COPY RECORDS
INCARNATION RECORDS

22.       [Q] 如果发现表中有坏块,如何检索其它未坏的数据
[A]
首先需要找到坏块的 ID (可以运行 dbverify 实现),假设为 <BID> ,假定文件编码为 <FID> 。运行下面的查询查找段名:
SELECT segment_name,segment_type,extent_id,block_id, blocks
from dba_extents t
where
file_id = <FID>
AND <BID> between block_id and (block_id + blocks - 1)
一旦找到坏段名称,若段是一个表,则最好建立一个临时表,存放好的数据。若段是索引,则删除它,再重建。
create table good_table
as
select from bad_table where rowid not in
(select rowid
from bad_table where substr(rowid,10,6) = <BID> )
在这里要注意 8 以前的受限 ROWID 与现在 ROWID 的差别。
还可以使用诊断事件 10231
SQL> ALTER SYSTEM SET EVENTS '10231 trace name context forever,level 10';
创建一个临时表 good_table 的表中除坏块的数据都检索出来
SQL>CREATE TABLE good_table as select * from bad_table;
最后关闭诊断事件
SQL> ALTER SYSTEM SET EVENTS '10231 trace name context off ';
关于 ROWID 的结构,还可以参考 dbms_rowid.rowid_create 函数。

23.       [Q] 我创建了数据库的所有用户,我可以删除这些用户吗
[A]ORACLE
数据库创建的时候,创建了一系列默认的用户和表空间,以下是他们的列表
·SYS/CHANGE_ON_INSTALL or INTERNAL
系统用户,数据字典所有者,超级权限所有者 (SYSDBA)
创建脚本: ?/rdbms/admin/sql.bsq and various cat*.sql
建议创建后立即修改密码
此用户不能被删除
·SYSTEM/MANAGER
数据库默认管理用户,拥有 DBA 角色权限
创建脚本: ?/rdbms/admin/sql.bsq
建议创建后立即修改密码
此用户不能被删除
·OUTLN/OUTLN
优化计划的存储大纲用户
创建脚本: ?/rdbms/admin/sql.bsq
建议创建后立即修改密码
此用户不能被删除
---------------------------------------------------
·SCOTT/TIGER, ADAMS/WOOD, JONES/STEEL, CLARK/CLOTH and BLAKE/PAPER.
实验、测试用户,含有例表 EMP DEPT
创建脚本: ?/rdbms/admin/utlsampl.sql
可以修改密码
用户可以被删除,在产品环境建议删除或锁定
·HR/HR (Human Resources), OE/OE (Order Entry), SH/SH (Sales History).
实验、测试用户,含有例表 EMPLOYEES DEPARTMENTS
创建脚本: ?/demo/schema/mksample.sql
可以修改密码
用户可以被删除,在产品环境建议删除或锁定
·DBSNMP/DBSNMP
Oracle Intelligent agent
创建脚本: ?/rdbms/admin/catsnmp.sql, called from catalog.sql
可以改变密码 -- 需要放置新密码到 snmp_rw.ora 文件
如果不需要 Intelligent Agents ,可以删除
---------------------------------------------------
以下用户都是可选安装用户 , 如果不需要,就不需要安装
·CTXSYS/CTXSYS
Oracle interMedia (ConText Cartridge)
管理用户
创建脚本: ?/ctx/admin/dr0csys.sql
·TRACESVR/TRACE
Oracle Trace server
创建脚本: ?/rdbms/admin/otrcsvr.sql
·ORDPLUGINS/ORDPLUGINS
Object Relational Data (ORD) User used by Time Series, etc.
创建脚本: ?/ord/admin/ordinst.sql
·ORDSYS/ORDSYS
Object Relational Data (ORD) User used by Time Series, etc
创建脚本: ?/ord/admin/ordinst.sql
·DSSYS/DSSYS
Oracle Dynamic Services and Syndication Server
创建脚本: ?/ds/sql/dssys_init.sql
·MDSYS/MDSYS
Oracle Spatial administrator user
创建脚本: ?/ord/admin/ordinst.sql
·AURORA$ORB$UNAUTHENTICATED/INVALID
Used for users who do not authenticate in Aurora/ORB
创建脚本: ?/javavm/install/init_orb.sql called from ?/javavm/install/initjvm.sql
·PERFSTAT/PERFSTAT
Oracle Statistics Package (STATSPACK) that supersedes UTLBSTAT/UTLESTAT
创建脚本: ?/rdbms/admin/statscre.sql
1.          [Q] 如何开启 / 关闭归档
[A]
如果开启归档,请保证 log_archive_start=true 开启自动归档,否则只能手工归档,如果是关闭了归档,则设置该参数为 false
注意:如果是 OPS/RAC 环境,需要先把 parallel_server = true 注释掉,然后执行如下步骤,最后用这个参数重新启动
1
、开启归档
a.
关闭数据库 shutdown immediate
b. startup mount
c. alter database archivelog
d. alter database opne
2
、禁止归档
a.
关闭数据库 shutdown immediate
b. startup mount
c. alter database noarchivelog
d. alter database open
归档信息可以通过如下语句查看
SQL> archive log list
Database log mode Archive Mode
Automatic archival Enabled
Archive destination E:/oracle/ora92/database/archive
Oldest online log sequence 131
Next log sequence to archive 133
Current log sequence 133

2.          [Q] 怎样设置定时归档
[A]9i
以上版本,保证归档的最小间隔不超过 n
设置 Archive_lag_target = n
单位:秒 范围: 0~7200

3.          [Q] 不同版本怎么导出 / 导入
[A]
导出用低版本,导入用当前版本
如果版本跨越太大,需要用到中间版本过渡

4.          [Q] 不同的字符集之前怎么导数据
[A]a.
前条件是保证导出 / 导入符合其他字符集标准,如客户环境与数据库字符集一致。
b.
修改 dmp 文件的 2 3 字节为目标数据库的字符集,注意要换成十六进制。
参考函数(以下函数中的 ID 是十进制的):
nls_charset_name
根据字符集 ID 获得字符集名称
nls_charset_id
根据字符集名称获得字符集 ID

5.          [Q] 怎么样备份控制文件
[A]
再线备份为一个二进制的文件
alter database backup controlfile to '$BACKUP_DEPT/controlfile.000' [reuse];
备份为文本文件方式
alter database backup controlfile to trace [resetlogs|noresetlogs];

6.          [Q] 控制文件损坏如何恢复
[A]1
、如果是损坏单个控制文件
只需要关闭数据库,拷贝一个好的数据文件覆盖掉坏的数据文件即可
或者是修改 init.ora 文件的相关部分
2
、如果是损失全部控制文件,则需要创建控制文件或从备份恢复
创建控制文件的脚本可以通过 alter database backup controlfile to trace 获取。

7.          [Q] 怎么样热备份一个表空间
[A]Alter tablespace
名称 begin backup;
host cp
这个表空间的数据文件 目的地 ;
Alter tablespace
名称 end backup;
如果是备份多个表空间或整个数据库,只需要一个一个表空间的操作下来就可以了。

8.          [Q] 怎么快速得到整个数据库的热备脚本
[A]
可以写一段类似的脚本
SQL>set serveroutput on
begin
dbms_output.enable(10000);
for bk_ts in (select distinct t.ts#,t.name from v$tablespace t,v$datafile d where t.ts#=d.ts#) loop
dbms_output.put_line('--'||bk_ts.name);
dbms_output.put_line('alter tablespace '||bk_ts.name||' begin backup;');
for bk_file in (select file#,name from v$datafile where ts#=bk_ts.ts#) loop
dbms_output.put_line('host cp '||bk_file.name||' $BACKUP_DEPT/');
end loop;
dbms_output.put_line('alter tablespace '||bk_ts.name||' end backup;');
end loop;
end;
/

9.          [Q] 丢失一个数据文件,但是没有备份,怎么样打开数据库
[A]
如果没有备份只能是删除这个数据文件了,会导致相应的数据丢失。
SQL>startup mount
--ARCHIVELOG
模式命令
SQL>Alter database datafile 'file name' offline;
--NOARCHIVELOG
模式命令
SQL>Alter database datafile 'file name' offline drop;
SQLl>Alter database open;
注意:该数据文件不能是系统数据文件

10.       [Q] 丢失一个数据文件,没有备份但是有该数据文件创建以来的归档怎么恢复
[A]
保证如下条件
a.
不能是系统数据文件
b.
不能丢失控制文件
如果满足以上条件,则
SQL>startup mount
SQL>Alter database create datafile 'file name' as 'file name' size ... reuse;
SQL>recover datafile n; -
文件号
或者
SQL>recover datafile 'file name';
或者
SQL>recover database;
SQL>Alter database open;

11.       [Q] 联机日志损坏如何恢复
[A]1
、如果是非当前日志而且归档,可以使用
Alter database clear logfile group n
来创建一个新的日志文件
如果该日志还没有归档,则需要用
Alter database clear unarchived logfile group n
2
、如果是当前日志损坏,一般不能 clear ,则可能意味着丢失数据
如果有备份,可以采用备份进行不完全恢复
如果没有备份,可能只能用 _allow_resetlogs_corruption=true 来进行强制恢复了,但是,这样的方法是不建议的,最好在有 Oracle support 的指导下进行。

12.       [Q] 怎么样创建 RMAN 恢复目录
[A]
首先,创建一个数据库用户,一般都是 RMAN ,并给予 recovery_catalog_owner 角色权限
sqlplus sys
SQL> create user rman identified by rman;
SQL> alter user rman default tablespace tools temporary tablespace temp;
SQL> alter user rman quota unlimited on tools;
SQL> grant connect, resource, recovery_catalog_owner to rman;
SQL> exit;
然后,用这个用户登录,创建恢复目录
rman catalog rman/rman
RMAN> create catalog tablespace tools;
RMAN> exit;
最后,你可以在恢复目录注册目标数据库了
rman catalog rman/rman target backdba/backdba
RMAN> register database;

13.       [Q] 怎么样在恢复的时候移动数据文件,恢复到别的地点
[A]
给一个 RMAN 的例子
run {
set until time 'Jul 01 1999 00:05:00';
allocate channel d1 type disk;
set newname for datafile '/u04/oracle/prod/sys1prod.dbf'
to '/u02/oracle/prod/sys1prod.dbf';
set newname for datafile '/u04/oracle/prod/usr1prod.dbf'
to '/u02/oracle/prod/usr1prod.dbf';
set newname for datafile '/u04/oracle/prod/tmp1prod.dbf'
to '/u02/oracle/prod/tmp1prod.dbf';
restore controlfile to '/u02/oracle/prod/ctl1prod.ora';
replicate controlfile from '/u02/oracle/prod/ctl1prod.ora';
restore database;
sql "alter database mount";
switch datafile all;
recover database;
sql "alter database open resetlogs";
release channel d1;
}

14.       [Q] 怎么从备份片( backuppiece )中恢复 (restore) 控制文件与数据文件
[A]
可以使用如下方法,在 RMAN 中恢复备份片的控制文件
restore controlfile from backuppiecefile;
如果是 9i 的自动备份,可以采用如下的方法
restore controlfile from autobackup;
但是,如果控制文件全部丢失,需要指定 DBID ,如 SET DBID=?
自动备份控制文件的默认格式是 %F ,这个格式的形式为
c-IIIIIIIIII-YYYYMMDD-QQ
,其中 IIIIIIIIII 就是 DBID
至于恢复 (restore) 数据文件, oracle 816 开始有个包 dbms_backup_restore
nomount 状态下就可以执行,可以读 815 甚至之前的备份片,读出来的文件用于恢复
可以在 SQLPLUS 中运行,如下
SQL>startup nomount
SQL> DECLARE
2 devtype varchar2(256);
3 done boolean;
4 BEGIN
5 devtype := dbms_backup_restore.deviceallocate('', params=>'');
6 dbms_backup_restore.restoresetdatafile;
7 dbms_backup_restore.restorecontrolfileto('E:/Oracle/oradata/penny/control01.ctl');
8 dbms_backup_restore.restoreDataFileto(1,'E:/Oracle/oradata/penny/system01.dbf');
9 dbms_backup_restore.restoreDataFileto(2,'E:/Oracle/oradata/penny/UNDOTBS01.DBF');
10 dbms_backup_restore.restoreDataFileto(3,'E:/ORACLE/ORADATA/PENNY/USERS01.DBF');
11 dbms_backup_restore.restorebackuppiece('D:/orabak/BACKUP_1_4_04F4IAJT.PENNY',done=>done);
12 END;
13 /
PL/SQL
过程已成功完成。
SQL> alter database mount;

15.       [Q] 执行 exec dbms_logmnr_d.build('Logminer.ora','file directory') ,提示下标超界,怎么办
[A]
完整错误信息如下 ,
SQL> exec dbms_logmnr_d.build('Logminer.ora','file directory')
BEGIN dbms_logmnr_d.build('Logminer.ora','file directory'); END;
*
ERROR
位于第 1 :
ORA-06532:
下标超出限制
ORA-06512:
"SYS.DBMS_LOGMNR_D", line 793
ORA-06512:
line 1
解决办法为:
1.
编辑位于 "$ORACLE_HOME/rdbms/admin" 目录下的文件 "dbmslmd.sql"
改变行 :
TYPE col_desc_array IS VARRAY(513) OF col_description;

TYPE col_desc_array IS VARRAY(700) OF col_description;
并保存文件
2.
运行改变后的脚本
SQLPLUS> Connect internal
SQLPLUS> @$ORACLE_HOME/rdbms/admin/dbmslmd.sql
3.
重新编译该包
SQLPLUS> alter package DBMS_LOGMNR_D compile body;

16.       [Q] 执行 execute dbms_logmnr.start_logmnr(DictFileName=>'DictFileName') 提示 ORA-01843: 无效的月份,这个是什么原因
[A]
我们分析 start_logmnr
PROCEDURE start_logmnr(
startScn IN NUMBER default 0 ,
endScn IN NUMBER default 0,
startTime IN DATE default TO_DATE('01-jan-1988','DD-MON-YYYY'),
endTime IN DATE default TO_DATE('01-jan-2988','DD-MON-YYYY'),
DictFileName IN VARCHAR2 default '',
Options IN BINARY_INTEGER default 0 );
可以知道,如果 TO_DATE('01-jan-1988','DD-MON-YYYY') 失败,将导致以上错误
所以解决办法可以为
1
Alter session set NLS_LANGUAGE=American
2
、用类似如下的方法执行
execute dbms_logmnr.start_logmnr (DictFileName=> 'f:/temp2/TESTDICT.ora', starttime => TO_DATE(
'01-01-1988','DD-MM-YYYY'), endTime=>TO_DATE('01-01-2988','DD-MM-YYYY'));
 
1.          [Q] 如果设置自动跟踪
[A]
system 登录
执行 $ORACLE_HOME/rdbms/admin/utlplan.sql 创建计划表
执行 $ORACLE_HOME/rdbms/admin/plustrce.sql 创建 plustrace 角色
如果想计划表让每个用户都能使用,则
SQL>create public synonym plan_table for plan_table;
SQL> grant all on plan_table to public;
如果想让自动跟踪的角色让每个用户都能使用,则
SQL> grant plustrace to public;
通过如下语句开启 / 停止跟踪
SET AUTOTRACE ON |OFF
| ON EXPLAIN | ON STATISTICS | TRACEONLY | TRACEONLY EXPLAIN

2.          [Q] 如果跟踪自己的会话或者是别人的会话
[A]
跟踪自己的会话很简单
Alter session set sql_trace true|false
Or
Exec dbms_session.set_sql_trace(TRUE);
如果跟踪别人的会话,需要调用一个包
exec dbms_system.set_sql_trace_in_session(sid,serial#,true|false)
跟踪的信息在 user_dump_dest 目录下可以找到或通过如下脚本获得文件名称(适用于 Win 环境,如果是 unix 需要做一定修改)
SELECT p1.value||'/'||p2.value||'_ora_'||p.spid||'.ora' filename
FROM
v$process p,
v$session s,
v$parameter p1,
v$parameter p2
WHERE p1.name = 'user_dump_dest'
AND p2.name = 'db_name'
AND p.addr = s.paddr
AND s.audsid = USERENV ('SESSIONID')
最后,可以通过 Tkprof 来解析跟踪文件,如
Tkprof
原文件 目标文件 sys=n

3.          [Q] 怎么设置整个数据库系统跟踪
[A]
其实文档上的 alter system set sql_trace=true 是不成功的
但是可以通过设置事件来完成这个工作,作用相等
alter system set events
'10046 trace name context forever,level 1';
如果关闭跟踪,可以用如下语句
alter system set events
'10046 trace name context off';
其中的 level 1 与上面的 8 都是跟踪级别
level 1
:跟踪 SQL 语句,等于 sql_trace=true
level 4
:包括变量的详细信息
level 8
:包括等待事件
level 12
:包括绑定变量与等待事件

4.          [Q] 怎么样根据 OS 进程快速获得 DB 进程信息与正在执行的语句
[A]
有些时候,我们在 OS 上操作,象 TOP 之后我们得到的 OS 进程,怎么快速根据 OS 信息获得 DB 信息呢?
我们可以编写如下脚本:
$more whoit.sh
#!/bin/sh
sqlplus /nolog <<EOF
connect / as sysdba
col machine format a30
col program format a40
set line 200
select sid,serial# ,username,osuser,machine,program,process,to_char(logon_time,'yyyy/mm/dd hh24:mi:ss')
from v/$session where paddr in
( select addr from v/$process where spid in($1));

select sql_text from v/$sqltext_with_newlines
where hash_value in
(select SQL_HASH_VALUE from v/$session where
paddr in (select addr from v/$process where spid=$1)
)
order by piece;

exit;
EOF
然后,我们只要在 OS 环境下如下执行即可
$./whoit.sh Spid

5.          [Q] 怎么样分析表或索引
[A]
命令行方式可以采用 analyze 命令
Analyze table tablename compute statistics;
Analyze index|cluster indexname estimate statistics;
ANALYZE TABLE tablename COMPUTE STATISTICS
FOR TABLE
FOR ALL [LOCAL] INDEXES
FOR ALL [INDEXED] COLUMNS;
ANALYZE TABLE tablename DELETE STATISTICS
ANALYZE TABLE tablename VALIDATE REF UPDATE
ANALYZE TABLE tablename VALIDATE STRUCTURE
[CASCADE]|[INTO TableName]
ANALYZE TABLE tablename LIST CHAINED ROWS [INTO TableName]
等等。
如果想分析整个用户或数据库,还可以采用工具包,可以并行分析
Dbms_utility(8i
以前的工具包 )
Dbms_stats(8i
以后提供的工具包 )

dbms_stats.gather_schema_stats(User,estimate_percent=>100,cascade=> TRUE);
dbms_stats.gather_table_stats(User,TableName,degree => 4,cascade => true);
这是对命令与工具包的一些总结
1
、对于分区表,建议使用 DBMS_STATS ,而不是使用 Analyze 语句。
a)
可以并行进行,对多个用户,多个 Table
b)
可以得到整个分区表的数据和单个分区的数据。
c)
可以在不同级别上 Compute Statistics :单个分区,子分区,全表,所有分区
d)
可以倒出统计信息
e)
可以用户自动收集统计信息
2
DBMS_STATS 的缺点
a)
不能 Validate Structure
b)
不能收集 CHAINED ROWS, 不能收集 CLUSTER TABLE 的信息,这两个仍旧需要使用 Analyze 语句。
c) DBMS_STATS
默认不对索引进行 Analyze ,因为默认 Cascade False ,需要手工指定为 True
3
、对于 oracle 9 里面的 External Table Analyze 不能使用,只能使用 DBMS_STATS 来收集信息。

6.          [Q] 怎么样快速重整索引
[A]
通过 rebuild 语句,可以快速重整或移动索引到别的表空间
rebuild
有重建整个索引数的功能,可以在不删除原始索引的情况下改变索引的存储参数
语法为
alter index index_name rebuild tablespace ts_name
storage(……);
如果要快速重建整个用户下的索引,可以用如下脚本,当然,需要根据你自己的情况做相应修改
SQL> set heading off
SQL> set feedback off
SQL> spool d:/index.sql
SQL> SELECT 'alter index ' || index_name || ' rebuild '
||'tablespace INDEXES storage(initial 256K next 256K pctincrease 0);'
FROM all_indexes
WHERE ( tablespace_name != 'INDEXES'
OR next_extent != ( 256 * 1024 )
)
AND owner = USER
SQL>spool off
另外一个合并索引的语句是
alter index index_name coalesce
,这个语句仅仅是合并索引中同一级的 leaf block
消耗不大,对于有些索引中存在大量空间浪费的情况下,有一些作用。

7.          [Q] 如何使用 Hint 提示
[A]
select/delete/update 后写 /*+ hint */
select /*+ index(TABLE_NAME INDEX_NAME) */ col1...
注意 /* + 之间不能有空格
如用 hint 指定使用某个索引

select /*+ index(cbotab) */ col1 from cbotab;
select /*+ index(cbotab cbotab1) */ col1 from cbotab;
select /*+ index(a cbotab1) */ col1 from cbotab a;
其中
TABLE_NAME
是必须要写的,且如果在查询中使用了表的别名,在 hint 也要用表的别名来代替表名 ;
INDEX_NAME
可以不必写, Oracle 会根据统计值选一个索引 ;
如果索引名或表名写错了,那这个 hint 就会被忽略 ;

8.          [Q] 怎么样快速复制表或者是插入数据
[A]
快速复制表可以指定 Nologging 选项
如: Create table t1 nologging
as select * from t2;
快速插入数据可以指定 append 提示,但是需要注意
noarchivelog
模式下,默认用了 append 就是 nologging 模式的。
archivelog 下,需要把表设置程 Nologging 模式。
insert /*+ append */ into t1
select * from t2
注意:如果在 9i 环境中并设置了 FORCE LOGGING ,则以上操作是无效的,并不会加快,当然,可以通过如下语句设置为 NO FORCE LOGGING
Alter database no force logging;
是否开启了 FORCE LOGGING ,可以用如下语句查看
SQL> select force_logging from v$database;

9.          [Q] 怎么避免使用特定索引
[A]
在很多时候, Oracle 会错误的使用索引而导致效率的明显下降,我们可以使用一点点技巧而避免使用不该使用的索引,如:
test, 有字段 a,b,c,d ,在 a,b,c 上建立联合索引 inx_a(a,b,c) ,在 b 上单独建立了一个索引 Inx_b(b)
在正常情况下, where a=? and b=? and c=? 会用到索引 inx_a
where b=?
会用到索引 inx_b
但是, where a=? and b=? and c=? group by b 会用到哪个索引呢?在分析数据不正确(很长时间没有分析)或根本没有分析数据的情况下, oracle 往往会使用索引 inx_b 。通过执行计划的分析,这个索引的使用,将大大耗费查询时间。
当然,我们可以通过如下的技巧避免使用 inx_b ,而使用 inx_a
where a=? and b=? and c=? group by b||'' --
如果 b 是字符
where a=? and b=? and c=? group by b+0 --
如果 b 是数字
通过这样简单的改变,往往可以是查询时间提交很多倍
当然,我们也可以使用 no_index 提示,相信很多人没有用过,也是一个不错的方法:
select /*+ no_index(t,inx_b) */ * from test t
where a=? and b=? and c=? group by b

10.       [Q]Oracle 什么时候会使用跳跃式索引扫描
[A]
这是 9i 的一个新特性跳跃式索引扫描 (Index Skip Scan).
例如表有索引 index(a,b,c) ,当查询条件为
where b=?
的时候,可能会使用到索引 index(a,b,c)
如,执行计划中出现如下计划:
INDEX (SKIP SCAN) OF 'TEST_IDX' (NON-UNIQUE)
Oracle
的优化器 ( 这里指的是 CBO) 能对查询应用 Index Skip Scans 至少要有几个条件 :
1
优化器认为是合适的。
2
索引中的前导列的唯一值的数量能满足一定的条件(如重复值很多)。
3
优化器要知道前导列的值分布 ( 通过分析 / 统计表得到 )
4
合适的 SQL 语句
等。

11.       [Q] 怎么样创建使用虚拟索引
[A]
可以使用 nosegment 选项,如
create index virtual_index_name on table_name(col_name) nosegment;
如果在哪个 session 需要测试虚拟索引,可以利用隐含参数来处理
alter session set "_use_nosegment_indexes" = true;
就可以利用 explain plan for select …… 来看虚拟索引的效果
利用 @$ORACLE_HOME/rdbms/admin/utlxpls 查看执行计划
最后,根据需要,我们可以删除虚拟索引,如普通索引一样
drop index virtual_index_name;
注意:虚拟索引并不是物理存在的,所以虚拟索引并不等同于物理索引,不要用自动跟踪去测试虚拟索引,因为那是实际执行的效果,是用不到虚拟索引的。

12.       [Q] 怎样监控无用的索引
[A]Oracle 9i
以上,可以监控索引的使用情况,如果一段时间内没有使用的索引,一般就是无用的索引
语法为:
开始监控: alter index index_name monitoring usage;
检查使用状态: select * from v$object_usage;
停止监控: alter index index_name nomonitoring usage;
当然,如果想监控整个用户下的索引,可以采用如下的脚本:
set heading off
set echo off
set feedback off
set pages 10000
spool start_index_monitor.sql
SELECT 'alter index '||owner||'.'||index_name||' monitoring usage;'
FROM dba_indexes
WHERE owner = USER;
spool off
set heading on
set echo on
set feedback on
------------------------------------------------
set heading off
set echo off
set feedback off
set pages 10000
spool stop_index_monitor.sql
SELECT 'alter index '||owner||'.'||index_name||' nomonitoring usage;'
FROM dba_indexes
WHERE owner = USER;
spool off
set heading on
set echo on
set feedback on

13.       [Q] 怎么样能固定我的执行计划
[A]
可以使用 OUTLINE 来固定 SQL 语句的执行计划
用如下语句可以创建一个 OUTLINE
Create oe replace outline OutLn_Name on
Select Col1,Col2 from Table
where ……
如果要删除 Outline ,可以采用
Drop Outline OutLn_Name;
对于已经创建了的 OutLine ,存放在 OUTLN 用户的 OL$HINTS 表下面
对于有些语句,你可以使用 update outln.ol$hints 来更新 outline
update outln.ol$hints(ol_name,'TEST1','TEST2','TEST2','TEST1)
where ol_name in ('TEST1','TEST2');
这样,你就把 Test1 OUTLINE Test2 OUTLINE 互换了
如果想利用已经存在的 OUTLINE ,需要设置以下参数
Alter system/session set Query_rewrite_enabled = true
Alter system/session set use_stored_outlines = true

14.       [Q]v$sysstat 中的 class 分别代表什么
[A]
统计类别
1
代表事例活动
2
代表 Redo buffer 活动
4
代表锁
8
代表数据缓冲活动
16
代表 OS 活动
32
代表并行活动
64
代表表访问
128
代表调试信息

15.       [Q] 怎么杀掉特定的数据库会话
[A] Alter system kill session 'sid,serial#';
或者
alter system disconnect session 'sid,serial#' immediate;
win 上,还可以采用 oracle 提供的 orakill 杀掉一个线程(其实就是一个 Oracle 进程)
Linux/Unix 上,可以直接利用 kill 杀掉数据库进程对应的 OS 进程

16.       [Q] 怎么快速查找锁与锁等待
[A]
数据库的锁是比较耗费资源的,特别是发生锁等待的时候,我们必须找到发生等待的锁,有可能的话,杀掉该进程。
这个语句将查找到数据库中所有的 DML 语句产生的锁,还可以发现,任何 DML 语句其实产生了两个锁,一个是表锁,一个是行锁。
可以通过 alter system kill session ‘sid,serial#’ 来杀掉会话
SELECT /*+ rule */ s.username,
decode(l.type,'TM','TABLE LOCK',
'TX','ROW LOCK',
NULL) LOCK_LEVEL,
o.owner,o.object_name,o.object_type,
s.sid,s.serial#,s.terminal,s.machine,s.program,s.osuser
FROM v$session s,v$lock l,dba_objects o
WHERE l.sid = s.sid
AND l.id1 = o.object_id(+)
AND s.username is NOT NULL
如果发生了锁等待,我们可能更想知道是谁锁了表而引起谁的等待
以下的语句可以查询到谁锁了表,而谁在等待。
SELECT /*+ rule */ lpad(' ',decode(l.xidusn ,0,3,0))||l.oracle_username User_name,
o.owner,o.object_name,o.object_type,s.sid,s.serial#
FROM v$locked_object l,dba_objects o,v$session s
WHERE l.object_id=o.object_id
AND l.session_id=s.sid
ORDER BY o.object_id,xidusn DESC
以上查询结果是一个树状结构,如果有子节点,则表示有等待发生。如果想知道锁用了哪个回滚段,还可以关联到 V$rollname ,其中 xidusn 就是回滚段的 USN

17.       [Q] 如何有效的删除一个大表 (extent 数很多 的表 )
[A]
一个有很多 (100k)extent 的表,如果只是简单地用 drop table 的话,会很大量消耗 CPU Oracle 要对 fet$ uet$ 数据字典进行操作),可能会用上几天的时间,较好的方法是分多次删除 extent ,以减轻这种消耗:
1. truncate table big-table reuse storage;
2. alter table big-table deallocate unused keep 2000m (
原来大小的 n-1/n);
3. alter table big-table deallocate unused keep 1500m ;
....
4. drop table big-table;

18.       [Q] 如何收缩临时数据文件的大小
[A]9i
以下版本采用
ALTER DATABASE DATAFILE 'file name' RESIZE 100M
类似的语句
9i
以上版本采用
ALTER DATABASE TEMPFILE 'file name' RESIZE 100M
注意,临时数据文件在使用时,一般不能收缩,除非关闭数据库或断开所有会话,停止对临时数据文件的使用。

19.       [Q] 怎么清理临时段
[A]
可以使用如下办法
1
使用如下语句查看一下认谁在用临时段
SELECT username,sid,serial#,sql_address,machine,program,
tablespace,segtype, contents
FROM v$session se,v$sort_usage su
WHERE se.saddr=su.session_addr
2
那些正在使用临时段的进程
SQL>Alter system kill session 'sid,serial#';
3
、把 TEMP 表空间 回缩一下
SQL>Alter tablespace TEMP coalesce;
还可以使用诊断事件
1
确定 TEMP 表空间 ts#
SQL> select ts#, name FROM v$tablespace;
TS# NAME
-----------------------
0 SYSYEM
1 RBS
2 USERS
3* TEMP
……
2
执行清理操作
alter session set events 'immediate trace name DROP_SEGMENTS level TS#+1'
说明:
temp
表空间的 TS# 3*, So TS#+ 1= 4
如果想清除所有表空间的临时段,则
TS# = 2147483647

20.       [Q] 怎么样 dump 数据库内部结构,如上面显示的控制文件的结构
[A]
常见的有
1
、分析数据文件块,转储数据文件 n 的块 m
alter system dump datafile n block m
2
、分析日志文件
alter system dump logfile logfilename;
3
、分析控制文件的内容
alter session set events 'immediate trace name CONTROLF level 10'
4
、分析所有数据文件头
alter session set events 'immediate trace name FILE_HDRS level 10'
5
、分析日志文件头
alter session set events 'immediate trace name REDOHDR level 10'
6
、分析系统状态,最好每 10 分钟一次,做三次对比
alter session set events 'immediate trace name SYSTEMSTATE level 10'
7
、分析进程状态
alter session set events 'immediate trace name PROCESSSTATE level 10'
8
、分析 Library Cache 的详细情况
alter session set events 'immediate trace name library_cache level 10'

21.       [Q] 如何获得所有的事件代码
[A]
事件代码范围一般从 10000 to 10999 ,以下列出了这个范围的事件代码与信息
SET SERVEROUTPUT ON
DECLARE
err_msg VARCHAR2(120);
BEGIN
dbms_output.enable (1000000);
FOR err_num IN 10000..10999
LOOP
err_msg := SQLERRM (-err_num);
IF err_msg NOT LIKE '%Message '||err_num||' not found%' THEN
dbms_output.put_line (err_msg);
END IF;
END LOOP;
END;
/
Unix 系统上,事件信息放在一个文本文件里
$ORACLE_HOME/rdbms/mesg/oraus.msg
可以用如下脚本查看事件信息
event=10000
while [ $event -ne 10999 ]
do
event=`expr $event + 1`
oerr ora $event
done
对于已经确保的 / 正在跟踪的事件,可以用如下脚本获得
SET SERVEROUTPUT ON
DECLARE
l_level NUMBER;
BEGIN
FOR l_event IN 10000..10999
LOOP
dbms_system.read_ev (l_event,l_level);
IF l_level > 0 THEN
dbms_output.put_line ('Event '||TO_CHAR (l_event)||
' is set at level '||TO_CHAR (l_level));
END IF;
END LOOP;
END;
/

22.       [Q] 什么是 STATSPACK ,我怎么使用它?
[A]Statspack
Oracle 8i 以上提供的一个非常好的性能监控与诊断工具,基本上全部包含了 BSTAT/ESTAT 的功能,更多的信息
可以参考附带文档 $ORACLE_HOME/rdbms/admin/spdoc.txt
安装 Statspack:
cd $ORACLE_HOME/rdbms/admin
sqlplus "/ as sysdba" @spdrop.sql --
卸载,第一次可以不需要
sqlplus "/ as sysdba" @spcreate.sql --
需要根据提示输入表空间名
使用 Statspack:
sqlplus perfstat/perfstat
exec statspack.snap; --
进行信息收集统计,每次运行都将产生一个快照号
--
获得快照号,必须要有两个以上的快照,才能生成报表
select SNAP_ID, SNAP_TIME from STATS$SNAPSHOT;
@spreport.sql --
输入需要查看的开始快照号与结束快照号
其他相关脚本 s:
spauto.sql -
利用 dbms_job 提交一个作业,自动的进行 STATPACK 的信息收集统计
sppurge.sql -
清除一段范围内的统计信息,需要提供开始快照与结束快照号
sptrunc.sql -
清除 (truncate) 所有统计信息
第五部分、 ORACLE网络与安全
1.        [Q] 如何限定特定 IP 访问数据库
[A]
可以利用登录触发器或者是修改 sqlnet.ora 9i 以上):
增加如下内容:
tcp.validnode_checking=yes
#
允许访问的 ip
tcp.inited_nodes=(ip1,ip2,……)
#
不允许访问的 ip
tcp.excluded_nodes=(ip1,ip2,……)

2.        [Q] 如何穿过防火墙连接数据库
[A]
这个问题只会在 WIN 平台出现, UNIX 平台会自动解决。
解决方法:
在服务器端的 SQLNET.ORA 应类似
SQLNET.AUTHENTICATION_SERVICES= (NTS)
NAMES.DIRECTORY_PATH= (TNSNAMES, ONAMES, HOSTNAME)
TRACE_LEVEL_CLIENT = 16
注册表的 HOME0 [HKEY_LOCAL_MACHINE]
USE_SHARED_SOCKET=TRUE

3.        [Q] 如何利用 hostname 方式连接数据库
host name
方式只支持 tcp/ip 协议的小局域网
修改 listener.ora 中的如下信息
(SID_DESC =
(GLOBAL_DBNAME = ur_hostname) --
你的机器名
(ORACLE_HOME = E:/oracle/ora92) --oracle home
(SID_NAME = orcl) --sid name
)
然后在客户端
sqlnet.ora 中,确保有
NAMES.DIRECTORY_PATH= (HOSTNAME)
你就可以利用数据库服务器的名称访问数据库了

4.        [Q]dbms_repcat_admin 能带来什么安全隐患
[A]
如果一个用户能执行 dbms_repcat_admin 包,将获得极大的系统权限。
以下情况可能获得该包的执行权限:
1
、在 sys grant execute on dbms_repcat_admin to public[|user_name]
2
、用户拥有 execute any procedure 特权(仅限于 9i 以下, 9i 必须显示授权)
如果用户通过执行如下语句:
exec sys.dbms_repcat_admin.grant_admin_any_schema('user_name');
该用户将获得极大的系统特权
可以从 user_sys_privs 中获得详细信息

5.        [Q] 在不知道用户密码的时候,怎么样跳转到另外一个用户执行操作后并不影响该用户 ?
[A]
我们通过如下的方法,可以安全使用该用户,然后再跳转回来,在某些时候比较有用
需要 Alter user 权限或 DBA 权限:
SQL> select password from dba_users where username='SCOTT';
PASSWORD
-----------------------------
F894844C34402B67
SQL> alter user scott identified by lion;
User altered.
SQL> connect scott/lion
Connected.
REM Do whatever you like...
SQL> connect system/manager
Connected.
SQL> alter user scott identified by values 'F894844C34402B67';
User altered.
SQL> connect scott/tiger
Connected.

6.        [Q] 如何加固你的数据库
[A]
要注意以下方面
1.
修改 sys, system 的口令。
2. Lock
,修改,删除默认用户: dbsnmp,ctxsys 等。
3.
REMOTE_OS_AUTHENT 改成 False ,防止远程机器直接登陆。
4.
O7_DICTIONARY_ACCESSIBILITY 改成 False
5.
把一些权限从 PUBLIC Role 取消掉。
6.
检查数据库的数据文件的安全性。不要设置成 666 之类的。检查其他 dba 用户。
7.
把一些不需要的服务(比如 ftp, nfs 等关闭掉)
8.
限制数据库主机上面的用户数量。
9.
定期检查 Metalink/OTN 上面的 security Alert 。比如: http://otn.oracle.com/deploy/security/alerts.htm
10.
把你的数据库与应用放在一个单独的子网中,要不然你的用户密码很容易被 sniffer 去。或者采用 advance security ,对用户登录加密。
11.
限止只有某些 ip 才能访问你的数据库。
12. lsnrctl
要加密码,要不然别人很容易从外面关掉你的 listener
13.
如果可能,不要使用默认 1521 端口

7.        [Q] 如何检查用户是否用了默认密码
[A]
如果使用默认密码,很可能就对你的数据库造成一定的安全隐患,那么可以使用如下的查询获得那些用户使用默认密码
select username "User(s) with Default Password!"
from dba_users
where password in
('E066D214D5421CCC', -- dbsnmp
'24ABAB8B06281B4C', -- ctxsys
'72979A94BAD2AF80', -- mdsys
'C252E8FA117AF049', -- odm
'A7A32CD03D3CE8D5', -- odm_mtr
'88A2B2C183431F00', -- ordplugins
'7EFA02EC7EA6B86F', -- ordsys
'4A3BA55E08595C81', -- outln
'F894844C34402B67', -- scott
'3F9FBD883D787341', -- wk_proxy
'79DF7A1BD138CF11', -- wk_sys
'7C9BA362F8314299', -- wmsys
'88D8364765FCE6AF', -- xdb
'F9DA8977092B7B81', -- tracesvr
'9300C0977D7DC75E', -- oas_public
'A97282CE3D94E29E', -- websys
'AC9700FD3F1410EB', -- lbacsys
'E7B5D92911C831E1', -- rman
'AC98877DE1297365', -- perfstat
'66F4EF5650C20355', -- exfsys
'84B8CBCA4D477FA3', -- si_informtn_schema
'D4C5016086B2DC6A', -- sys
'D4DF7931AB130E37') -- system
/

8.        [Q] 如何修改默认的 XDB 监听端口
[A] Oracle9i
默认的 XML DB HTTP 的默认端口设为 8080 ,这是一个太常用的端口了,很多别的 WebServer 都会使用这个端口,
如果我们安装了它,最好修改一下,避免冲突,如果不使用呢,就最好不要安装
提供三种修改的方法
1.dbca
,选择你的数据库,然后 Standard Database Features->Customize->Oracle XML DB option ,进入这个画面你应该就知道怎么改了。
2.OEM console
,在 XML Database 的配置里面修改
3.
oracle 提供的包:
--
HTTP/WEBDAV 端口从 8080 改到 8081
SQL> call dbms_xdb.cfg_update(updateXML(dbms_xdb.cfg_get(),
'/xdbconfig/sysconfig/protocolconfig/httpconfig/http-port/text()',8081))
/
--
FTP 端口从 2100 改到 2111
SQL> call dbms_xdb.cfg_update(updateXML(dbms_xdb.cfg_get(),
'/xdbconfig/sysconfig/protocolconfig/ftpconfig/ftp-port/text()',2111))
/
SQL> commit;
SQL> exec dbms_xdb.cfg_refresh;
--
检查修改是否已经成功
SQL> select dbms_xdb.cfg_get from dual;

第六部分、OS相关与其它
1.        [Q] 怎么样生成日期格式的文件
[A]
LINUX/UNIX 上,使用 `date +%y%m%d` (` 这个是键盘上 ~ 所在的那个键 ) $(date +%y%m%d) ,如:
touch exp_table_name_`date +%y%m%d`.dmp
DATE=$(date +%y%m%d)
或者
DATE=$(date +%Y%m%d --date '1 days ago') #
获取昨天或多天前的日期
Windows
上,使用 %date:~4,10% ,其中 4 是开始字符, 10 是提取长度,表示从 date 生成的日期中,提取从 4 开始长度是 10 的串。你可以改成其它你需要的数字,如:
Echo %date:~4,10%
如果想得到更精确的时间, win 上面还可以使用 time

2.        [Q] 测试磁盘与阵列性能
[A]
用类似如下的方法测试写能力
time dd if=/dev/zero of=/oradata/biddb/testind/testfile.dbf bs=1024000 count=1000
期间系统 IO 使用可以用 (unix):
iostat -xnp 2
显示 Busy 程度

3.        [Q] 怎么配置 SSH 密匙
[A]
可以防止 " 中间人 " 的进攻方式
1
ssh-keygen ssh-keygen -d(ssh 2.x) 生成钥匙
2
、然后拷贝公匙到你想登录的服务器,改名为 authorized_keys ,如果是 3.0 以下版本,需要改为 authorized_keys2
3
、还可以利用 config 文件进一步简化操作

Host *bj
HostName
机器名或 IP
User
用户名
有了这个配置文件,你就可以利用 ssh bj 来访问指定的机器了,也就可以利用 scp sftp 来传送文件了。

4.        [Q]FTP 怎么在脚本中自动上传 / 下载
[A]
可以把 FTP 写到 shell 脚本中,如
ftp -n -i
主机 IP <<EOF
user username pass
cd
目标目录
put file
get file
#
查询文件
ls
#
退出
bye
EOF
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值