linux signal 用法和注意事項

http://blog.chinaunix.net/uid-9354-id-2425031.htmlhtml



因此但願能用相同方式處理信號的屢次出現,最好用sigaction.信號只出現並處理一次,能夠用signal.程序員

 

signal函數每次設置具體的信號處理函數(非SIG_IGN)只能生效一次,每次在進程響應處理信號時,隨即將信號處理函數恢復爲默認處理方式.因此若是想屢次相同方式處理某個信號,一般的作法是,在響應函數開始,再次調用signal設置,以下圖:編程

int sig_int(); //My signal handler

    ...
    signal(SIGINT, sig_int);
    ...

int sig_int()
{
    signal(SIGINT, sig_int);
    ....
}異步

 

 

這種代碼段的一個問題是:在信號發生以後到信號處理程序中調用s i g n a l函數之間有一個
時間窗口。在此段時間中,可能發生另外一次中斷信號。第二個信號會形成執行默認動做,而對
中斷信號則是終止該進程。這種類型的程序段在大多數狀況下會正常工做,使得咱們認爲它們
正確,而實際上卻並非如此。
另外一個問題是:在進程不但願某種信號發生時,它不能關閉該信號。


sigaction: 
1.在信號處理程序被調用時,系統創建的新信號屏蔽字會自動包括正被遞送的信號。所以保證了在處理一個
給定的信號時,若是這種信號再次發生,那麼它會被阻塞到對前一個信號的處理結束爲止
2.響應函數設置後就一直有效,不會重置
3.對除S I G A L R M之外的全部信號都企圖設置S A _ R E S TA RT標誌,因而被這些信號中斷
的系統調用(read,write)都能自動再起動。不但願再起動由S I G A L R M信號中斷的系統調用的緣由是但願對I / O操做能夠設置時間限制。
 
因此但願能用相同方式處理信號的屢次出現,最好用sigaction.信號只出現並處理一次,能夠用signal

//////////////////////////////////////////////////////////////////////////////////////函數

 

 

信號是Linux編程中很是重要的部分,本文將詳細介紹信號機制的基本概念、Linux對信號機制的大體實現方法、如何使用信號,以及有關信號的幾個系統調用。 

信號機制是進程之間相互傳遞消息的一種方法,信號全稱爲軟中斷信號,也有人稱做軟中斷。從它的命名能夠看出,它的實質和使用很象中斷。因此,信號能夠說是進程控制的一部分。 

1、信號的基本概念 

本節先介紹信號的一些基本概念,而後給出一些基本的信號類型和信號對應的事件。基本概念對於理解和使用信號,對於理解信號機制都特別重要。下面就來看看什麼是信號。 

一、基本概念 

軟中斷信號(signal,又簡稱爲信號)用來通知進程發生了異步事件。進程之間能夠互相經過系統調用kill發送軟中斷信號。內核也能夠由於內部事件而給進程發送信號,通知進程發生了某個事件。注意,信號只是用來通知某進程發生了什麼事件,並不給該進程傳遞任何數據。 

收 到信號的進程對各類信號有不一樣的處理方法。處理方法能夠分爲三類:第一種是相似中斷的處理程序,對於須要處理的信號,進程能夠指定處理函數,由該函數來處 理。第二種方法是,忽略某個信號,對該信號不作任何處理,就象未發生過同樣。第三種方法是,對該信號的處理保留系統的默認值,這種缺省操做,對大部分的信 號的缺省操做是使得進程終止。進程經過系統調用signal來指定進程對某個信號的處理行爲。 

在進程表的表項中有一個軟中斷信號域,該域中每一位對應一個信號,當有信號發送給進程時,對應位置位。由此能夠看出,進程對不一樣的信號能夠同時保留,但對於同一個信號,進程並不知道在處理以前來過多少個。 

二、信號的類型 

發出信號的緣由不少,這裏按發出信號的緣由簡單分類,以瞭解各類信號: 

