调试常用的 __FILE__, __FUNCTION__, __LINE__


调试常用的 __FILE__, __FUNCTION__, __LINE__

没想到 VC6 不支持 __FUNCTION__

所以我写了如下的奇怪代码


//用来记录当前行和当前函数//也可说是记录 堆栈
void log_stack(const char *file, int line, const char * function);


//当然还要对 __FUNCTION__ 宏作点修饰,因为这个宏只是在函数里面才起作用
//据说 VC6 也是不支持 __FUNCTION__ 的
#ifndef __FUNCTION__
    #define __FUNCTION__ "Global"
#endif

#define DEBUG_NEW_HOOK

#ifdef DEBUG_NEW_HOOK
  //就是先写跟踪信息再实际调用函数
  #define  debug_new_check_point(a)  log_stack(__FILE__, __LINE__, __FUNCTION__); debug_new_check(a, false)
  #define  debug_new_check_free(a)  log_stack(__FILE__, __LINE__, __FUNCTION__); debug_new_check(a, true)
  //#define  printfd2(a,b)  printf(a,b)
  //#define  printfd3(a,b,c)  printf(a,b,c)
  //#define  printfd4(a,b,c,d)  printf(a,b,c,d)
#else

#endif


背景知识:

--------------------------------------------------
在C语言中以编程的方式获取函数名

http://www.sina.com.cn 2006年07月26日 09:03 天极yesky


  作者:谢启东编译

  仅仅为了获取函数名,就在函数体中嵌入硬编码的字符串,这种方法单调乏味还易导致错误,不如看一下怎样使用新的C99特性,在程序运行时获取函数名吧。 

  对象反射库、调试工具及代码分析器,经常会需要在运行时访问函数的名称,直到不久前,唯一能完成此项任务并且可移植的方法,是手工在函数体内嵌入一个带有该函数名的硬编码字符串,不必说,这种方法非常单调无奇,并且容易导致错误。本文将要演示怎样使用新的C99特性,在运行时获取函数名。

  那么怎样以编程的方式从当前运行的函数中得到函数名呢?

  答案是:使用__FUNCTION__ 及相关宏。

  引出问题

  通常,在调试中最让人心烦的阶段,是不断地检查是否已调用了特定的函数。对此问题的解决方法,一般是添加一个cout或printf()——如果你使用C语言,如下所示:
void myfunc()
{
cout<<"myfunc()"<<endl;
//其他代码
}


  通常在一个典型的工程中,会包含有数千个函数,要在每个函数中都加入一条这样的输出语句,无疑难过上“蜀山”啊,因此,需要有一种机制,可以自动地完成这项操作。

  获取函数名

  作为一个C++程序员,可能经常遇到 __TIME__、__FILE__、__DATE__ 这样的宏,它们会在编译时,分别转换为包含编译时间、处理的转换单元名称及当前时间的字符串。

  在最新的ISO C标准中,如大家所知的C99,加入了另一个有用的、类似宏的表达式__func__,其会报告未修饰过的(也就是未裁剪过的)、正在被访问的函数名。请注意,__func__不是一个宏,因为预处理器对此函数一无所知;相反,它是作为一个隐式声明的常量字符数组实现的:
static const char __func__[] = "function-name";


  在function-name处,为实际的函数名。为激活此特性,某些编译器需要使用特定的编译标志,请查看相应的编译器文档,以获取具体的资料。

  有了它,我们可免去大多数通过手工修改,来显示函数名的苦差事,以上的例子可如下所示进行重写:
void myfunc()
{
cout<<"__FUNCTION__"<<endl;
}


  官方C99标准为此目的定义的__func__标识符,确实值得大家关注,然而,ISO C++却不完全支持所有的C99扩展,因此,大多数的编译器提供商都使用 __FUNCTION__ 取而代之,而 __FUNCTION__ 通常是一个定义为 __func__ 的宏,之所以使用这个名字,是因为它已受到了大多数的广泛支持。

  在Visual Studio 2005中,默认情况下,此特性是激活的,但不能与/EP和/P编译选项同时使用。请注意在IDE环境中,不能识别__func__ ,而要用__FUNCTION__ 代替。

  Comeau的用户也应使用 __FUNCTION__ ,而不是 __func__ 。

  C++ BuilderX的用户则应使用稍稍不同的名字:__FUNC__ 。

  GCC 3.0及更高的版本同时支持 __func__ 和__FUNCTION__ 。

  一旦可自动获取当前函数名,你可以定义一个如下所示显示任何函数名的函数:
