电话本------改进值分离命令和参数

通过cmd[3]=0、strcpy(file, cmd + 4)实现命令的解析,比较诡异。
必须要知道命令头只有三个字节的秘密
若将来命令格式发生变化,麻烦大了
只有在while循环中改动,除了要理解命令解析外,还要理解其他很多语义
另外,其他命令解析,也需要进行strcpy,重复就是罪恶!
需要把命令解析的功能提取处理
另一个问题:要输入“-ld c:\test”后,再输入“-ld”,可以发现“c:\test”依然存在
单步调试,观察cmd在内存中的变化
在scanf之前,对cmd进行清0操作

// 2.17.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <windows.h>
#include <string.h>

const unsigned int MAX_LENGTH_OF_CMD = 300;
const unsigned int MAX_LENGTH_OF_CMD_HEAD = 10;

const char* DEFAULT_FILE_NAME = "temp.txt";

/*
strDefaultFileFullPath要确保足够的存储空间,建议为MAX_PATH
*/
bool GetDefaultFileFullPath(char *strDefaultFileFullPath, const char *strDefaultFileName, const char *strArgv0)
{
	int i = strlen(strArgv0) - 1;
	while(i >= 0)
	{
		if(strArgv0[i] == '\\')
			break;

		i--;
	}

	if(i < 0)
	{
		strcpy(strDefaultFileFullPath, strDefaultFileName);
	}
	else
	{
		strcpy(strDefaultFileFullPath, strArgv0);
		strcpy(strDefaultFileFullPath + i + 1, strDefaultFileName);
	}

	return true;
}

void DisplayDefaultFile(const char *strDefaultFileFullPath)
{
	FILE *fp = fopen(strDefaultFileFullPath, "r+");
	if(fp != NULL)
	{
		while(1)
		{
			char buf[MAX_PATH];
			if(fgets(buf, MAX_PATH, fp) == NULL)
				break;

			printf("%s", buf);
		}

		fclose(fp);
	}
	else
	{
		printf("Default file does not exist.\n");
	}
}

/*
确保strCommandHead和strParameter有足够的空间,建议分别设为MAX_LENGTH_OF_CMD_HEAD、MAX_LENGTH_OF_CMD
*/
bool InterpretCommand(const char* strCommand, char *strCommandHead, char *strParameter)
{
	if(strlen(strCommand) < 3)
		return false;

	char cmd[MAX_LENGTH_OF_CMD];
	memset(cmd, 0, MAX_LENGTH_OF_CMD);
	strcpy(cmd, strCommand);

	cmd[3] = 0;
	
	strcpy(strCommandHead, cmd);

	if(strlen(strCommand) > 3)
		strcpy(strParameter, cmd + 4);

	return true;
}

void EnterCommandLoop()
{
	char cmd[MAX_LENGTH_OF_CMD];

	while(1)
	{
		printf("Please input your command : ");

		memset(cmd, 0, MAX_LENGTH_OF_CMD);
		if(scanf("%[^\n]", cmd) == 0)
		{
			fflush(stdin);
			continue;
		}

		fflush(stdin);

		char cmd_head[MAX_LENGTH_OF_CMD_HEAD];
		char cmd_parameter[MAX_LENGTH_OF_CMD];
		memset(cmd_head, 0, MAX_LENGTH_OF_CMD_HEAD);
		memset(cmd_parameter, 0, MAX_LENGTH_OF_CMD);
		if(!InterpretCommand(cmd, cmd_head, cmd_parameter))
		{
			printf("Command false.\n");
			continue;
		}
		
		printf("Your command is : %s\n", cmd);

		if(strcmp(cmd_head, "-ld") == 0)
		{
			printf("Needed load file is %s\n", cmd_parameter);
			continue;
		}

		if(strcmp(cmd_head, "-sa") == 0)
		{
			printf("will export phone book to %s\n", cmd_parameter);
			continue;
		}

		if(strcmp(cmd_head, "-se") == 0)
		{
			printf("will save phone book\n");
			continue;
		}

		if(strcmp(cmd_head, "-ad") == 0)
		{
			printf("will add item(name phone): %s\n", cmd_parameter);
			continue;
		}

		if(strcmp(cmd_head, "-qt") == 0)
		{
			printf("quit now ....\n");
			break;
		}

		printf("False Command\n");
	}
}

int _tmain(int argc, _TCHAR* argv[])
{
	char DefaultFileFullPath[MAX_PATH];
	GetDefaultFileFullPath(DefaultFileFullPath, DEFAULT_FILE_NAME, argv[0]);

	printf("Default file path: %s\n", DefaultFileFullPath);

	DisplayDefaultFile(DefaultFileFullPath);

	EnterCommandLoop();

	return 0;
}


 我们做的改进有消除了每个if   else 里面的一句重复代码  ,置0后解决了-ld  问题  函数封装之后接解决了  如果命令的格式发生变化,只是需要改变函数里面的数字 3 就可以了

 

 

 

你可能感兴趣的:(C++,C语言)