C++控制台程序的命令行參數輸入
1. C/C++語言中的main函數,經常帶有參數argc,argv,如下:
int main(int argc, char** argv) int main(int argc, char* argv[])
argc(第一個形參)必須是整型變量,標識該控制台命令調用中接收到的參數個數,注意,包括該命令程序名字itself。
argv( 第二個形參)必須是指向字符串的指針數組,用於保存argc個參數的具體數據。注意數組起始索引為0開始,到argc-1結束。即:argv[0]為命令行中可執行程序名本身,argv[1]為命令行中第二個參數的內容,依次類推。
加上形參說明后,因此C程序中的main函數的函數頭應寫為:
main (int argc,char *argv[])
2. 控制台程序參數獲取
由於main函數不能被其它函數調用,因此不可能在程序內部取得實際值。那么,在何處把實參值賦予main函數的形參呢? 實際上,main函數的參數值是從操作系統命令行上獲得的。
如何在操作系統命令行獲取參數呢?
(1)在VS中設置時右鍵項目->屬性->調試->命令參數,在命令參數中添加所需參數,字符串之間用空格分開即可。如果是.txt文件,要放在當前目錄下(.cpp所在目錄)【實質為exe生成目錄位置】,不然找不到。
(2)或者:假如你的程序是hello.exe,如果在命令行運行該程序,(首先應該在命令行下用 cd 命令進入到 hello.exe 文件所在目錄) 運行命令為:
hello.exe data.txt //.txt也在.exe所在目錄下
注意:調用控制台程序命令行中輸入的參數在位置上,並沒有與控制台程序main函數中 的兩個形參 一 一 對應的。 因為,main的形參只有二個,而命令行中的參數個數原則上未加限制。argc參數表示了命令行中參數的個數(注意:文件名本身也算一個參數),argc的值是在輸入命令行時由系統按實際參數的個數自動賦予的,即,argv[0]為命令行中可執行程序名本身,argv[1]為命令行中第二個參數的內容,依次類推。
在調用一個可執行程序時,某些情況下需要向程序傳遞參數。如我們可以在控制台中鍵入notepad.exe,回車后將執行記事本程序。
如果我們希望在打開notepad時同時打開一個文本文件,可以在notepad.exe 。后面跟上文件的路徑和名字,如notepad.exe example.txt(文件在當前路徑)
那么程序中如何能得到這些輸入參數呢?這個工作是編譯器幫我們完成的,編譯器將輸入參數的信息放入main函數的參數列表中。
main函數的參數列表保存了輸入參數的信息,第一個參數argc記錄了輸入參數的個數, 第二個參數是字符串數組的,字符串數組的每個單元是char*類型的,指向一個c風格字符串。
例子如下:
notepad.exe example.txt data.txt train.txt
第一個參數 第二個參數 第三個參數 第四個參數
因此:argc = 4
控制台執行上面程序后,程序讀取main函數參數為: argc是4,就是說argv數組中有四個有效單元
argv數組中的第一個單元指向的字符串總是可執行程序的名字,以后的單元指向的字符串依次是程序調用時的參數。
第一單元argv[0]指向的字符串是"notepad.exe"
第二單元argv[1]指向的字符串是"example.txt"
第三單元argv[2]指向的字符串是"data.txt"
第四單元argv[3]指向的字符串是"train.txt"
`上述這個賦值過程是編譯器自動完成的,我們只需要利用argc,argv[]讀出數據就可以了。
3. 簡單示例,控制台程序的不同調用方法,及輸出結果:
int main(int argc, char* argv[]) { cout << "參數個數:" << argc << endl; cout << "參數內容:"; for (int i = 0; i < argc; ++i) { cout << argv[i] << endl; } //std::cout << "Hello World!\n"; system("pause"); }
(1) 控制台調用:
1、按住Shift鍵,鼠標右鍵快捷方式,先打開Powershell窗口。
2、輸入 start cmd 回車
3、這樣就可以打開cmd窗口了,並且cmd的工作目錄就是當前的目錄。
輸入命令行:
ProjectTest.exe 1111 zhongguo renmin 傳見過 中國人民解放拒絕
輸出:
(2)利用VSIDE中的控制台參數進行調用:
程序:
#include <iostream> int main(int argc, char **argv) { while (argc-- > 0) { //printf("argv[%d]:%s\n", argc, argv[argc]); printf("argc:%d,*argv++:%s\n", argc,*argv++); //printf("argc:%d,*++argv:%s\n", *++argv); } }
在VS中設置設置 comant argument:
不進行設置時:
設置參數時:
(3)批處理腳本中.bat文件中,調用可執行文件
編輯“main測試.bat”文件內容為:
ProjectTest Jack 中國 boin747 1.234 3.3333 8888 zhongguo pause
在cmd運行.bat文件
雙擊:
好像識別漢字出現了問題,這里只是測試,能達到示意main函數的傳參即可,我就不深因了。
如果把漢字改為數字和字母,則運行效果如預期:
4. 較為復雜的命令行程序,規范化輸入,編寫規范
下面摘錄一個例程,該程序用於取出在argv中的命令行選項。在例子中,將支持下列用法:
//C/C++處理main()函數命令行 #include <iostream> #include <vector> #include <string> #include <ctype.h>//調用atoi函數 using namespace std; const char* const prog_name = "prog"; const char* const prog_version = "version 1.0 (2011/8/24)"; //退出函數 inline void usage(int exit_value = 0){ cerr<<prog_name<<" usage!"<<endl; exit(exit_value); } int main(int argc,char* argv[]){ //聲明用於記錄用戶指定選項的變量 bool debug_on = false; bool ofile_on = false; bool limit_on = false; string ofile_name;//記錄出現的輸出文件名 int limit = -1;//限制值 vector <string> file_names;//記錄文件名 cout<<"argc:"<<argc<<endl; for(int i = 1;i < argc; ++i){//讀取argv中的每個選項 //輸出第i+1個參量 cout<<"argv["<<i<<"]:"<<argv[i]<<endl; char *pchar = argv[i]; switch(pchar[0]){//確定選項類型:-h,-d,-v,-l,-o;或者其他 case '-':{ cout<<"case \'-\' found"<<endl; switch(pchar[1]){//確定用戶指定的選項:h,d,v,l,o case 'd'://處理調試: cout<<"-d found:debugging turned on!"<<endl; debug_on = true; break; case 'v'://處理版本請求 cout<<"-v found:version info displayed!"<<endl; cout<<prog_name<<":"<<prog_version<<endl; return 0; case 'h'://處理幫助 cout<<"-h found:help info!"<<endl; usage(); case 'o'://處理輸出文件 cout<<"-o found:output file!"<<endl; ofile_on = true; break; case 'l'://處理限制量 cout<<"-l found:resorce limit!"<<endl; limit_on = true; break; default://無法識別的選項 cerr<<prog_name<<":error:unrecognition option -:"<<pchar<<endl; usage(-1); } break; } default://不以'-'開頭,是文件名 if(ofile_on){//輸出文件名 cout<<"file name:"<<pchar<<endl; ofile_name = pchar; ofile_on = false;//復位 } else if(limit_on){//限制值 limit_on = false; limit = atoi(pchar); if(limit<0){ cerr<<prog_name<<":error:negative value for limit!"<<endl; usage(-2); } } else{//文件名 file_names.push_back(pchar); } break; } } if(file_names.empty()){ cerr<<prog_name<<":error:no file for processing!"<<endl; usage(3); } else{ cout<<(file_names.size() == 1 ? "File":"Files")<< " to be processed are the followed:"<<endl; for(int i = 0;i < file_names.size();++i){ cout<<file_names[i]<<"\t"<<endl; } } if(limit != -1){ cout<<"user-specified limit:"<<limit<<endl; } if(!ofile_name.empty()){ cout<<"user-specified ofile:"<<ofile_name<<endl; } }