void show_name(const char * name)
{
cout<<name<<endl;
}

void myfunc()
{
show_name(__FUNCTION__); //输出:myfunc
}

void foo()
{
show_name(__FUNCTION__); //输出:foo
}


  因为 __FUNCTION__ 会在函数大括号开始之后就立即初始化,所以,foo()及myfunc()函数可在参数列表中安全地使用它,而不用担心重载。

  签名与修饰名

  __FUNCTION__ 特性最初是为C语言设计的,然而,C++程序员也会经常需要有关他们函数的额外信息,在Visual Studio 2005中,还支持另外两种非标准的扩展特性:__FUNCDNAME__ 与 __FUNCSIG__ ,其分别转译为一个函数的修饰名与签名。函数的修饰名非常有用,例如,在你想要检查两个编译器是否共享同样的ABI时,就可派得上用场,另外,它还能帮助你破解那些含义模糊的链接错误,甚至还可用它从一个DLL中调用另一个用C++链接的函数。在下例中,show_name()报告了函数的修饰名:
void myfunc()
{
show_name(__FUNCDNAME__); //输出:?myfunc@@YAXXZ
}


  一个函数的签名由函数名、参数列表、返回类型、内含的命名空间组成。如果它是一个成员函数,它的类名和const/volatile限定符也将是签名的一部分。以下的代码演示了一个独立的函数与一个const成员函数签名间的不同之处,两个函数的名称、返回类型、参数完全相同:
void myfunc()
{
show_name(__FUNCSIG__); // void __cdecl myfunc(void)
}

struct S
{
void myfunc() const 
{
show_name(__FUNCSIG__); //void __thiscall S::myfunc(void) const
}
};


### 使用 `split` 方法概述 #### Java 中的 `split()` 方法 在 Java 编程语言中,`split()` 是一种用于字符串操作的强大工具。此方法允许依据指定的正则表达式来切割字符串,并返回一个由这些片段组成的数组[^1]。 ```java String text = "apple,banana,cherry"; String[] result = text.split(","); for (String item : result) { System.out.println(item); } ``` 上述代码展示了如何利用逗号作为分隔符将字符串 `"apple,banana,cherry"` 切割成三个独立的部分并打印出来。 #### Python 中的 `split()` 方法 Python 提供了一个简单易用的 `split()` 函数,它可以根据给定的字符或模式对字符串进行分割。需要注意的是,在对象内部调用该函数时应采用点运算符的形式[^2]。 ```python text = "(apple, banana, cherry)" result = text.strip('()').replace(' ', '').split(',') print(result) ``` 这段脚本先去除了原始字符串两端多余的括号和空白空间,再按照逗号将其划分为列表项。 #### NumPy 库里的 `split` 和 `array_split` 方法 对于数值计算库 NumPy 而言,则存在两种不同的分裂机制——`split` 及 `array_split` 。前者适用于希望获得大小相等(尽可能接近)的结果集;而后者不强制要求各部分保持一致长度,因此更为灵活[^3][^4]。 ```python import numpy as np arr = np.arange(8) # 使用 array_split 不考虑均匀分布 sub_arrays = np.array_split(arr, 3) print([a.tolist() for a in sub_arrays]) # 使用 split 需要能整除的情况 even_subarrays = np.split(arr[:6], 3) # 前六个元素可被三均分 print([b.tolist() for b in even_subarrays]) ``` 这里给出了两个例子分别说明了这两种不同类型的划分方式。 #### JavaScript 中的 `split()` 方法 最后来看一看 Web 开发常用的客户端脚本语言 JavaScript 下是如何运用 `split()` 来处理字符串的。下面的例子演示了怎样基于空格把一句话分解开来[^5]: ```javascript var sentence = "hello world! hello man!"; var words = sentence.split(" "); console.log(words); // 输出 ["hello", "world!", "", "hello", "man!"] ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值