武侠世界2-try catch思考

以前一直不知道trycatch具体应用到什么地方,之前待过的几家公司也看不到有类似的代码。从网上搜来的,描述trycatch优点有下面几点。1、把错误处理和真正的工作分开来;  2、代码更易组织,更清晰,复杂的工作任务更容易实现;  3、毫无疑问,更安全了,不至于由于一些小的疏忽而使程序意外崩溃了;  4、由于C++中的trycatch可以分层嵌套,所以它…

大家好,又见面了,我是你们的朋友全栈君。

以前一直不知道try catch具体应用到什么地方,之前待过的几家公司也看不到有类似的代码。
从网上搜来的,描述try catch优点有下面几点。
1、 把错误处理和真正的工作分开来;
  2、 代码更易组织,更清晰,复杂的工作任务更容易实现;
  3、 毫无疑问,更安全了,不至于由于一些小的疏忽而使程序意外崩溃了;
  4、 由于C++中的try catch可以分层嵌套,所以它提供了一种方法使得程序的控制流可以安全的跳转到上层(或者上上层)的错误处理模块中去。(不同于return语句,异常处理的控制流是可以安

全地跨越一个或多个函数 )。
  5、 还有一个重要的原因就是,由于目前需要开发的软件产品总是变得越来越复杂、越来越庞大,如果系统中没有一个可靠的异常处理模型,那必定是一件十分糟糕的局面。

//总结出来就是:减少了判断语句的使用(if),程序更加简洁明了,程序更加健壮。
直到在武侠世界源码里面看到下面的一些代码:
在World的main函数里

__ENTER_FUNCTION
__LEAVE_FUNCTION
这两个宏的定义是
#define __ENTER_FUNCTION {try{

#define __LEAVE_FUNCTION }catch(…){AssertSpecial(FALSE,__FUNCTION__);}}
也就是说武侠世界里的大多数函数的开头和结尾都会用到这两个函数,至于在代码在哪里抛出错误呢?
下面是武侠世界World的主循环函数

