ORACLE編譯失效物件小結
在日常資料庫維護過程中,我們會發現資料庫中一些物件(包Package、儲存過程Procedure、函式Function、檢視View、同義詞.....)會失效,呈現無效狀態(INVALID)。有時候需要定期檢查資料庫中存在哪些失效物件,對於存在異常的物件需要重新編譯,有些自動失效的物件,一般會在下次呼叫的時候,會被重新編譯,所以這些不需要人工干預。那麼為什麼物件突然會失效呢?又如何快速、高效的編譯失效物件呢?哪些失效的物件不需要我們去重新編譯呢?
資料庫物件失效原因
資料庫物件失效的原因很多,下面大致歸納了一些常見的原因(有些漏掉的,希望大家補充):
1: 當被引用物件的結構變更時,都會使得相關的依賴物件轉變為INVALID狀態。
資料庫中的物件(儲存過程,函式,包,檢視,觸發器),它們往往需要直接或者間接的引用其它物件,物件的依賴包括直接和間接二種,其中直接依賴是指儲存物件直接依賴於被引用物件,而間接依賴是指物件間接依賴於被引用物件
要檢視被引用的物件,可以透過下面SQL檢視
select * from dba_dependencies where name='&objectname';
select * from all_dependencies where name='&objectname';
select * from user_dependencies where name='&objectname';
舉個簡單例子,檢視V_TEST引用了表TEST,TEST表修改了表結構時,會導致檢視V_TEST變為無效物件。
SQL> CREATE TABLE TEST ( ID NUMBER(10));
Table created.
SQL> CREATE VIEW V_TEST AS SELECT * FROM TEST;
View created.
SQL> SELECT OBJECT_NAME, STATUS FROM DBA_OBJECTS WHERE OBJECT_NAME='V_TEST';
OBJECT_NAME STATUS
------------------- ----------------
V_TEST VALID
--修改表結構,增加一個欄位NAME後,檢視V_TEST變為無效
SQL> ALTER TABLE TEST ADD NAME VARCHAR(12);
Table altered.
SQL> SELECT OBJECT_NAME, STATUS FROM DBA_OBJECTS WHERE OBJECT_NAME='V_TEST';
OBJECT_NAME STATUS
------------------- ----------------
V_TEST INVALID
--查詢檢視V_TEST後,資料庫會重新編譯檢視
SQL> SELECT * FROM V_TEST;
no rows selected
SQL> SELECT OBJECT_NAME, STATUS FROM DBA_OBJECTS WHERE OBJECT_NAME='V_TEST';
OBJECT_NAME STATUS
------------------- ----------------
V_TEST VALID
其實不管檢視,像儲存過程,函式、包等,如果程式碼本身沒有什麼錯誤,只是引用的物件發生了變化。也會失效。但並不影響呼叫,因為ORACLE在呼叫時會自動重新編譯的,如果其它物件變化後導致編譯有錯誤。這時呼叫時重新編譯後也是錯誤並處於失效狀態,所以呼叫會出錯。
2:釋出SQL指令碼時(包、儲存過程、函式等),沒有充分測試,編譯時出錯,這時物件變為無效。
3: 資料庫升級、遷移時,出現大量無效物件(本質原因,個人臆測歸結為原因1)。
4: 諸如此類各種情況:例如,Oracle 會自動維護分割槽索引,對於全域性索引,如果在對分割槽表操作時,沒有指定update index,則會導致全域性索引失效,需要重建。
編譯失效物件的方法
統計失效的物件:
select owner, object_type, status, count(*)
from dba_objects
where status='INVALID'
group by owner, object_type, status
order by owner, object_type
檢視具體失效物件
col owner for a20;
col object_name for a32;
col object_type for a16
col status for a8
select owner, object_name, object_type, status
from dba_objects
where status='INVALID'
order by 1, 2,3;
下文大多參考博文 ,修改並作了總結、整理
1: 使用ALTER *** COMPLIE語句手工進行編譯,這個適用於少數、個別物件失效
alter package <schema name>.<package_name> compile;
alter package <schema name>.<package_name> compile body;
alter view <schema name>.<view_name> compile;
alter trigger <schema).<trigger_name> compile;
2:執行@$ORACLE_HOME/rdbms/admin/utlrp.sql指令碼編譯資料庫失效物件。
許多情況下,由於資料庫的升級或遷移,會導致資料庫中的物件失效。由於物件之間可能存在複雜的倚賴關係,所以手工編譯通常無法順利透過。通常我們會在Oracle的升級指導中看到這個指令碼,Oracle強烈推薦在migration/upgrade/downgrade之後,透過執行此指令碼編譯失效物件。但是注意,Oracle提醒,此指令碼需要用SQLPLUS以SYSDBA身份執行,並且當時資料庫中最好不要有活動事物或DDL操作,否則極容易導致死鎖的出現(這是很容易理解的)。
Oracle highly recommends running this script towards the end of of any migration/upgrade/downgrade.
另外,utlrp.sql 裡面其實呼叫了$ORACLE_HOME/rdbms/admin/utlrcmp.sql來編譯失效物件。
3:ORACLE提供了自動編譯的介面dbms_utility.compile_schema(user,false); 呼叫這個過程就會編譯所有失效的過程、函式、觸發器、包
exec dbms_utility.compile_schema( 'SCOTT' )
4: 一些網友書寫的編譯失效物件(經過整理)
SQL 1: 編譯失效物件
set heading off;
set feedback off;
set echo off;
Set lines 999;
Spool run_invalid.sql
select
'ALTER ' || OBJECT_TYPE || ' ' ||
OWNER || '.' || OBJECT_NAME || ' COMPILE;'
from
dba_objects
where
status = 'INVALID'
and
object_type in ('PACKAGE','FUNCTION','PROCEDURE','TRIGGER','JAVA SOURCE','JAVA CLASS','')
;
spool off;
set heading on;
set feedback on;
set echo on;
@run_invalid.sql
SQL 2:在上面的方法中,只能知道某某編譯失敗,不清楚失敗原因,可以用PL/SQL實現更詳細的錯誤資訊
DECLARE
v_objname user_objects.object_name%TYPE;
v_objtype user_objects.object_type%TYPE;
CURSOR cur IS
SELECT object_name,object_type
FROM USER_OBJECTS
WHERE status = 'INVALID'
AND object_type IN ('FUNCTION','JAVA SOURCE','JAVA CLASS','PROCEDURE','PACKAGE','TRIGGER');
BEGIN
OPEN cur;
LOOP
FETCH cur into v_objname, v_objtype;
EXIT WHEN cur%NOTFOUND;
BEGIN
EXECUTE Immediate 'alter ' || v_objtype || ' ' || v_objname||' Compile';
dbms_output.put_line('編譯' || v_objtype || ' ' || v_objname || '()成功');
EXCEPTION
WHEN OTHERS THEN
dbms_output.put_line('編譯' || v_objtype ||' ' || v_objname || '()失敗.' || SQLERRM);
END;
END LOOP;
CLOSE cur;
END;
參考資料:
http://jzhil2004.blog.163.com/blog/static/275585042010117113214172/
http://blog.csdn.net/tianlesoftware/article/details/4843600
https://www.cnblogs.com/kerrycode/p/3723715.html
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/25469263/viewspace-2789976/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Kaldi在ubuntu 18.04下編譯小結Ubuntu編譯
- 重編譯 invalid 物件(轉)編譯物件
- C++物件模型:編譯分析C++物件模型編譯
- Rust 交叉編譯與條件編譯總結Rust編譯
- 小程式反編譯教程編譯
- 編譯ROCKSDB總結編譯
- MYSQL索引失效的各種情況小結MySql索引
- GCC編譯過程(預處理->編譯->彙編->連結)GC編譯
- 小程式原理之: WXSS 編譯編譯
- 編譯連結過程編譯
- 高峰期謹慎編譯業務物件編譯物件
- 如何反編譯微信小程式?編譯微信小程式
- 編譯、彙編、連結、載入、顯示編譯
- 編譯deno,deno結構解析編譯
- 從編譯連結到cmake編譯
- 編譯器後端總結編譯後端
- Android編譯通過,執行編譯錯誤問題總結Android編譯
- [Vue]元件編寫小結Vue元件
- 微信小程式反編譯~2020年微信小程式編譯
- 一個nvcc編譯的小問題編譯
- JavaScript建立物件方法例項小結JavaScript物件
- GCC編譯和連結過程GC編譯
- C語言中編譯和連結C語言編譯
- Oracle 編譯儲存過程卡死解決方法Oracle編譯儲存過程
- oracle DG 日誌傳輸小結Oracle
- 程式的編譯和連結原理分析編譯
- android反編譯相關命令總結Android編譯
- Oracle儲存過程編譯卡死的解決方法Oracle儲存過程編譯
- 今天好多人 phpstrom 編譯器註冊碼失效了,最新可用註冊碼PHP編譯
- Java編譯與反編譯Java編譯
- [譯] JavaScript — 繼承、委託模式和物件連結JavaScript繼承模式物件
- 微信小程式“反編譯”實戰(一):解包微信小程式編譯
- flutter 編譯報錯總結(不斷更新)Flutter編譯
- vc-vs2019編譯報錯總結編譯
- C語言-->(十四)結構體、巨集、編譯C語言結構體編譯
- IOS繫結touchend事件失效iOS事件
- libusb android ndk編譯--編譯mipsAndroid編譯
- hadoop編譯—+2.x編譯Hadoop編譯