C++面向?qū)ο笳Z(yǔ)言自制多級(jí)菜單功能實(shí)現(xiàn)代碼
因?yàn)橐鲆粋€(gè)小應(yīng)用,需要一個(gè)菜單類(lèi),在網(wǎng)上找了許久,也沒(méi)有找到一款心儀的菜單類(lèi),索性用C++語(yǔ)言,自制一個(gè)命令行級(jí)別的菜單類(lèi),并制作成庫(kù),現(xiàn)記錄下來(lái),供以后借鑒。
一、特性
- 無(wú)限制條目
- 無(wú)限制層級(jí)
- 用戶自定義條目和動(dòng)作
- 腳本式生成菜單類(lèi)
二、代碼實(shí)現(xiàn)
(一)菜單類(lèi)
菜單類(lèi)主要負(fù)責(zé)菜單的創(chuàng)建、修改、刪除,是包含菜單結(jié)構(gòu)組織和響應(yīng)函數(shù)的模型,用戶擁有充分的自主性,可根據(jù)需要自定義菜單顯示和響應(yīng)函數(shù)。其中菜單項(xiàng)使用vector容器數(shù)據(jù)結(jié)構(gòu),根據(jù)用戶命令可進(jìn)行菜單創(chuàng)建、修改、刪除,而顯示和響應(yīng)函數(shù)則利用函數(shù)指針進(jìn)行保存,體現(xiàn)了回調(diào)函數(shù)的特性。
/*
菜單類(lèi)
(c) hele 2024
菜單類(lèi)主要負(fù)責(zé)菜單的創(chuàng)建、修改、刪除,是包含菜單結(jié)構(gòu)組織和響應(yīng)函數(shù)的模型,用戶擁有充分的自主性,需要自定義菜單顯示和響應(yīng)函數(shù)
*/
class HeleMenu {
private:
void (*p_action)()=nullptr; //回調(diào)函數(shù)指針,菜單響應(yīng)函數(shù)
void (*p_display)(string &name, vector<string> &content, unsigned int &index)=nullptr;//回調(diào)函數(shù)指針,菜單顯示函數(shù)
public:
string name; //當(dāng)前菜單名稱(chēng)
HeleMenu *p_father; //父節(jié)點(diǎn)指針,默認(rèn)NULL
vector<string> p_childrenName; //下級(jí)菜單項(xiàng)名稱(chēng),默認(rèn)empty無(wú)下級(jí)菜單
vector<HeleMenu *> p_children; //下級(jí)菜單項(xiàng),默認(rèn)empty無(wú)下級(jí)菜單
unsigned int index; //菜單項(xiàng)選擇指示位,默認(rèn)0
static HeleMenu * parseMenu(string bat, void (*p_display[])(string&, vector<string>&, unsigned int&), void (*p_action[])()); //解析腳本生成菜單
HeleMenu(string name = "", HeleMenu *p_father = nullptr); //帶菜單名稱(chēng)、父節(jié)點(diǎn)指針的構(gòu)造函數(shù),默認(rèn)菜單名為無(wú)名,默認(rèn)父節(jié)點(diǎn)值為空
void addToMenus(); //添加菜單項(xiàng)到菜單本
void addValue(string value); //添加菜單單個(gè)葉節(jié)點(diǎn)
int addValues(vector<string> values); //添加菜單葉節(jié)點(diǎn)
void changeValue(string value); //修改菜單葉節(jié)點(diǎn)
HeleMenu *getChild(unsigned int index = 0); //獲取指定序號(hào)的子節(jié)點(diǎn),默認(rèn)第0項(xiàng)子節(jié)點(diǎn)
string getValue(); //獲取菜單葉節(jié)點(diǎn)
void removeAllItem(); //刪除菜單所有節(jié)點(diǎn)
void attachAction(void (*p_action)()); //添加動(dòng)作回調(diào)函數(shù)
void attachDisplay(void (*p_display)(string&, vector<string>&, unsigned int&)); //添加顯示回調(diào)函數(shù) 0.菜單名,1.顯示內(nèi)容,2.選中項(xiàng)
void action(); //菜單響應(yīng)函數(shù)
void display(); //菜單顯示函數(shù)
};/*解析腳本生成菜單*/
HeleMenu * HeleMenu::parseMenu(string bat, void (*p_display[])(string&, vector<string>&, unsigned int&), void (*p_action[])()) {
vector<char> stack_simul; //符號(hào)棧
vector<HeleMenu*> stack_p; //指針棧
vector<string> stack_name; //名稱(chēng)棧
HeleMenu * root = nullptr; //初始化根菜單指針
uint8_t countAction = 0;
for (uint8_t i = 0; i < bat.length();) {
string name = ""; //菜單名
uint8_t step = 0; //菜單名長(zhǎng),即距下次讀取的步長(zhǎng)
char a = bat[i];
if ('{' == a) { //有子菜單
HeleMenu *father = nullptr;
if (stack_p.size() > 0) { //有父菜單
father = stack_p.back(); //彈出指針棧
}
if (stack_name.size() > 0) {
name = stack_name.back(); //讀取名稱(chēng)
}
HeleMenu * m1 = new HeleMenu(name, father); //構(gòu)建菜單
m1->attachDisplay(p_display[countAction]);
m1->attachAction(p_action[countAction++]);
m1->addToMenus();
stack_simul.push_back('{'); //壓入符號(hào)棧
stack_p.push_back(m1); //壓入菜單指針
i++;
} else if ('}' == a) { //子菜單結(jié)束
stack_simul.pop_back(); //彈出符號(hào)棧
root = stack_p.back(); //彈出菜單指針棧
stack_p.pop_back();
stack_name.pop_back(); //彈出名稱(chēng)棧
i++;
} else if (',' == a) {
i++;
} else { //若有菜單名或值
for (uint8_t j = i; j < bat.length() && '{' != bat[j] && '}' != bat[j] && ',' != bat[j]; j++,step++) { //讀菜單名或值
name.append(&bat[0]+j,1); //適應(yīng)中文
}
stack_name.push_back(name);
i += step;
if (stack_simul.size() > 0 && '{' == stack_simul.back() && ('}' == bat[i] || ',' == bat[i])) { //若為值
stack_p.back()->addValue(name);
}
}
}
return root;
}
/*帶菜單名稱(chēng)、父節(jié)點(diǎn)指針的構(gòu)造函數(shù),默認(rèn)菜單名為無(wú)名,默認(rèn)父節(jié)點(diǎn)值為空*/
HeleMenu::HeleMenu(string name, HeleMenu *p_father) {
this->name = name;
this->p_father = p_father;
this->index = 0;
this->p_action = nullptr;
this->p_display = nullptr;
}
/*添加菜單項(xiàng)到菜單本*/
void HeleMenu::addToMenus() {
if (this->p_father != nullptr) {
this->p_father->p_childrenName.push_back(this->name); //賦予菜單內(nèi)容
this->p_father->p_children.push_back(this); //將菜單指針注冊(cè)到父菜單
}
}
/*添加菜單單個(gè)葉節(jié)點(diǎn)*/
void HeleMenu::addValue(string value) {
this->p_childrenName.push_back(value);
this->p_children.push_back(nullptr); //添加空指針,表示無(wú)下級(jí)菜單,即葉節(jié)點(diǎn)
}
/*添加菜單葉節(jié)點(diǎn)*/
int HeleMenu::addValues(vector<string> values) {
unsigned int i = 0;
for (; i < values.size(); i++) {
this->p_childrenName.push_back(values[i]); //賦予值項(xiàng)
this->p_children.push_back(nullptr); //添加空指針,表示無(wú)下級(jí)菜單,即葉節(jié)點(diǎn)
};
return i;
}
/*添加動(dòng)作回調(diào)函數(shù)*/
void HeleMenu::attachAction(void (*p_action)()) {
this->p_action = p_action;
}
/*添加顯示回調(diào)函數(shù)*/
void HeleMenu::attachDisplay(void (*p_display)(string&, vector<string>&, unsigned int&)) {
this->p_display = p_display;
}
/*菜單響應(yīng)函數(shù)*/
void HeleMenu::action() {
this->p_action(); //回調(diào)動(dòng)作函數(shù)
}
/*將某項(xiàng)葉節(jié)點(diǎn)改成對(duì)應(yīng)值,實(shí)現(xiàn)菜單動(dòng)態(tài)顯示*/
void HeleMenu::changeValue(string value) {
this->p_childrenName.at(this->index) = value;
}
/*菜單顯示函數(shù)*/
void HeleMenu::display() {
this->p_display(this->name, this->p_childrenName, this->index); //回調(diào)顯示函數(shù),1.顯示內(nèi)容,2.選中項(xiàng)
}
/*獲取指定序號(hào)的子節(jié)點(diǎn)*/
HeleMenu * HeleMenu::getChild(unsigned int index) {
return this->p_children.at(index);
}
/*獲取某項(xiàng)葉節(jié)點(diǎn)值,實(shí)現(xiàn)菜單動(dòng)態(tài)顯示*/
string HeleMenu::getValue() {
return this->p_childrenName.at(this->index);
}
/*刪除菜單所有節(jié)點(diǎn)*/
void HeleMenu::removeAllItem() {
for(auto i:this->p_children) { //釋放子節(jié)點(diǎn)內(nèi)存
free(i);
}
this->p_childrenName.clear();
this->p_children.clear();
this->index = 0; //序號(hào)防越界,恢復(fù)默認(rèn)值
}(二)菜單瀏覽器類(lèi)
菜單瀏覽器類(lèi)主要負(fù)責(zé)菜單結(jié)構(gòu)的瀏覽導(dǎo)航。私有變量是2個(gè)菜單類(lèi)指針,1個(gè)是根目錄指針,1個(gè)是當(dāng)前目錄指針。
/*
菜單瀏覽器類(lèi)
(c)hele 2024
菜單瀏覽器主要負(fù)責(zé)菜單結(jié)構(gòu)的瀏覽
*/
class HeleMenuViewer {
private:
static HeleMenu *p_rootMenu, *p_currentMenu; //內(nèi)置根菜單指針、當(dāng)前菜單項(xiàng)指針
public:
static void init(HeleMenu *p_rootMenu); //初始化函數(shù)
static HeleMenu * getCurrentMenu(); //獲取當(dāng)前菜單指針
static HeleMenu * getRootMenu(); //獲取根菜單指針
static void gotoFather(); //跳轉(zhuǎn)到父菜單
static void gotoChild(); //跳轉(zhuǎn)到子菜單,序號(hào)指示在菜單類(lèi)里
static void gotoChild(unsigned int index, unsigned int selected=0); //跳轉(zhuǎn)到指定序號(hào)的菜單,默認(rèn)其選中子菜單第0項(xiàng)
static void gotoRoot(); //跳轉(zhuǎn)到根菜單
static void selectUp(); //向上選擇
static void selectDown(); //向下選擇
static void action(); //當(dāng)前菜單響應(yīng)
static void display(); //顯示當(dāng)前菜單
};/*初始化菜單管理類(lèi)的內(nèi)置根菜單和當(dāng)前菜單指針為空指針*/
HeleMenu *HeleMenuViewer::p_rootMenu = nullptr;
HeleMenu *HeleMenuViewer::p_currentMenu = nullptr;
/*當(dāng)前菜單響應(yīng)*/
void HeleMenuViewer::action() {
p_currentMenu->action();
}
/*顯示當(dāng)前菜單*/
void HeleMenuViewer::display() {
p_currentMenu->display();
}
/*獲取當(dāng)前菜單指針*/
HeleMenu * HeleMenuViewer::getCurrentMenu() {
return p_currentMenu;
}
/*獲取根菜單指針*/
HeleMenu * HeleMenuViewer::getRootMenu() {
return p_rootMenu;
}
/*跳轉(zhuǎn)到父菜單*/
void HeleMenuViewer::gotoFather() {
if (p_currentMenu->p_father != nullptr) {
p_currentMenu = p_currentMenu->p_father;
}
}
/*跳轉(zhuǎn)到子菜單,序號(hào)指示在菜單類(lèi)里*/
void HeleMenuViewer::gotoChild() {
if (p_currentMenu->p_children.size() > 0 && p_currentMenu->p_children[p_currentMenu->index] != nullptr) { 防止無(wú)子項(xiàng)
p_currentMenu = p_currentMenu->p_children[p_currentMenu->index];
}
}
/*跳轉(zhuǎn)到指定序號(hào)的菜單,默認(rèn)其選中子菜單第0項(xiàng)*/
void HeleMenuViewer::gotoChild(unsigned int index, unsigned int selected) {
if (index < p_currentMenu->p_children.size()) { //防止越界
p_currentMenu->index = index; //更新菜單指示位置
gotoChild();
if (selected < p_currentMenu->p_children.size()) {
p_currentMenu->index = selected;
}
}
}
/*跳轉(zhuǎn)到根菜單*/
void HeleMenuViewer::gotoRoot() {
p_currentMenu = p_rootMenu;
}
/*初始化函數(shù),為根菜單指針賦值*/
void HeleMenuViewer::init(HeleMenu *p_rootMenu) {
HeleMenuViewer::p_rootMenu = p_rootMenu;
}
/*向下選擇*/
void HeleMenuViewer::selectDown() {
if (p_currentMenu->p_childrenName.size() > 0) { //防除0錯(cuò)誤
p_currentMenu->index = (p_currentMenu->index + 1) % p_currentMenu->p_childrenName.size();
}
}
/*向上選擇*/
void HeleMenuViewer::selectUp() {
if (p_currentMenu->p_childrenName.size() > 0) { //防除0錯(cuò)誤
p_currentMenu->index = (p_currentMenu->index - 1 + p_currentMenu->p_childrenName.size()) % p_currentMenu->p_childrenName.size();
}
}(三)庫(kù)的生成
網(wǎng)上的資料有很多了,在此僅簡(jiǎn)單記錄。
##靜態(tài)庫(kù)生成與調(diào)用,用HeleMenu.cpp生成庫(kù)## #編譯生成.o鏈接文件 g++ -c HeleMenu.cpp #利用.o文件生成靜態(tài)庫(kù),文件名必須以lib***.a形式命名 ar -crv libHeleMenu.a HeleMenu.o #調(diào)用靜態(tài)庫(kù)生成目標(biāo)程序 g++ -o main.exe CreateAndShowMenu.cpp -L . -lHeleMenu g++ CreateAndShowMenu.cpp libHeleMenu.a -o main.exe ------------------------------------------------- ##動(dòng)態(tài)庫(kù)生成與調(diào)用,用HeleMenu.cpp生成庫(kù)## #生成.o鏈接文件 g++ -c HeleMenu.cpp #利用.o文件生成動(dòng)態(tài)庫(kù),文件名必須以lib***.so形式命名 g++ -shared -fpic -o libHeleMenu.so HeleMenu.o #調(diào)用動(dòng)態(tài)庫(kù)生成目標(biāo)程序 g++ -o main.exe CreateAndShowMenu.cpp -L . -lHeleMenu g++ CreateAndShowMenu.cpp libHeleMenu.so -o main.exe ------------------------------------------------- 代碼編譯優(yōu)化 -O0 禁止編譯器優(yōu)化,默認(rèn)此項(xiàng) -O1 嘗試優(yōu)化編譯時(shí)間和可執(zhí)行文件大小 -O2 更多的優(yōu)化,嘗試幾乎全部的優(yōu)化功能,但不會(huì)進(jìn)行“空間換時(shí)間”的優(yōu)化方法 -O3 在-O2基礎(chǔ)上再打開(kāi)一些優(yōu)化選項(xiàng) -Os 對(duì)生成文件大小進(jìn)行優(yōu)化。會(huì)打開(kāi)-O2開(kāi)的全部選項(xiàng),除了那些會(huì)增加文件大小的
三、使用示例
主要有2種方法實(shí)現(xiàn)用戶自定義的菜單類(lèi),共同點(diǎn)是attachDisplay和attachAction所帶的參數(shù)均為用戶自定義的函數(shù)。
(一)手動(dòng)生成
/*手動(dòng)生成菜單*/
HeleMenu *m1 = new HeleMenu();
m1->attachDisplay(display_root);
m1->attachAction(action_root);
HeleMenuViewer::init(m1); //初始化根菜單
//
HeleMenu *m2 = new HeleMenu("歷史記錄",m1);
m2->attachDisplay(display);
m2->attachAction(action);
m2->addToMenus();
m2 = new HeleMenu("操作",m1);
m2->addValues({"保存","不保存"});
m2->attachDisplay(display);
m2->attachAction(action_operate);
m2->addToMenus();
m2 = new HeleMenu("菜單",m1);
m2->attachDisplay(display);
m2->attachAction(action);
m2->addToMenus();
m1 = m2; //構(gòu)建下一層子菜單
m2 = new HeleMenu("對(duì)比度", m1);
m2->addValues({"1", "2", "3", "4"});
m2->attachDisplay(display_compare);
m2->attachAction(action_compare);
m2->addToMenus();
m1->addValues({/*對(duì)比度,*/ "全部清除","重啟","關(guān)機(jī)"/*,"校準(zhǔn)","關(guān)于"*/});
m2 = new HeleMenu("校準(zhǔn)",m1);
m2->addValue("確認(rèn)");
m2->attachDisplay(display);
m2->attachAction(action_adjust);
m2->addToMenus();
m2 = new HeleMenu("關(guān)于",m1);
m2->addValue("(c)hele 2024\n這是一個(gè)菜單類(lèi),可以幫助你生成自定義菜單,同時(shí)還可以設(shè)置動(dòng)作");
m2->attachDisplay(display);
m2->attachAction(action);
m2->addToMenus();
HeleMenuViewer::gotoRoot(); //到達(dá)根菜單
while (true) { //啟動(dòng)
system("cls");
HeleMenuViewer::display();
HeleMenuViewer::action();
}(二)腳本生成
主要利用菜單類(lèi)parseMenu函數(shù)實(shí)現(xiàn),寫(xiě)了1個(gè)解析器,可以實(shí)現(xiàn)菜單類(lèi)的自動(dòng)生成。
/*腳本生成菜單*/
void (*p_display[])(string&, vector<string>&, unsigned int&) = {/*root*/display_root, /*log*/display, /*operate*/display, /*menu*/display, /*constrast*/display_compare, /*adjust*/display, /*about*/display};
void (*p_action[])() = {/*root*/action_root, /*log*/action, /*operate*/action_operate, /*menu*/action, /*constrast*/action_compare, /*adjust*/action_adjust, /*about*/action};
HeleMenu *m1 = HeleMenu::parseMenu("{歷史{},操作{save,unsave},菜單{對(duì)比度{1,2,3,4},clearAll,rePower,shutdown,校準(zhǔn){confirm},關(guān)于{(c)hele 2024,這是一個(gè)菜單}}}", p_display, p_action);
HeleMenuViewer::init(m1);
HeleMenuViewer::gotoRoot(); //到達(dá)根菜單
while (true) { //啟動(dòng)
system("cls");
HeleMenuViewer::display();
HeleMenuViewer::action();
}parseMenu所帶的參數(shù)共3個(gè),第1個(gè)是菜單結(jié)構(gòu)字符串,也就是生成菜單結(jié)構(gòu)的腳本,后面2個(gè)參數(shù)分別是顯示函數(shù)指針數(shù)組和響應(yīng)函數(shù)指針數(shù)組。為便于理解,下面我將用戶自定義菜單結(jié)構(gòu)展開(kāi):
{
log{
},
operate{
save,unsave
},
menu{
constrast{
1,2,3,4
},
clearAll,
rePower,
shutdown,
adjust{
confirm
},
about{
(c)hele 2024
}
}
}每一個(gè)'{'都意味著該項(xiàng)目有子菜單,每一個(gè)'}'意味著該菜單結(jié)束,每一個(gè)','都意味著有同級(jí)菜單,以上這3個(gè)符號(hào)均是關(guān)鍵詞,均是英文字符。所有菜單除內(nèi)容中間可以有空格外,其余地方不能有多余的空格。支持中文。
(三)演示
主要使用上、下、左、右、空格、回車(chē)、退出這些按鍵,只實(shí)現(xiàn)部分功能。