(1) 與進程終止相關的信號。當進程退出,或者子進程終止時,發出這類信號。 
(2) 與進程例外事件相關的信號。如進程越界,或企圖寫一個只讀的內存區域(如程序正文區),或執行一個特權指令及其餘各類硬件錯誤。 
(3) 與在系統調用期間遇到不可恢復條件相關的信號。如執行系統調用exec時,原有資源已經釋放,而目前系統資源又已經耗盡。 
(4) 與執行系統調用時遇到非預測錯誤條件相關的信號。如執行一個並不存在的系統調用。 
(5) 在用戶態下的進程發出的信號。如進程調用系統調用kill向其餘進程發送信號。 
(6) 與終端交互相關的信號。如用戶關閉一個終端,或按下break鍵等狀況。 
(7) 跟蹤進程執行的信號。 

Linux支持的信號列表以下。不少信號是與機器的體系結構相關的,首先列出的是POSIX.1中列出的信號: 

信號 值 處理動做 發出信號的緣由 
---------------------------------------------------------------------- 
SIGHUP 1 A 終端掛起或者控制進程終止 
SIGINT 2 A 鍵盤中斷(如break鍵被按下) 
SIGQUIT 3 C 鍵盤的退出鍵被按下 
SIGILL 4 C 非法指令 
SIGABRT 6 C 由abort(3)發出的退出指令 
SIGFPE 8 C 浮點異常 
SIGKILL 9 AEF Kill信號 
SIGSEGV 11 C 無效的內存引用 
SIGPIPE 13 A 管道破裂: 寫一個沒有讀端口的管道 
SIGALRM 14 A 由alarm(2)發出的信號 
SIGTERM 15 A 終止信號 
SIGUSR1 30,10,16 A 用戶自定義信號1 
SIGUSR2 31,12,17 A 用戶自定義信號2 
SIGCHLD 20,17,18 B 子進程結束信號 
SIGCONT 19,18,25 進程繼續(曾被中止的進程) 
SIGSTOP 17,19,23 DEF 終止進程 
SIGTSTP 18,20,24 D 控制終端(tty)上按下中止鍵 
SIGTTIN 21,21,26 D 後臺進程企圖從控制終端讀 
SIGTTOU 22,22,27 D 後臺進程企圖從控制終端寫 

下面的信號沒在POSIX.1中列出,而在SUSv2列出 

信號 值 處理動做 發出信號的緣由 
-------------------------------------------------------------------- 
SIGBUS 10,7,10 C 總線錯誤(錯誤的內存訪問) 
SIGPOLL A Sys V定義的Pollable事件,與SIGIO同義 
SIGPROF 27,27,29 A Profiling定時器到 
SIGSYS 12,-,12 C 無效的系統調用 (SVID) 
SIGTRAP 5 C 跟蹤/斷點捕獲 
SIGURG 16,23,21 B Socket出現緊急條件(4.2 BSD) 
SIGVTALRM 26,26,28 A 實際時間報警時鐘信號(4.2 BSD) 
SIGXCPU 24,24,30 C 超出設定的CPU時間限制(4.2 BSD) 
SIGXFSZ 25,25,31 C 超出設定的文件大小限制(4.2 BSD) 

(對於SIGSYS,SIGXCPU,SIGXFSZ,以及某些機器體系結構下的SIGBUS,Linux缺省的動做是A (terminate),SUSv2 是C (terminate and dump core))。 

下面是其它的一些信號 

信號 值 處理動做 發出信號的緣由 
---------------------------------------------------------------------- 
SIGIOT 6 C IO捕獲指令,與SIGABRT同義 
SIGEMT 7,-,7 
SIGSTKFLT -,16,- A 協處理器堆棧錯誤 
SIGIO 23,29,22 A 某I/O操做如今能夠進行了(4.2 BSD) 
SIGCLD -,-,18 A 與SIGCHLD同義 
SIGPWR 29,30,19 A 電源故障(System V) 
SIGINFO 29,-,- A 與SIGPWR同義 
SIGLOST -,-,- A 文件鎖丟失 
SIGWINCH 28,28,20 B 窗口大小改變(4.3 BSD, Sun) 
SIGUNUSED -,31,- A 未使用的信號(will be SIGSYS) 

(在這裏,- 表示信號沒有實現;有三個值給出的含義爲,第一個值一般在Alpha和Sparc上有效,中間的值對應i386和ppc以及sh,最後一個值對應mips。信號29在Alpha上爲SIGINFO / SIGPWR ,在Sparc上爲SIGLOST。) 