void ServerManager::Loop( )
{
__ENTER_FUNCTION

	while( IsActive() )
	{
		BOOL ret = FALSE ;
		UINT uTime = g_pTimeManager->CurrentTime() ;

		_MY_TRY
		{
			ret = Select( ) ;
			Assert( ret ) ;

			ret = ProcessExceptions( ) ;
			Assert( ret ) ;

			ret = ProcessInputs( ) ;
			Assert( ret ) ;

			ret = ProcessOutputs( ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = ProcessCommands( ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = HeartBeat( ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pOnlineUser->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

        _MY_TRY
        {
            ret = g_pCountryManager->HeartBeat( uTime ) ;
            Assert( ret ) ;
        }
        _MY_CATCH
        {
        }

		_MY_TRY
		{
			ret = g_pTeamList->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pChatCenter->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pGuildManager->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pCityManager->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pMailCenter->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pSceneInfo->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}

		_MY_TRY
		{
			ret = g_pWorldTimeInfo->HeartBeat( uTime ) ;
			Assert( ret ) ;
		}
		_MY_CATCH
		{
		}
	};

__LEAVE_FUNCTION
}

其实这写得相当明了,Select( ) 就是接收网络信息,HeartBeat( )就是心跳等等。
里面的Assert是自己定义的一个宏

#define Assert(expr) ((VOID)((expr)?0:(__assert__(__FILE__,__LINE__,__FUNCTION__,#expr),0)))
#define AssertEx(expr,msg) ((VOID)((expr)?0:(__assertex__(__FILE__,__LINE__,__FUNCTION__,#expr,msg),0)))
#define AssertSpecial(expr,msg) ((VOID)((expr)?0:(__assertspecial__(__FILE__,__LINE__,__FUNCTION__,#expr,msg),0)))

VOID __assert__ ( const CHAR * file , UINT line , const CHAR * func , const CHAR * expr )
{
	CHAR szTemp[1024] = {0};
	
#ifdef __LINUX__ //换个格式
	sprintf( szTemp, "[%s][%d][%s][%s]\n", file, line, func, expr ) ;
#else
	sprintf( szTemp, "[%s][%d][%s][%s]", file, line, func, expr ) ;
#endif
	__show__(szTemp) ;
}

VOID __show__( const CHAR* szTemp )
{

#ifdef __LINUX__
	printf("Assert:%s",szTemp);
#endif

	//保存日志
#ifndef GAME_CLIENT
	FILE* f = fopen( "./Log/assert.log", "a" ) ;
	if (f != NULL)
	{
		fwrite( szTemp, 1, strlen(szTemp), f ) ;
		fwrite( "\r\n", 1, 2, f ) ;
		fclose(f) ;
	}
#endif

#if defined(__WINDOWS__)
	static MyLock lock ;
	if( g_Command_Assert!=1 )
	{
		lock.Lock() ;
		INT iRet = ::MessageBoxA( NULL, szTemp, "异常", MB_OK ) ;
		lock.Unlock() ;
	}
#elif defined(__LINUX__)
	
#endif

#ifdef GAME_CLIENT
	throw(std::string(szTemp));
#else
	throw(1);
#endif
}

最终会在__show__中throw错误,做成了一个统一的错误处理接口。
这是相当于用
Assert( ret ) ;

代替了
if ( !ret )
{

printf(“error\n”);
return;
}
这样使程序更清晰明了。
至于为什么还要定义
#define _MY_TRY try
#define _MY_CATCH catch(…)
大家可以思考一下

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/125686.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • source insight 序列号_gxworks3安装序列号

    source insight 序列号_gxworks3安装序列号2019独角兽企业重金招聘Python工程师标准>>>…

    2022年9月28日
    0
  • ubuntu16.04 安装 Eric6「建议收藏」

    从安装qt,到安装qtpy,到安装Eric6,这是一个很痛苦的过程。总是会有一大段的错误,然后在网上各种搜索,然后去改,然后还是有新的错误,又去找答案,一直重复,我都快崩溃了。最后,终于,找到这一篇博客:http://blog.csdn.net/suxiang198/article/details/52042526。这篇博客解决了大部分坑,不过到后面部分还是出现了问题,安装不上去。最后,终于在E

    2022年4月8日
    47
  • Codelf插件的使用

    Codelf插件的使用当你找到这个插件的时候说明你已经知道了这个插件的作用,这里只说关于变量名和star的使用。1、去应用商店下载安装插件2、直接使用http://unbug.github.io/codelf/变量名:输入一个中文名字,会给出一大堆建议的英文变量名(比如说”产品”,突然英文名字忘了,就可以如下操作)你可能会说,我直接百度翻译、有道翻译不就OK了吗?但是”产品”只是个常见的单词,”产品资源”、”气质

    2022年6月4日
    99
  • action context_session.getAttribute

    action context_session.getAttributeformURL:http://apps.hi.baidu.com/share/detail/16057446 ActionContext(Action上下文)ActionContext介绍通过上面用户注册例子的学习,我们知道Xwork与Web无关性,我们的Action不用去依赖于任何Web容器,不用和那些JavaServlet复杂的请求(Request)、响应(Respon

    2022年9月9日
    0
  • Vue电商后台管理系统(1)

    Vue电商后台管理系统(1)Vue电商后台管理系统(1)登录在components文件夹下创建登录组件,Login.vue,并快速生成template、script和style骨架。配置路由,进入router文件夹,导入Login组件,创建路由并重定向首页为登录界面,进入首页时会自动跳转至登录页面,配置如下:绘制页面:<template><divclass=”login_container”><divclass=”login_box”><!–

    2022年5月9日
    79
  • git 自己搭建服务器_github搭建服务器

    git 自己搭建服务器_github搭建服务器1.环境:git服务器:CentOS-7.6-x86 本地客户端:window102.软件安装yuminstallgit-y3.创建一个操作系统用户git,设置密码,并为其建立一个.ssh目录。addusergitpasswdgitchmod755/home/gitsugitcdmkdir.ssh&&chmod700.sshtouch.ssh/authorized_keys&&chmod600…

    2022年10月5日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号