- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
来自这里:Logic error in my defined Mutex class and the way I use it in producer consumer program - pthreads
the way you pass around references(!) to your mutex class is plainly asking for trouble, it defies any kind of encapsulation.
为什么这是个问题?我应该按值传递然后编写复制构造函数吗?
在这种情况下,缺乏封装会造成什么危害?我应该如何封装什么?
此外,Why is passing references to the Mutex class not a good design?
Passing a reference to the lock is a bad idea -- you don't "use" the lock, you only acquire and then you give it back. Moving it around makes it hard to track your use of the (critical) resource. Passing a reference to a mutex variable rather than a lock is maybe not so bad, but still it makes it harder to know what parts of the program may deadlock, so its something to avoid.
请用简单的语言举例说明 - 为什么传递引用是个坏主意?
最佳答案
我认为这是糟糕的抽象和糟糕的封装。 mutex
通常是在删除复制构造函数的情况下默认构造的,具有引用同一逻辑对象的多个互斥对象很容易出错,即它可能导致死锁和其他竞争条件,因为程序员或读者可以假设它们是不同的实体。
此外,通过指定您正在使用的内部互斥锁,您将公开线程的实现细节,从而破坏 Mutex 类的抽象。如果您使用的是 pthread_mutex_t
,那么您很可能会使用内核线程 (pthreads)。
封装也被破坏,因为您的 Mutex 不是一个单独的封装实体,而是分散到多个(可能是悬挂的)引用中。
如果你想将一个pthread_mutex_t
封装到一个类中,你可以这样做
class Mutex {
public:
void lock();
void unlock();
// Default and move constructors are good!
// You can store a mutex in STL containers with these
Mutex();
Mutex(Mutex&&);
~Mutex();
// These can lead to deadlocks!
Mutex(const Mutex&) = delete;
Mutex& operator= (const Mutex&) = delete;
Mutex& operator= (Mutex&&) = delete;
private:
pthread_mutex_t internal_mutex;
};
Mutex 对象旨在在实现文件中声明的共享范围内共享,而不是在本地声明并作为引用在函数中传递。理想情况下,您只会将参数传递给您需要的线程构造函数。传递对在与相关函数(在本例中为线程执行)处于相同“级别”的范围内声明的对象的引用通常会导致代码错误。如果声明互斥锁的范围不再存在会怎样? mutex
的析构函数是否会使互斥锁的内部实现无效?如果互斥量通过传递进入整个其他模块,并且该模块启动自己的线程并认为互斥量永远不会阻塞,会发生什么情况,这可能会导致严重的死锁。
还有一种情况,你想使用互斥量移动构造函数,比如在互斥量工厂模式中,如果你想创建一个新的互斥量,你将进行一个函数调用,该函数将返回一个互斥量,然后你将添加该互斥量到您的互斥量列表或传递给通过某种共享数据请求它的线程(上述列表对于此共享数据是个好主意)。然而,获得这样一个正确的互斥锁工厂模式可能非常棘手,因为您需要锁定对互斥锁公共(public)列表的访问。尝试应该是一件有趣的事情!
如果作者的意图是避免全局范围,那么在实现文件中将其声明为静态对象一次就足够了。
关于c++ - 为什么传递对 Mutex 类的引用不是一个好的设计?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34553232/
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: template pass by value or const reference or…? 以下对于将函数
我用相同的参数列表重载了一个运算符两次。但返回类型不同: T& operator()(par_list){blablabla} const T& operator()(par_list){bla
假设我有实现接口(interface) I 的 Activity A。我的 ViewModel 类 (VM) 持有对实现接口(interface) I 的对象的引用: class A extends
PHP 如何解释 &$this ?为什么允许? 我遇到了以下问题,这看起来像是 PHP 7.1 和 7.2 中的错误。它与 &$this 引用和跨命名空间调用以及 call_user_func_arr
谁能解释一下下面“&”的作用: class TEST { } $abc =& new TEST(); 我知道这是引用。但是有人可以说明我为什么以及什么时候需要这样的东西吗?或者给我指向一个对此有很好解
引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字。一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量。 C++ 引用 vs 指针 引用很容易与指针混淆,它们之间有三
目录 引言 背景 结论 引言 我选择写C++中的引用是因为我感觉大多数人误解了引用。而我之所以有这个感受是因为我主持过很多C++的面试,并且我很少
Perl 中的引用是指一个标量类型可以指向变量、数组、哈希表(也叫关联数组)甚至函数,可以应用在程序的任何地方 创建引用 定义变量的时候,在变量名前面加个 \,就得到了这个变量的一个引用 $sc
我编写了一个将从主脚本加载的 Perl 模块。该模块使用在主脚本中定义的子程序(我不是维护者)。 对于主脚本中的一个子例程,需要扩展,但我不想修补主脚本。相反,我想覆盖我的模块中的函数并保存对原始子例
我花了几个小时试图掌握 F# Quotations,但我遇到了一些障碍。我的要求是从可区分的联合类型中取出简单的函数(只是整数、+、-、/、*)并生成一个表达式树,最终将用于生成 C 代码。我知道使用
很多时候,问题(尤其是那些标记为 regex 的问题)询问验证密码的方法。似乎用户通常会寻求密码验证方法,包括确保密码包含特定字符、匹配特定模式和/或遵守最少字符数。这篇文章旨在帮助用户找到合适的密码
我想通过 MIN 函数内的地址(例如,C800)引用包含文本的最后一个单元格。你能帮忙吗? Sub Set_Formula() ' ----------------------------- Dim
使用常规的 for 循环,我可以做类似的事情: for (let i = 0; i < objects.length; i++) { delete objects[i]; } 常规的 for-
在 Cucumber 中,您定义了定义 BDD 语法的步骤;例如,您的测试可能有: When I navigate to step 3 然后你可以定义一个步骤: When /^I navigate t
这是什么UnaryExpression的目的,以及应该怎样使用? 最佳答案 它需要一个 Expression对象并用另一个 Expression 包裹它.例如,如果您有一个用于 lambda 的表达式
给出以下内容 $("#identifier div:first, #idetifier2").fadeOut(300,function() { // I need to reference jus
我不知道我要找的东西的正确术语,但我要找的是一个完整的引用,可以放在双引号之间的语句,比如 *, node()、@* 以及所有列出的 here加上任何其他存在的。 我链接到的答案提供了一些细节,但还
This question's answers are a community effort。编辑现有答案以改善此职位。它当前不接受新的答案或互动。 这是什么? 这是常见问答的集合。这也是一个社区Wi
Closed. This question does not meet Stack Overflow guidelines。它当前不接受答案。 想改善这个问题吗?更新问题,以便将其作为on-topic
考虑下一个代码: fn get_ref(slice: &'a Vec, f: fn(&'a Vec) -> R) -> R where R: 'a, { f(slice) } fn m
我是一名优秀的程序员,十分优秀!