處理動做一項中的字母含義以下 
A 缺省的動做是終止進程 
B 缺省的動做是忽略此信號 
C 缺省的動做是終止進程並進行內核映像轉儲(dump core) 
D 缺省的動做是中止進程 
E 信號不能被捕獲 
F 信號不能被忽略 

上 面介紹的信號是常見系統所支持的。以表格的形式介紹了各類信號的名稱、做用及其在默認狀況下的處理動做。各類默認處理動做的含義是:終止程序是指進程退 出;忽略該信號是將該信號丟棄,不作處理;中止程序是指程序掛起,進入中止情況之後還能從新進行下去,通常是在調試的過程當中(例如ptrace系統調 用);內核映像轉儲是指將進程數據在內存的映像和進程在內核結構中存儲的部份內容以必定格式轉儲到文件系統,而且進程退出執行,這樣作的好處是爲程序員提 供了方便,使得他們能夠獲得進程當時執行時的數據值,容許他們肯定轉儲的緣由,而且能夠調試他們的程序。 

注意 信號SIGKILL和SIGSTOP既不能被捕捉,也不能被忽略。信號SIGIOT與SIGABRT是一個信號。能夠看出,同一個信號在不一樣的系統中值可能不同,因此建議最好使用爲信號定義的名字,而不要直接使用信號的值。 

2、信 號 機 制 

上 一節中介紹了信號的基本概念,在這一節中,咱們將介紹內核如何實現信號機制。即內核如何向一個進程發送信號、進程如何接收一個信號、進程怎樣控制本身對信 號的反應、內核在什麼時機處理和怎樣處理進程收到的信號。還要介紹一下setjmp和longjmp在信號中起到的做用。 

一、內核對信號的基本處理方法 

內 核給一個進程發送軟中斷信號的方法,是在進程所在的進程表項的信號域設置對應於該信號的位。這裏要補充的是,若是信號發送給一個正在睡眠的進程,那麼要看 該進程進入睡眠的優先級,若是進程睡眠在可被中斷的優先級上,則喚醒進程;不然僅設置進程表中信號域相應的位,而不喚醒進程。這一點比較重要,由於進程檢 查是否收到信號的時機是:一個進程在即將從內核態返回到用戶態時;或者,在一個進程要進入或離開一個適當的低調度優先級睡眠狀態時。 

內核處理一個進程收到的信號的時機是在一個進程從內核態返回用戶態時。因此,當一個進程在內核態下運行時,軟中斷信號並不當即起做用,要等到將返回用戶態時才處理。進程只有處理完信號纔會返回用戶態,進程在用戶態下不會有未處理完的信號。 

內 核處理一個進程收到的軟中斷信號是在該進程的上下文中,所以,進程必須處於運行狀態。前面介紹概念的時候講過,處理信號有三種類型:進程接收到信號後退 出;進程忽略該信號;進程收到信號後執行用戶設定用系統調用signal的函數。當進程接收到一個它忽略的信號時,進程丟棄該信號,就象沒有收到該信號似 的繼續運行。若是進程收到一個要捕捉的信號,那麼進程從內核態返回用戶態時執行用戶定義的函數。並且執行用戶定義的函數的方法很巧妙,內核是在用戶棧上創 建一個新的層,該層中將返回地址的值設置成用戶定義的處理函數的地址,這樣進程從內核返回彈出棧頂時就返回到用戶定義的函數處,從函數返回再彈出棧頂時, 才返回原先進入內核的地方。這樣作的緣由是用戶定義的處理函數不能且不容許在內核態下執行(若是用戶定義的函數在內核態下運行的話,用戶就能夠得到任何權 限)。 

在信號的處理方法中有幾點特別要引發注意。第一,在一些系統中,當一個進程處理完中斷信號返回用戶態以前,內核清除用戶區中設 定的對該信號的處理例程的地址,即下一次進程對該信號的處理方法又改成默認值,除非在下一次信號到來以前再次使用signal系統調用。這可能會使得進程 在調用signal以前又獲得該信號而致使退出。在BSD中,內核再也不清除該地址。但不清除該地址可能使得進程由於過多過快的獲得某個信號而致使堆棧溢 出。爲了不出現上述狀況。在BSD系統中,內核模擬了對硬件中斷的處理方法,即在處理某個中斷時,阻止接收新的該類中斷。 

