- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我正在从头开始设计一个库,并希望获得尽可能良好的公共(public) API。我希望编译器因滥用而对我大喊大叫。因此,我给自己制定了以下规则:
整个库的真实(即深入和完整)const 正确性
所有东西(局部变量、成员变量、成员函数),预计不会改变的都被声明为const
.该常量性应传播到所有嵌套成员和类型。
明确和表达所有权
根据 C++ 核心指南,我将其定义为(iff 在当且仅当 的数学意义上):
unique_ptr<T>
或 T&&
iff 函数正在使用它(即取得所有权)shared_ptr<const T>
或 T const&
iff函数只读取它shared_ptr<T>
或 T&
iff 函数正在修改它而不取得所有权unique_ptr<T>
或 T
iff函数将所有权转移给调用者shared_ptr<const T>
或 T const&
iff 调用者应该只读取它(尽管调用者可以构建它的拷贝 - 给定 T
是可复制的)shared_ptr<T>
, T&
或 T*
(因为它会导致无法控制的副作用,我试图通过设计避免)隐藏的实现细节
现在我将使用抽象接口(interface),工厂将实现作为 unique_ptr<Interface>
返回.不过,我愿意接受可以解决我下面描述的问题的替代模式。
我不关心虚拟表查找,并希望通过各种方式避免动态转换(我认为它们是一种代码味道)。
现在,给定两个类 A
和 B
, 其中B
拥有可变数量的 A
秒。我们还有 B
-实现BImpl
(A
的实现在这里可能没有用):
class A
{};
class B {
public:
virtual ~B() = default;
virtual void addA(std::unique_ptr<A> aObj) = 0;
virtual ??? aObjs() const = 0;
};
class BImpl : public B {
public:
virtual ~BImpl() = default;
void addA(std::unique_ptr<A> aObj) override;
??? aObjs() const override;
private:
std::vector<unique_ptr<A>> aObjs_;
};
我被 B
的返回值卡住了A
vector 的 setter/getter 小号:aObjs()
.
它应该提供 A
的列表s 作为只读值,不转移所有权(规则 2.5。上面的 const 正确性)并且仍然为调用者提供对所有 A
的轻松访问。 s,例如用于基于范围的 for
或标准算法,例如 std::find
.
我为那些 ???
想出了以下选项:
std::vector<std::shared_ptr<const A>> const&
我每次调用 aObjs()
时都必须构建一个新 vector (我可以将其缓存在 BImpl
中)。这感觉不仅效率低下且不必要地复杂,而且看起来也不是最理想的。
替换aObjs()
通过一对函数( aObjsBegin()
和 aObjsEnd()
)转发 BImpl::aObjs_
的常量迭代器.
等等。我需要做那个 unique_ptr<A>::const_iterator
一个unique_ptr<const A>::const_iterator
得到我心爱的 const 正确性。再次令人讨厌的类型转换或中间对象。并且用户无法在基于范围的 for
中轻松使用它.
我缺少什么明显的解决方案?
编辑:
B
应该总是能够修改 A
它持有,因此宣布aObjs_
作为vector<std::unique_ptr<const A>>
不是一个选项。
让 B
坚持迭代器概念迭代A
s, 既不是 B
的选项将持有 C
的列表s 和特定的 D
(或没有)。
最佳答案
您可以返回一个 vector 包装器,而不是直接返回 vector ,它允许您仅使用 const 指针访问内容。这听起来可能很复杂,但事实并非如此。只需制作一个薄 wrapper 并添加一个 begin()
和 end()
允许迭代的函数:
struct BImpl : B {
virtual ~BImpl() = default;
void addA(std::unique_ptr<A> aObj) override;
ConstPtrVector<A> aObjs() const override {
return aObjs_;
}
private:
std::vector<unique_ptr<A>> aObjs_;
};
ConstPtrVector
看起来像这样:
template<typename T>
ConstPtrVector {
ConstPtrVector(const std::vector<T>& vec_) : vec{vec_} {}
MyConstIterator<T> begin() const {
return vec.begin();
}
MyConstIterator<T> end() const {
return vec.end();
}
private:
const std::vector<T>& vec;
};
并且您可以实现 MyConstIterator
以一种将指针作为常量返回的方式:
template<typename T>
struct MyConstIterator {
MyConstIterator(std::vector<unique_ptr<T>>::const_iterator it_) : it{std::move(it_)} {}
bool operator==(const MyConstIterator& other) const {
return other.it == it;
}
bool operator!=(const MyConstIterator& other) const {
return other.it != it;
}
const T* operator*() const {
return it->get();
}
const T* operator->() const {
return it->get();
}
MyConstIterator& operator++() {
++it;
return *this;
}
MyConstIterator& operator--() {
--it;
return *this;
}
private:
std::vector<unique_ptr<T>>::const_iterator it;
};
当然,您可以通过实现类似接口(interface)的 vector 来概括此迭代器和包装器。
然后,要使用它,您可以使用基于范围的 for 循环或基于经典迭代器的循环。
顺便说一句:不拥有原始指针并没有错。只要他们仍然不拥有。如果您想避免由于原始指针导致的错误,请查看 observer_ptr<T>
,它可能会有用。
关于c++ - 在抽象接口(interface)中不转移所有权的指针 vector 的常量正确访问器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39257307/
如果我将我的个人 repo 转移到一个组织(由我创建),我将失去所有 见解 例如来自原始 Repo 的流量历史记录、拉取请求、贡献者、 fork 等? 最佳答案 拉取请求被保留:参见“About re
如何为解析 if-then[-else] 案例制定正确的规则?这是一些语法: { module TestGram (tparse) where } %tokentype { String }
如何为解析 if-then[-else] 案例制定正确的规则?这是一些语法: { module TestGram (tparse) where } %tokentype { String }
我读过有关mutex的信息,这些信息由线程拥有,并且只能由拥有的线程使用。在this answer中,该解决方案建议每个进程在发出互斥信号之前,必须拥有互斥锁的所有权。我必须在这里承认自己的愚蠢,不知
我只能从回调函数之一中想到 curl_close() 。 但是 php 抛出了一个警告: PHP 警告:curl_close():尝试从回调中关闭 cURL 句柄。 任何想法如何做到这一点? 最佳答案
带有冲突的语法的精简版本: body: variable_list function_list; variable_list: variable_list variable | /* empty
我创建了新的开发者帐户,然后将应用程序转移到新帐户。然后我在新帐户下创建了相同的标识符。并构建App并上传到AppStore。 I have got the warning with WARNING
我想像这样管理类主任的所有 Activity : 此外所有 Activity 都扩展基本 Activity 以使用公共(public) View 。 在这种情况下,我想处理传输 Activity ,例
使用 C 中的简单链表实现,我如何告诉 Splint 我正在转让 data 的所有权? typedef struct { void* data; /*@null@*/ void* ne
请参阅以下 yacc 代码。如果我删除生产因素:'!' expr,解析冲突消失。这里发生了什么? %{ #include #include %} %token TRUE %token FALSE
是否可以将 props 向下传输到子组件,其中 { ..this.props } 用于更简洁的语法,但是排除某些 props,如 className 或 id? 最佳答案 您可以使用解构来完成这项工作
如果我有以下数据框: date A B M S 20150101 8 7 7.5 0 20150101 10 9 9
我需要将一个 __m128i 变量(比如 v)移动 m 位,以便位移动所有变量(因此,结果变量表示 v*2^m)。执行此操作的最佳方法是什么?! 请注意 _mm_slli_epi64 分别移动 v0
我需要这样调用我的程序: ./program hello -r foo bar 我从 argv[1] 中打招呼,但我在使用值 bar 时遇到问题,我是否也应该将“r:”更改为其他内容? while((
我是新来的 Bison我在转换/减少冲突方面遇到了麻烦...我正在尝试从文件加载到 array data[] : struct _data { char name[50]; char sur
当然有很多关于解决移位/归约错误的文档和方法。 Bison 文档建议正确的解决方案通常是%期待它们并处理它。 当你遇到这样的事情时: S: S 'b' S | 't' 您可以像这样轻松解决它们: S:
我有以下(大量精简的)快乐语法 %token '{' { Langle } '}' { Rangle } '..' { DotDot } '::' { ColonC
我的 Bison 解析器中有很多错误,即使它运行良好,我也想了解这些冲突。代码如下: 词法分析器: id ([[:alpha:]]|_)([[:alnum:]]|_)* %% {id
在我的项目中,我有这样的情况,一个 Activity 应该将值(value)转移到另一个 Activity 。并且根据这个值应该选择需要的菜单元素。我试图在 bundle 的帮助下做到这一点,但我不知
我一直在阅读 NSIndexPaths 以获得 uitableviews 等。但是我很难操纵现有的索引路径。 我想在保留行的同时采用现有的索引路径递增/移动每个部分。因此 indexPath.sect
我是一名优秀的程序员,十分优秀!