条款21(二):必须返回对象时,别妄想返回其reference

127 篇文章 7 订阅
39 篇文章 3 订阅

条款21:必须返回对象时,别妄想返回其reference

Don’t try to return a reference when you must return an object.
在上一篇中,我们分别接受了stack和heap两种方式的构造对象,然而,无论是on-the-stack,还是on-the-heap的做法,都因为对operator* 返回的结果调用构造函数而产生了问题,而最开始的目标就是避免如此的构造函调用动作!

于是,另一个想法则是基于:

  • 让operator* 返回的reference指向一个被定义于函数内部的static Rational对象。
const Rational& operator* ( const Rational& lhs,
                            const Rational& rhs)
{
    static Rational result;    //还是很糟糕!定义static对象,该函数将返回它的reference
    result = ... ;             //将lhs乘以rhs,并将结果置于result之内
    return result; 
}   

上面的代码之所以糟糕,是因为如果对于以下的使用代码:

bool operator==(const Rational& lhs,
                const Rational& rhs);   //一个针对Rational而写的operator==
Rational a, b, c, d;
///
if ((a * b) == (c * d))
{
    乘积相等所执行的动作
}
else
{
    乘积不等所执行的动作
}

出现的问题是:

  • 无论a, b, c, d的值是多少,表达式((a * b) == (c * d))总是被判定为True!

让我们将上述的if判断语句写成等价的函数形式:

if (operator==(operator*(a, b), operator*(c, d))

上面这句代码,可以看到在operator==被调用之前,已经有两个operator* 被调用了,并且每一个都返回reference指向operator* 内部定义的static Rational对象。
因此,operator==被要求将“operator* 内的static Rational对象值” 拿来和“operator* 内的static Rational对象值” 进行比较,这自然就是相等了。
值得注意的是,两次的operator*的调用确实是改变了static Rational对象值,但是由于他们都是返回reference,因此调用段看到的永远是static Rational对象的“现值”

于是,一个“必须返回新对象”的函数的正确写法是:

  • 直接让这个函数返回一个新对象
incline const Rational operator* ( const Rational& lhs, const Raitonal& rhs)
{
    return Rational(lhs.n * rhs.n, lhs.d * rhs.n);
}

当然了,operator*返回值的构造成本和析构成本是必须支出的,但是这只是获得正确行为的小小代价。
总之,

  • 当我们必须在“返回一个reference和返回一个object”之间进行选择时,选择行为正确的那一个。

最后:

绝不要返回pointer或reference指向一个local stack对象,或返回reference指向一个heap-allocated对象,或返回pointer指向一个local static对象而有可能同时需要多个这样的对象。

很好,我可以帮你编写一个简单的Python脚本来实现山海经跑图。 首先你需要导入两个Python库:PIL和pytesseract。PIL库用于图像处理,pytesseract库用于OCR识别文字。 接下来,你需要从图片中提取出文字。你可以使用PIL库中的Image模块打开图片,然后使用pytesseract库中的image_to_string函数将图片中的文字转换为字符串。 然后,你需要将山海经中的地名和路线提取出来。你可以使用Python的正则表达式来匹配这些字符串,并将它们保存到一个列表中。 最后,你可以使用Python的Web自动化库(如selenium)来模拟人工操作,让程序自动跑图。 以下是一个简单的Python脚本,用于提取山海经中的地名和路线,并自动跑图: ```python from PIL import Image import pytesseract import re from selenium import webdriver import time # 打开图片 img = Image.open('shanhaijing.jpg') # OCR识别图片中的文字 text = pytesseract.image_to_string(img, lang='chi_sim') # 提取地名和路线 place_pattern = re.compile(r'【(.*?)】') # 地名正则表达式 route_pattern = re.compile(r'→(.*?)\n') # 路线正则表达式 places = place_pattern.findall(text) routes = route_pattern.findall(text) # 打印地名和路线 for i in range(len(places)): print(f'{i + 1}、{places[i]}:{routes[i]}') # 自动跑图 driver = webdriver.Chrome() driver.get('https://shanhaimap.mihoyo.com/') # 点击开始游戏按钮 start_game_btn = driver.find_element_by_class_name('start-game') start_game_btn.click() # 等待动画播放完成 time.sleep(5) # 点击跑图按钮 run_btn = driver.find_element_by_class_name('run-btn') run_btn.click() # 依次点击地名和路线 for i in range(len(places)): time.sleep(1) place_btn = driver.find_element_by_xpath(f"//*[text()='{places[i]}']") place_btn.click() time.sleep(1) route_btn = driver.find_element_by_xpath(f"//*[text()='{routes[i]}']") route_btn.click() ``` 请注意,这只是一个简单的脚本,可能需要根据你的具体情况进行修改。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值