第二個要 引發注意的是,若是要捕捉的信號發生於進程正在一個系統調用中時,而且該進程睡眠在可中斷的優先級上,這時該信號引發進程做一次longjmp,跳出睡眠 狀態,返回用戶態並執行信號處理例程。當從信號處理例程返回時,進程就象從系統調用返回同樣,但返回了一個錯誤代碼,指出該次系統調用曾經被中斷。這要注 意的是,BSD系統中內核能夠自動地從新開始系統調用。 

第三個要注意的地方:若進程睡眠在可中斷的優先級上,則當它收到一個要忽略的信號時,該進程被喚醒,但不作longjmp,通常是繼續睡眠。但用戶感受不到進程曾經被喚醒,而是象沒有發生過該信號同樣。 

第 四個要注意的地方:內核對子進程終止(SIGCLD)信號的處理方法與其餘信號有所區別。當進程檢查出收到了一個子進程終止的信號時,缺省狀況下,該進程 就象沒有收到該信號似的,若是父進程執行了系統調用wait,進程將從系統調用wait中醒來並返回wait調用,執行一系列wait調用的後續操做(找 出僵死的子進程,釋放子進程的進程表項),而後從wait中返回。SIGCLD信號的做用是喚醒一個睡眠在可被中斷優先級上的進程。若是該進程捕捉了這個 信號,就象普通訊號處理同樣轉處處理例程。若是進程忽略該信號,那麼系統調用wait的動做就有所不一樣,由於SIGCLD的做用僅僅是喚醒一個睡眠在可被 中斷優先級上的進程,那麼執行wait調用的父進程被喚醒繼續執行wait調用的後續操做,而後等待其餘的子進程。 

若是一個進程調用signal系統調用,並設置了SIGCLD的處理方法,而且該進程有子進程處於僵死狀態,則內核將向該進程發一個SIGCLD信號。 

二、setjmp和longjmp的做用 

前面在介紹信號處理機制時,屢次提到了setjmp和longjmp,但沒有仔細說明它們的做用和實現方法。這裏就此做一個簡單的介紹。 

在 介紹信號的時候,咱們看到多個地方要求進程在檢查收到信號後,從原來的系統調用中直接返回,而不是等到該調用完成。這種進程忽然改變其上下文的狀況,就是 使用setjmp和longjmp的結果。setjmp將保存的上下文存入用戶區,並繼續在舊的上下文中執行。這就是說,進程執行一個系統調用,當由於資 源或其餘緣由要去睡眠時,內核爲進程做了一次setjmp,若是在睡眠中被信號喚醒,進程不能再進入睡眠時,內核爲進程調用longjmp,該操做是內核 爲進程將原先setjmp調用保存在進程用戶區的上下文恢復成如今的上下文,這樣就使得進程能夠恢復等待資源前的狀態,並且內核爲setjmp返回1,使 得進程知道該次系統調用失敗。這就是它們的做用。 

3、有關信號的系統調用 

前面兩節已經介紹了有關信號的大部分知 識。這一節咱們來了解一下這些系統調用。其中,系統調用signal是進程用來設定某個信號的處理方法,系統調用kill是用來發送信號給指定進程的。這 兩個調用能夠造成信號的基本操做。後兩個調用pause和alarm是經過信號實現的進程暫停和定時器,調用alarm是經過信號通知進程定時器到時。所 以在這裏,咱們還要介紹這兩個調用。 

一、signal 系統調用 

系統調用signal用來設定某個信號的處理方法。該調用聲明的格式以下: 
void (*signal(int signum, void (*handler)(int)))(int); 
在使用該調用的進程中加入如下頭文件: 
#include <signal.h> 

上述聲明格式比較複雜,若是不清楚如何使用,也能夠經過下面這種類型定義的格式來使用(POSIX的定義): 
typedef void (*sighandler_t)(int); 
sighandler_t signal(int signum, sighandler_t handler); 
但這種格式在不一樣的系統中有不一樣的類型定義,因此要使用這種格式,最好仍是參考一下聯機手冊。 

在調用中,參數signum指出要設置處理方法的信號。第二個參數handler是一個處理函數,或者是 
SIG_IGN:忽略參數signum所指的信號。 
SIG_DFL:恢復參數signum所指信號的處理方法爲默認值。 

傳遞給信號處理例程的整數參數是信號值,這樣可使得一個信號處理例程處理多個信號。系統調用signal返回值是指定信號signum前一次的處理例程或者錯誤時返回錯誤代碼SIG_ERR。下面來看一個簡單的例子: 

