- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我收藏了 Creature
使用 std::make_shared
在我的应用程序的一部分中创建和拥有的对象和 std::shared_ptr
.
我还跟踪了零个或一个的选择 Creature
在World
使用 std::weak_ptr<Creature>
的对象.
void World::SetSelection(const std::shared_ptr<Creature>& creature) {
selection = creature;
}
std::shared_ptr<Creature> World::GetSelection() const {
return selection.lock();
}
GetSelection
的来电者负责检查指针是否为空。如果是,则表示当前没有选择。
This all works perfectly to my liking: when the selected Creature
死于自然原因(在申请的其他地方),GetSelection
开始返回 nullptr
再次好像什么都没有被选择。
但是在那种情况下 World::selection
成员仍然指向 std::shared_ptr
的控制 block 。这可能很大,因为我使用 std::make_shared
创建我的 Creature
对象(我意识到 Creature
对象在正确的时间被正确销毁,但它的内存仍然被分配)。我正在考虑更改 GetSelection
对此:
std::shared_ptr<Creature> World::GetSelection() {
const auto ret = selection.lock();
if (!ret)
selection.reset();
return ret;
}
这会在我发现不再需要内存时立即释放内存。恼人的是,这个版本的GetSelection
不能是 const
.
哪个版本GetSelection
在这种情况下会被认为是最佳实践吗?
如果在模板代码中发生类似的事情,答案是否会改变,其中 sizeof(T)
是未知的,可能是巨大的?或者在 C++14 中 std::make_shared<T[]>
可以参与吗?
如果第二个版本总是最好的,那么 std::weak_ptr<T>::expired
的基本原理是什么?和 lock
自己不做?
最佳答案
首先应该注意的是,std::make_shared
的放置策略是可选的,即标准不强制执行此优化。这是一个非约束性要求,这意味着完全符合要求的实现可能会选择放弃它。
回答您的问题:
鉴于您似乎只有一个选择(因此您并没有通过保留许多这样的控制 block 来增加内存使用量),我主张保持简单。内存是瓶颈吗?这对我来说是微优化。您应该编写更简单的代码,您可以在其中应用 const
,然后在需要时返回并稍后进行优化。
答案不会无条件地改变,它会根据问题域和你的瓶颈是什么而改变。如果您正在分配一个“巨大”的对象(比如一百千字节),并且该对象的空间在一个相对未使用的控制 block 中四处游荡,直到被替换,这可能不是您的瓶颈,而且可能不值得编写更多代码(本质上更容易出错、难以维护和破译)来“解决”。
由于std::weak_ptr::lock
和std::weak_ptr::expired
是const
,在解释下const
对于 C++11,它们必须是线程安全的。因此,给定一些 std::weak_ptr
,同时调用 lock()
和 expired()
的任意组合必须是安全的。在引擎盖下,std::weak_ptr
存储一个指向控制 block 的指针,它通过它查看检查/增量/等。原子计数器来确定对象是否已经过期,或者看它是否可以获取锁。如果您想在 std::weak_ptr
内部实现优化,您必须以某种方式检查控制 block 的状态,然后在指针过期时自动删除指向控制 block 的指针。这会在每次访问 std::weak_ptr
时产生开销(即使这可以简单地用原子完成,它仍然会有开销),所有这些都是为了一个小的优化。
关于c++ - 如果我碰巧注意到它已过期,我应该在 weak_ptr 上调用 reset 吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25274871/
每次暂存文件时,如果您需要取消暂存文件,Git 都会提供有用的说明: (use "git reset HEAD ..." to unstage) 不过体面Git Tutorials by Atlass
我需要添加几个文件以将它们组合到一个提交中,但我必须排除其中一个。在 this answer ,执行此操作的代码是: git add -u git reset -- file_to_ignore.da
这个问题在这里已经有了答案: What are typical use cases of git-reset's --merge and --keep flags? (4 个答案) 关闭 6 年前。
有时候,进行了错误的提交,但是还没有push到远程分支,想要撤销本次提交,可以使用git reset –-soft/hard命令。 1、二者区别: git reset –-soft:回退到某个版
我认为:软重置:从重置向量启动。硬复位:拉CPU的电平。 最佳答案 硬复位当然意味着整个CPU芯片及其所有外设都被复位。造成这种情况的原因可能有很多:复位引脚被外部拉高、时钟故障、片内低电压检测、看门
$python manage.py reset Unknown command: 'reset' Type 'manage.py help' for usage. 在django 1.6中是否取消了这
我是 git 的新手,所以问题可能很简单,git reset --hard HEAD 和 git reset --hard 有什么区别? 最佳答案 HEAD 在您未指定该参数时是隐含的。 但是,您可以
我使用 apollo-link-state 来本地存储错误,但清除缓存后出现以下错误。 我已在 apollo 客户端配置选项中将 errors 的默认值设置为空数组 []。 但是,在 apolloCl
我正在使用 bool 数组来存储标志(类似于“已更改”)。数组的大小是静态的,在编译时已知。 我需要定期重置数组,即将所有元素设置为 false。我应该使用常规数组和类似 memset 或 memcp
在 git 文档(和许多 SO 线程)中,推荐使用这种重置方法: $ git reset --soft HEAD^ ;# go back to WIP state $ git reset
在我的实验中,我没能发现两者之间的任何功能差异 git reset --hard 和 git reset --merge 使用说明也没有给出任何提示 --hard res
如何重置所有列过滤器?调用 reset() 似乎重置了表,但过滤器的所有字段保持不变。 最佳答案 您可以将输入值绑定(bind)到表的过滤器,如下所示: 注意 [value] 绑定(bind)。 关
使用 std::unique_ptr::reset,您可以轻松地将您的实例恢复到新状态。 C++11 之前,为了实现类似的行为,我看到很多类都定义了一个 Reset() 方法来重置其所有内部成员。但现
为了恢复工作树和索引中的更改,此答案 ( https://stackoverflow.com/a/5812972/8278160) 建议运行以下命令: git reset --hard 运行它是否与运
我目前正在测试竞技场。我本来以为这段代码会编译,但在运行时失败了,令人惊喜的是,编译器发现了这个问题。但我不知道它的推理是否正确。有人能给我解释一下吗?。错误:。来自umpalo的相关代码:
我目前正在测试竞技场。我本来以为这段代码会编译,但在运行时失败了,令人惊喜的是,编译器发现了这个问题。但我不知道它的推理是否正确。有人能给我解释一下吗?。错误:。来自umpalo的相关代码:
我正在尝试在不触发“重置”事件的情况下重置我的收藏。我已经设置了我的收藏来收听“重置”和“添加”事件 @.listenTo(@options.muses, 'add', @addOne) @.list
根据http://en.cppreference.com/w/cpp/memory/unique_ptr/reset , void reset( pointer ptr = pointer() );
我有一个别名,unstage,用于从暂存区域中删除更改。 unstage = reset -- 我注意到 git 的帮助建议改为 git reset HEAD。我还注意到 git rm --cache
这个问题在这里已经有了答案: What's the difference between HEAD^ and HEAD~ in Git? (17 个答案) 关闭 6 年前。 git reset --
我是一名优秀的程序员,十分优秀!