c語(yǔ)言是面向過(guò)程的語(yǔ)言,c++是面向?qū)ο蟮?,不過(guò)面向?qū)ο筮€是需要面向過(guò)程的支持的,所以你會(huì)了C++,基本上離會(huì)c語(yǔ)言就不遠(yuǎn)了...
創(chuàng)新互聯(lián)公司是一家專業(yè)提供龍華企業(yè)網(wǎng)站建設(shè),專注與成都網(wǎng)站設(shè)計(jì)、做網(wǎng)站、成都h5網(wǎng)站建設(shè)、小程序制作等業(yè)務(wù)。10年已為龍華眾多企業(yè)、政府機(jī)構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站制作公司優(yōu)惠進(jìn)行中。
linux是操作系統(tǒng),我不知道你說(shuō)的命令是什么意思,是linux的命令行命令呢?(--即 類似于windows DOS 窗口下的命令;)還是說(shuō)的是linux提供的函數(shù)接口,如 read() write()lseek()函數(shù)調(diào)用;
linux命令行命令是用c語(yǔ)言寫(xiě)出來(lái)的命令工具,相當(dāng)于應(yīng)用程序,不過(guò)用C++也可以寫(xiě)的,不過(guò)使用的是C++的編譯器進(jìn)行編譯罷了,最后生成的程序都能在linux上面跑
命令是查詢當(dāng)前登錄的每個(gè)用戶,它的輸出包括用戶名、終端類型、登錄日期及遠(yuǎn)程主機(jī),在Linux系統(tǒng)中輸入who命令輸出如下:
我們先man一下who,在幫助文檔里可以看到,who命令是讀取/var/run/utmp文件來(lái)得到以上信息的。
我們?cè)賛an一下utmp,知道utmp這個(gè)文件,是二進(jìn)制文件,里面保存的是結(jié)構(gòu)體數(shù)組,這些數(shù)組是struct utmp結(jié)構(gòu)體的。
struct utmp {
short ut_type;
pid_t ut_pid;
char ut_line[UT_LINESIZE];
char ut_id[4];
char ut_user[UT_NAMESIZE];
char ut_host[UT_HOSTSIZE];
struct {
int32_t tv_sec;
int32_t tv_usec;
} ut_tv;
/***等等***/
};
要實(shí)現(xiàn)who只需要把utmp文件的所有結(jié)構(gòu)體掃描過(guò)一遍,把需要的信息顯示出來(lái)就可以了,我們需要的信息有ut_user、ut_line、ut_tv、ut_host。
老師給的初始代碼:who1.c運(yùn)行結(jié)果如下:
需要注意的是utmp中所保存的時(shí)間是以秒和微妙來(lái)計(jì)算的,所以我們需要把這個(gè)時(shí)間轉(zhuǎn)換為我們能看懂的時(shí)間,利用命令man -k time | grep 3搜索C語(yǔ)言中和時(shí)間相關(guān)的函數(shù):
經(jīng)過(guò)搜索發(fā)現(xiàn)了一個(gè)ctime()函數(shù),似乎可以滿足我們的需求,于是對(duì)代碼中關(guān)于時(shí)間的printf進(jìn)行修改:
printf("%s",ctime(utbufp-ut_time));
編譯運(yùn)行發(fā)現(xiàn)出來(lái)的結(jié)果雖然已經(jīng)轉(zhuǎn)換成了我們能看懂的時(shí)間格式,但是很明顯這個(gè)時(shí)間是錯(cuò)的:
搜索一下ut_time這個(gè)宏,發(fā)現(xiàn)它被定義為int32_t類型:
但是ctime()函數(shù)中要求參數(shù)的類型是time_t類型,所以重新定義一下類型,編譯運(yùn)行之后,發(fā)現(xiàn)時(shí)間已經(jīng)改成了正確的,但是發(fā)現(xiàn)()中的內(nèi)容被換行了,猜想ctime()函數(shù)的返回值可能自動(dòng)在最后補(bǔ)了一個(gè)字符\n:
一開(kāi)始想通過(guò)\r\b來(lái)實(shí)現(xiàn)“退行”,但實(shí)踐后發(fā)現(xiàn)并不可取,最后考慮到直接修改字符串中最后一個(gè)字符為\0,讓其字符串結(jié)束,使輸出達(dá)到與系統(tǒng)who命令一樣的效果,即在輸出語(yǔ)句前添加如下代碼:
cp[strlen(cp)-1] = '\0'
最后編譯執(zhí)行效果,發(fā)現(xiàn)解決了該問(wèn)題:
雖然能看出基本上和who指令的執(zhí)行結(jié)果一致,但是并非完全一樣,主要在兩點(diǎn),第一是時(shí)間格式不一樣,第二個(gè)是比who執(zhí)行的結(jié)果多了幾條,需要注意的是utmp中保存的用戶,不僅僅是已經(jīng)登陸的用戶,還有系統(tǒng)的其他服務(wù)所需要的“用戶”,所以在顯出所有登陸用戶的時(shí)候,應(yīng)該過(guò)濾掉其他用戶,只保留登陸用戶。我們可以通過(guò)ut_type來(lái)區(qū)別,登陸用戶的ut_type是USER_PROCESS。
先用if語(yǔ)句對(duì)執(zhí)行結(jié)果進(jìn)行過(guò)濾,效果如下:
接著解決時(shí)間格式問(wèn)題,利用man命令收到了兩個(gè)非常有用的函數(shù):localtime()和strftime(),localtime()是把從1970-1-1零點(diǎn)零分到當(dāng)前時(shí)間系統(tǒng)所偏移的秒數(shù)時(shí)間轉(zhuǎn)換為本地時(shí)間,strftime()則是用來(lái)定義時(shí)間格式的,如:年-月-日,利用這兩個(gè)函數(shù)對(duì)時(shí)間進(jìn)行修改后,結(jié)果顯示終于和系統(tǒng)中who命令一模一樣:
最終完整的代碼如下:
#include stdio.h
#include stdlib.h
#include utmp.h
#include fcntl.h
#include unistd.h
#include time.h
#define SHOWHOST
void show_time(long timeval){
char format_time[40];
struct tm *cp;
cp = localtime(timeval);
strftime(format_time,40,"%F %R",cp);
printf("%s",format_time);
}
int show_info( struct utmp *utbufp )
{
if(utbufp-ut_type == USER_PROCESS){
printf("%-8.8s", utbufp-ut_name);
printf(" ");
printf("%-8.8s", utbufp-ut_line);
printf(" ");
show_time(utbufp-ut_time);
printf(" ");
#ifdef SHOWHOST
printf("(%s)", utbufp-ut_host);
#endif
printf("\n");
}
return 0;
}
int main()
{
struct utmp current_record;
int utmpfd;
int reclen = sizeof(current_record);
if ( (utmpfd = open(UTMP_FILE, O_RDONLY)) == -1 ){
perror( UTMP_FILE );
exit(1);
}
while ( read(utmpfd, current_record, reclen) == reclen )
show_info(current_record);
close(utmpfd);
return 0;
}
根據(jù)調(diào)用需求,可以分為兩種:
一、僅執(zhí)行系統(tǒng)命令,不需要該命令的打印結(jié)果。
這種情況可以用system函數(shù)。形式為
system(cmd);
其中cmd為char*類型的字符串,包含要執(zhí)行的命令,命令的執(zhí)行結(jié)果會(huì)輸出到標(biāo)準(zhǔn)輸出。
比如
system("mkdir?test");
這個(gè)執(zhí)行,會(huì)在當(dāng)前文件夾下創(chuàng)建test文件夾。
二、需要命令執(zhí)行的打印。
雖然同樣可以使用system并重定向到文件,然后打開(kāi)文件讀取,最終刪除文件。但這樣做比較繁瑣,更好的做法是使用popen。
FILE *fp = popen(cmd);
執(zhí)行cmd中的命令,然后可以以C文件操作方式,讀取命令的輸出結(jié)果。比如:
if((fp=popen("pwd","r"))==NULL)//執(zhí)行獲取當(dāng)前目錄的系統(tǒng)命令pwd。
{
printf("執(zhí)行失敗\n");//fp為NULL表示命令執(zhí)行失敗。?
}
else
{
char?s[100];
while(fgets(s,?100,?fp))//獲取文件內(nèi)容。
printf("%s",?s);//輸出結(jié)果。
pclose(fp);//關(guān)閉。
}
Linux C編程中,調(diào)用Linux命令用system函數(shù)最簡(jiǎn)單了,這個(gè)函數(shù)原理是在你編寫(xiě)的那個(gè)程序的內(nèi)部啟動(dòng)另一個(gè)程序或命令,從而創(chuàng)建一個(gè)新進(jìn)程,并等待這個(gè)進(jìn)程執(zhí)行完畢退出。如果正常執(zhí)行,system函數(shù)將返回被執(zhí)行程序或命令的退出碼;如果無(wú)法運(yùn)行這個(gè)程序或命令,將返回錯(cuò)誤代碼127;如果是其他錯(cuò)誤,返回-1。這個(gè)函數(shù)的原型是:
#include stdlib.h
int system(const char *string);
參數(shù)string是將要執(zhí)行的程序文件名或路徑,如果是啟動(dòng)一個(gè)命令就是一個(gè)命令字符串。