#include <signal.h> 
#include <unistd.h> 
#include <stdio.h> 
void sigroutine(int dunno) { /* 信號處理例程,其中dunno將會獲得信號的值 */ 
switch (dunno) { 
case 1: 
printf("Get a signal -- SIGHUP "); 
break; 
case 2: 
printf("Get a signal -- SIGINT "); 
break; 
case 3: 
printf("Get a signal -- SIGQUIT "); 
break; 

return; 


int main() { 
printf("process id is %d ",getpid()); 
signal(SIGHUP, sigroutine); //* 下面設置三個信號的處理方法 
signal(SIGINT, sigroutine); 
signal(SIGQUIT, sigroutine); 
for (;;) ; 


其中信號SIGINT由按下Ctrl-C發出,信號SIGQUIT由按下Ctrl-發出。該程序執行的結果以下: 

localhost:~$ ./sig_test 
process id is 463 
Get a signal -SIGINT //按下Ctrl-C獲得的結果 
Get a signal -SIGQUIT //按下Ctrl-獲得的結果 
//按下Ctrl-z將進程置於後臺 
[1]+ Stopped ./sig_test 
localhost:~$ bg 
[1]+ ./sig_test & 
localhost:~$ kill -HUP 463 //向進程發送SIGHUP信號 
localhost:~$ Get a signal – SIGHUP 
kill -9 463 //向進程發送SIGKILL信號,終止進程 
localhost:~$ 

二、kill 系統調用 

系統調用kill用來向進程發送一個信號。該調用聲明的格式以下: 
int kill(pid_t pid, int sig); 
在使用該調用的進程中加入如下頭文件: 
#include <sys/types.h> 
#include <signal.h> 

該 系統調用能夠用來向任何進程或進程組發送任何信號。若是參數pid是正數,那麼該調用將信號sig發送到進程號爲pid的進程。若是pid等於0,那麼信 號sig將發送給當前進程所屬進程組裏的全部進程。若是參數pid等於-1,信號sig將發送給除了進程1和自身之外的全部進程。若是參數pid小於- 1,信號sig將發送給屬於進程組-pid的全部進程。若是參數sig爲0,將不發送信號。該調用執行成功時,返回值爲0;錯誤時,返回-1,並設置相應 的錯誤代碼errno。下面是一些可能返回的錯誤代碼: 
EINVAL:指定的信號sig無效。 
ESRCH:參數pid指定的進程或進程組不存在。注意,在進程表項中存在的進程,多是一個尚未被wait收回,但已經終止執行的僵死進程。 
EPERM: 進程沒有權力將這個信號發送到指定接收信號的進程。由於,一個進程被容許將信號發送到進程pid時,必須擁有root權力,或者是發出調用的進程的UID 或EUID與指定接收的進程的UID或保存用戶ID(savedset-user-ID)相同。若是參數pid小於-1,即該信號發送給一個組,則該錯誤 表示組中有成員進程不能接收該信號。 

三、pause系統調用 

系統調用pause的做用是等待一個信號。該調用的聲明格式以下: 
int pause(void); 
在使用該調用的進程中加入如下頭文件: 
#include <unistd.h> 

該調用使得發出調用的進程進入睡眠,直到接收到一個信號爲止。該調用老是返回-1,並設置錯誤代碼爲EINTR(接收到一個信號)。下面是一個簡單的範例: 

#include <unistd.h> 
#include <stdio.h> 
#include <signal.h> 
void sigroutine(int unused) { 
printf("Catch a signal SIGINT "); 


int main() { 
signal(SIGINT, sigroutine); 
pause(); 
printf("receive a signal "); 


在這個例子中,程序開始執行,就象進入了死循環同樣,這是由於進程正在等待信號,當咱們按下Ctrl-C時,信號被捕捉,而且使得pause退出等待狀態。 

四、alarm和 setitimer系統調用 

系統調用alarm的功能是設置一個定時器,當定時器計時到達時,將發出一個信號給進程。該調用的聲明格式以下: 
unsigned int alarm(unsigned int seconds); 
在使用該調用的進程中加入如下頭文件: 
#include <unistd.h> 

系 統調用alarm安排內核爲調用進程在指定的seconds秒後發出一個SIGALRM的信號。若是指定的參數seconds爲0,則再也不發送 SIGALRM信號。後一次設定將取消前一次的設定。該調用返回值爲上次定時調用到發送之間剩餘的時間,或者由於沒有前一次定時調用而返回0。 

注意,在使用時,alarm只設定爲發送一次信號,若是要屢次發送,就要屢次使用alarm調用。 

對於alarm,這裏再也不舉例。如今的系統中不少程序再也不使用alarm調用,而是使用setitimer調用來設置定時器,用getitimer來獲得定時器的狀態,這兩個調用的聲明格式以下: 
int getitimer(int which, struct itimerval *value); 
int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue); 
在使用這兩個調用的進程中加入如下頭文件: 
#include <sys/time.h> 

該系統調用給進程提供了三個定時器,它們各自有其獨有的計時域,當其中任何一個到達,就發送一個相應的信號給進程,並使得計時器從新開始。三個計時器由參數which指定,以下所示: 
TIMER_REAL:按實際時間計時,計時到達將給進程發送SIGALRM信號。 
ITIMER_VIRTUAL:僅當進程執行時才進行計時。計時到達將發送SIGVTALRM信號給進程。 
ITIMER_PROF:當進程執行時和系統爲該進程執行動做時都計時。與ITIMER_VIR-TUAL是一對,該定時器常常用來統計進程在用戶態和內核態花費的時間。計時到達將發送SIGPROF信號給進程。 

定時器中的參數value用來指明定時器的時間,其結構以下: 
struct itimerval { 
struct timeval it_interval; /* 下一次的取值 */ 
struct timeval it_value; /* 本次的設定值 */ 
}; 

該結構中timeval結構定義以下: 
struct timeval { 
long tv_sec; /* 秒 */ 
long tv_usec; /* 微秒,1秒 = 1000000 微秒*/ 
}; 

在setitimer 調用中,參數ovalue若是不爲空,則其中保留的是上次調用設定的值。定時器將it_value遞減到0時,產生一個信號,並將it_value的值設 定爲it_interval的值,而後從新開始計時,如此往復。當it_value設定爲0時,計時器中止,或者當它計時到期,而it_interval 爲0時中止。調用成功時,返回0;錯誤時,返回-1,並設置相應的錯誤代碼errno: 
EFAULT:參數value或ovalue是無效的指針。 
EINVAL:參數which不是ITIMER_REAL、ITIMER_VIRT或ITIMER_PROF中的一個。 

下面是關於setitimer調用的一個簡單示範,在該例子中,每隔一秒發出一個SIGALRM,每隔0.5秒發出一個SIGVTALRM信號: 

#include <signal.h> 
#include <unistd.h> 
#include <stdio.h> 
#include <sys/time.h> 
int sec; 

void sigroutine(int signo) { 
switch (signo) { 
case SIGALRM: 
printf("Catch a signal -- SIGALRM "); 
break; 
case SIGVTALRM: 
printf("Catch a signal -- SIGVTALRM "); 
break; 

return; 


int main() { 
struct itimerval value,ovalue,value2; 
sec = 5; 

printf("process id is %d ",getpid()); 
signal(SIGALRM, sigroutine); 
signal(SIGVTALRM, sigroutine); 

value.it_value.tv_sec = 1; 
value.it_value.tv_usec = 0; 
value.it_interval.tv_sec = 1; 
value.it_interval.tv_usec = 0; 
setitimer(ITIMER_REAL, &value, &ovalue); 

value2.it_value.tv_sec = 0; 
value2.it_value.tv_usec = 500000; 
value2.it_interval.tv_sec = 0; 
value2.it_interval.tv_usec = 500000; 
setitimer(ITIMER_VIRTUAL, &value2, &ovalue); 

for (;;) ; 


該例子的屏幕拷貝以下: 

localhost:~$ ./timer_test 
process id is 579 
Catch a signal – SIGVTALRM 
Catch a signal – SIGALRM 
Catch a signal – SIGVTALRM 
Catch a signal – SIGVTALRM 
Catch a signal – SIGALRM 
Catch a signal –GVTALRM 

本文簡單介紹了Linux下的信號,若是但願瞭解其餘調用,請參考聯機手冊或其餘文檔。ui

相關文章
相關標籤/搜索