四、不足與展望
運(yùn)用C++面向?qū)ο笏季S進(jìn)行編程,代碼體積大,效率低;
利用腳本生成菜單是個(gè)新穎的思路,但容錯(cuò)性不好,沒(méi)有對(duì)腳本進(jìn)行規(guī)范化的檢查,對(duì)用戶不友好;
可以利用ArduinoSTL庫(kù),將此庫(kù)移植進(jìn)Arduino系列單片機(jī)項(xiàng)目中;
使用了動(dòng)態(tài)分配內(nèi)存技術(shù),對(duì)于RAM較小的單片機(jī),容易內(nèi)存溢出。
五、參考資料
- GCC編譯步驟、動(dòng)態(tài)庫(kù)和靜態(tài)庫(kù)的創(chuàng)建和使用、ELF庫(kù)簡(jiǎn)介及查看方法
- C語(yǔ)言(函數(shù)指針數(shù)組)詳解
- 字符編碼詳解及利用C++ STL string遍歷中文字符串
- 自制的Arduino多級(jí)菜單類(lèi)
- C++庫(kù)文件string,vector
- https://www.aigei.com/item/c_mian_xiang.html
- https://download.csdn.net/download/hele_two/89414475?spm=1001.2014.3001.5503
到此這篇關(guān)于C++面向?qū)ο笳Z(yǔ)言自制多級(jí)菜單的文章就介紹到這了,更多相關(guān)C++自制多級(jí)菜單內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件
這篇文章主要為大家介紹一個(gè)C++?OpenCV的實(shí)戰(zhàn)——文檔照片轉(zhuǎn)換成掃描文件,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-09-09
C++動(dòng)態(tài)調(diào)用動(dòng)態(tài)鏈接庫(kù)(DLL或SO)的方法實(shí)現(xiàn)
動(dòng)態(tài)鏈接庫(kù)是一種Windows操作系統(tǒng)下常見(jiàn)的可執(zhí)行文件格式,它包含了一些可被其他應(yīng)用程序調(diào)用的函數(shù)和數(shù)據(jù),本文主要介紹了C++動(dòng)態(tài)調(diào)用動(dòng)態(tài)鏈接庫(kù)(DLL或SO),感興趣的可以了解一下2024-01-01
C語(yǔ)言詳解strcmp函數(shù)的分析及實(shí)現(xiàn)
strcmp函數(shù)語(yǔ)法為“int strcmp(char *str1,char *str2)”,其作用是比較字符串str1和str2是否相同,如果相同則返回0,如果不同,前者大于后者則返回1,否則返回-12022-05-05

