由于c是char范例,取值范畴是[-128,127],假如宏EOF的值在char的取值范畴之外,那么if语句将老是失败,这种“危险”人们一般那边料获得!导致本例错误的责任并不在用户,是函数getchar误导了利用者。
l 【法则6-2-3】不要将正常值和错误符号混在一起返回。正常值用输出参数得到,而错误符号用return语句返回。
回首上例,C尺度库函数的设计者为什么要将getchar声明为令人模糊的int范例呢?他会那么傻吗?
在正常环境下,getchar简直返回单个字符。但假如getchar遇到文件竣事符号或产生读错误,它必需返回一个符号EOF。为了区别于正常的字符,只好将EOF界说为负数(凡是为负1)。因此函数getchar就成了int范例。
我们在实际事情中,常常会遇到上述令工钱难的问题。为了制止呈现误解,我们应该将正常值和错误符号分隔。即:正常值用输出参数得到,而错误符号用return语句返回。
函数getchar可以改写成 BOOL GetChar(char *c);
固然gechar比GetChar机动,譬喻 putchar(getchar()); 可是假如getchar用错了,它的机动性又有什么用呢?
2 【发起6-2-1】有时候函数原本不需要返回值,但为了增加机动性如支持链式表达,可以附加返回值。
譬喻字符串拷贝函数strcpy的原型:
char *strcpy(char *strDest,const char *strSrc);
strcpy函数将strSrc拷贝至输出参数strDest中,同时函数的返回值又是strDest。这样做并非添枝加叶,可以得到如下机动性:
char str[20];
int length = strlen( strcpy(str, “Hello World”) );
2 【发起6-2-2】假如函数的返回值是一个工具,有些场所用“引用通报”替换“值通报”可以提高效率。而有些场所只能用“值通报”而不能用“引用通报”,不然会堕落。
譬喻:
class String
{…
// 赋值函数
String & operate=(const String &other);
// 相加函数,假如没有friend修饰则只许有一个右侧参数
friend String operate+( const String &s1, const String &s2);
private:
char *m_data;
}
String的赋值函数operate = 的实现如下:
String & String::operate=(const String &other)
{
if (this == &other)
return *this;
delete m_data;
m_data = new char[strlen(other.data)+1];
strcpy(m_data, other.data);
return *this; // 返回的是 *this的引用,无需拷贝进程
}
对付赋值函数,该当用“引用通报”的方法返回String工具。假如用“值通报”的方法,固然成果仍然正确,但由于return语句要把 *this拷贝到生存返回值的外部存储单位之中,增加了不须要的开销,低落了赋值函数的效率。譬喻:
String a,b,c;
…
a = b; // 假如用“值通报”,将发生一次 *this 拷贝
a = b = c; // 假如用“值通报”,将发生两次 *this 拷贝
String的相加函数operate + 的实现如下:
String operate+(const String &s1, const String &s2)
{
String temp;
delete temp.data; // temp.data是仅含‘\0’的字符串
temp.data = new char[strlen(s1.data) + strlen(s2.data) +1];
strcpy(temp.data, s1.data);
strcat(temp.data, s2.data);
return temp;
}