- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在阅读 std::thread documentation at cppreference (并不总是 100% 准确,我知道)并注意到当传递“指向数据成员的指针”(不是“指向成员的指针”)时 std::thread
的行为的以下定义-function") 作为它的第一个参数 (f
) 和一个所需类的对象作为它的第二个参数 (t1
在复制到 thread-local-storage 之后):
If N == 1 and f is pointer to a member data object of a class, then it is accessed. The value of the object is ignored. Effectively, the following code is executed: t1.*f if and the type of t1 is either T, reference to T or reference to type derived from T. (*t1).*f otherwise.
现在,我不打算以这种方式使用 std::thread
,但我对这个定义感到困惑。显然,唯一发生的事情是数据成员被访问并且值被忽略,这似乎根本没有任何可观察到的副作用,这意味着(据我所知)它也可能是一个没有操作。 (我可能遗漏了一些明显的东西......?)
起初,我认为这可能是一个打印错误,意思是说数据成员被访问然后被调用(因为它可能是一个可调用对象,即使它不是一个函数)但我用下面的测试GCC-4.7 中的代码,确实没有调用:
#include <iostream>
#include <thread>
struct S
{
void f() {
std::cout << "Calling f()" << std::endl;
}
struct {
void operator()() {
std::cout << "Calling g()" << std::endl;
}
} g;
};
int main(int, char**)
{
S s;
s.f(); // prints "Calling f()"
s.g(); // prints "Calling g()"
std::cout << "----" << std::endl;
auto x = &S::f; // ptr-to-mem-func
auto y = &S::g; // ptr-to-data-mem
(s.*x)(); // prints "Calling f()"
(s.*y)(); // prints "Calling g()"
std::cout << "----" << std::endl;
std::thread t(x, &s);
t.join();
// "Calling f()" printed by now
std::thread u(y, &s);
u.join();
// "Calling g()" not printed
return 0;
}
这个定义似乎没有完成任何事情的目的是什么?为什么不让传递“指向数据成员可调用的指针”就像传递“指向成员函数的指针”一样,并使传递“指向数据成员不可调用的指针”成为错误?事实上,这似乎是实现它的最简单方法,因为在其他上下文中调用“指向数据成员的指针”具有等效于调用“指向成员函数的指针”的语法(除非模板特化和 SFINAE 规则的变化莫测,这使得很难将它们同等对待......?)
这不是我实际代码所需要的东西,但这个定义存在的事实让我怀疑我遗漏了一些基本的东西,这让我很担心......任何人都可以启发我吗?
最佳答案
这是因为通用绑定(bind)工具,C++11 标准不仅定义了线程的启动方式,还定义了如何启动 std::bind
。和 std::function
工作。
事实上,C++11 标准的第 30.3.1.2/3 段指定了类 std::thread
的可变参数构造函数。 :
template <class F, class ...Args> explicit thread(F&& f, Args&&... args);
Effects: Constructs an object of type thread. The new thread of execution executes
INVOKE (DECAY_-
with the calls to
COPY ( std::forward<F>(f)), DECAY_COPY (std::forward<Args>(args))...)DECAY_COPY
being evaluated in the constructing thread. Any return value from this invocation is ignored. [...]
忽略什么DECAY_COPY
确实(与问题无关),这就是第 20.8.2 段定义 INVOKE
的方式伪函数:
Define INVOKE (f, t1, t2, ..., tN) as follows:
— (t1.*f)(t2, ..., tN) when f is a pointer to a member function of a class T and t1 is an object of type T or a reference to an object of type T or a reference to an object of a type derived from T;
— ((*t1).*f)(t2, ..., tN) when f is a pointer to a member function of a class T and t1 is not one of the types described in the previous item;
— t1.*f when N == 1 and f is a pointer to member data of a class T and t1 is an object of type T or a reference to an object of type T or a reference to an object of a type derived from T;
— (*t1).*f when N == 1 and f is a pointer to member data of a class T and t1 is not one of the types described in the previous item;
— f(t1, t2, ..., tN) in all other cases.
现在问题变成了:
Why does the C++11 Standard defines the
INVOKE
facility that way?
答案是 here .
关于c++ - 带有指向数据成员的指针的 std::thread,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15080015/
test = (function(){var key = 200; return {getKey : function(){return key} }; })(); test.
如果这个问题可能一直被问到,我很抱歉,但我进行了搜索,但找不到足够的答案。 如果公共(public)成员/方法正在访问私有(private)成员/字段,如何禁用它们的继承? 所以考虑一下: publi
重要的澄清:一些评论者似乎认为我是从 union 复制的。仔细查看 memcpy,它从一个普通的旧 uint32_t 地址复制而来,该地址不包含在 union 中。另外,我正在(通过 memcpy)复
spinner 通常只显示一个字符串,在我想分配 IDpersonne 和 Name 的情况下,旋转器必须告诉我名字。当我得到选定的项目时,我必须得到 ID。我该怎么做? 最佳答案 我假设您已将项目排
A 类的实例是 B 类的公共(public)成员。B 类的实例也是 A 的公共(public)成员。在什么情况下可能需要这种实现?我的意思是是否有一个或多个标准场景需要这种实现方式?更具体的细节:我有
我如何设置我的 web.config 以使用表单例份验证,将成员身份提供程序设置为 ActiveDirectoryMembershipProvider 并使用内置登录控件。这样我就可以使用有效的事件目
这个问题已经有答案了: Should methods in a Java interface be declared with or without a public access modifier?
因此根据定义,类中的私有(private)数字在序列化时以类名作为前缀。这对我来说是一个问题,我希望能够序列化/保存/反序列化一个确切的对象,但是 php 所做的是给我另一个 classname+va
我实现了一个成员? clojure 中的函数如下: (defn member? [item seq] (cond (empty? seq) false (= item (first
我在这里的问题似乎总是与使用函数有关。它仍然让我困惑!在本教科书练习中,我被要求按值传递结构,然后调整它并按引用传递。最初我设计的代码是在 main 中完成所有工作。现在我正在传递值。所以我添加了新函
所以我有这些变量 List files, images = new List(); string rootStr; 还有这个线程函数 private static int[] thread_searc
我对 C++ 模板和尝试弄清楚部分模板特化还比较陌生。我正在使用模板实现几个相关的数据结构:用于概率存在/不存在查询的布隆过滤器(基于位数组),以及用于丰度查询的计数布隆过滤器(带有整数数组)。我从以
例如在 java 中,我在外部类和内部类中声明并初始化了一个 JButton,我决定在某些情况下将其隐藏,这是一种安全的编程实践吗? 最佳答案 内部类的全部目的是它们可以访问到环绕内部类的外部类。 所
我有一个使用库进行通信的类: class Topic { Topic( Type T, String name ); }; class Reader { Reader (Topic, Stri
我在两个单独的文件中有以下代码。 package animal; public class Frog { protected void ribbit() { Syste
我有一个分数列表。使用这些,我需要从 redis 排序集中提取值。 我知道我可以使用 zrangebyscore - 但如果我提供的列表中的分数不连续怎么办?在这种情况下,我不能依赖 zrangeby
过去几年我一直被 C# 编码宠坏了,现在我又回到了 C++ 并发现我在处理本应很简单的东西时遇到了麻烦。我正在为 gamedev 使用名为 DarkGDK 的第三方库(任何以 db 为前缀的命令),但
我正在关注 Brian Harvey 从 2011 年开始在 UC Berkeley site 上的 SICP 讲座。 .他正在使用 STk interpreter教这门课,我正在使用带有 DrRac
在这段代码中,为什么在运算符重载中无法访问我的类的私有(private)字段? (请注意,这只是一个 MRE,不是完整代码) template class Frac template Frac o
在命名命名空间类中,我将一个类(位于全局命名空间中)声明为友元。 但是,后一个类不能访问前一个类的私有(private)成员。为什么是这样?有什么办法可以解决吗? Bob.h namespace AB
我是一名优秀的程序员,十分优秀!