gpt4 book ai didi

c++ - 锁定自由原子状态类 - 是否正确?

转载 作者:太空狗 更新时间:2023-10-29 21:05:20 26 4
gpt4 key购买 nike

我只是在寻找有关我尝试在结构上实现原子读/写的反馈(明显的缺陷/改进方法)。

会有一个写线程和多个读线程。目的是防止读者对结构产生不一致的看法,同时又不会过多地妨碍作者。

我正在使用获取和添加原子原语,在本例中由 Qt 框架提供。

例如:

/* global */
OneWriterAtomicState<Point> atomicState;

/* Writer */
while(true) {
MyStruct s = atomicState.getState()
s.x += 2; s.y += 2;
atomicState.setState(s);
}

/* Reader */
while(true) {
MyStruct s = atomicState.getState()
drawBox(s.x,s.y);
}

OneWriterAtomicState 实现:

template <class T>
class OneWriterAtomicState
{
public:
OneWriterAtomicState()
: seqNumber(0)
{
}

void setState(T& state) {
this->seqNumber.fetchAndAddOrdered(1);
this->state = state;
this->seqNumber.fetchAndAddOrdered(1);
}

T getState(){
T result;
int seq;
bool seq_changed = true;

/* repeat while seq is ODD or if seq changes during read operation */
while( (seq=this->seqNumber.fetchAndAddOrdered(0)) & 0x01 || seq_changed ) {
result = this->state;
seq_changed = (this->seqNumber.fetchAndAddOrdered(0)!=seq);
}
return result;
}


private:
QAtomicInt seqNumber;
T state;
}

这是版本二(memcpy,读取器产生,希望修复 getState() ):

template <class T>
class OneWriterAtomicState
{

public:
OneWriterAtomicState()
: seqNumber(0)
{
/* Force a compile-time error if T is NOT a type we can copy with memcpy */
Q_STATIC_ASSERT(!QTypeInfo<T>::isStatic);
}

void setState(T* state) {
this->seqNumber.fetchAndAddOrdered(1);
memcpy(&this->state,state,sizeof(T));
this->seqNumber.fetchAndAddOrdered(1);
}

void getState(T* result){
int seq_before;
int seq_after = this->seqNumber.fetchAndAddOrdered(0);
bool seq_changed = true;
bool firstIteration = true;

/* repeat while seq_before is ODD or if seq changes during read operation */
while( ((seq_before=seq_after) & 0x01) || seq_changed ) {

/* Dont want to yield on first attempt */
if(!firstIteration) {
/* Give the writer a chance to finish */
QThread::yieldCurrentThread();
} else firstIteration = false;

memcpy(result,&this->state,sizeof(T));
seq_after = this->seqNumber.fetchAndAddOrdered(0);
seq_changed = (seq_before!=seq_after);
}
}

bool isInitialized() { return (seqNumber>0); }

private:
QAtomicInt seqNumber;
T state;
} ;

#endif // ONEWRITERATOMICSTATE_H

最佳答案

算法不太正确。这是读者获得不一致数据的一种可能的线程交错:

state initialized to {0,0} and seqNumber to 0

Writer:
seqNumber = 1;
state.x = 1;

Reader:
seq = seqNumber; //1
result = state; //{1,0}
seq_changed = (seqNumber != seq); //false

Writer:
state.y = 1;
seqNumber = 2;

Reader:
jumps back to the start of the loop
seq = seqNumber; //2
steps out of the loop because seq == 2 and seq_changed == false

所以问题是 seqNumber 在两个地方被读取,写入者有可能在两次读取之间更新值。

while( (seq=this->seqNumber.fetchAndAddOrdered(0)) & 0x01 || seq_changed ) {
result = this->state;
seq_changed = (this->seqNumber.fetchAndAddOrdered(0)!=seq);
//If writer updates seqNumber here to even number bad things may happen
}

每次迭代只应读取一次:

T getState(){
T result;
int seq;
int newseq = seqNumber.fetchAndAddOrdered(0);
bool seq_changed = true;

while( (seq = newseq) & 0x01 || seq_changed ) {
result = state;
newseq = seqNumber.fetchAndAddOrdered(0);
seq_changed = (newseq != seq);
}
return result;
}

我相信这应该能正常工作,但我不保证任何事情。 :) 至少你应该编写一个测试程序,就像你的例子中的那个,但在阅读器中添加一个不一致值的检查。

值得考虑的一件事是使用原子增量 (fetchAndAdd) 有点矫枉过正。只有一个线程写入 seqNumber,因此您可以执行简单的原子存储释放和加载获取操作,并且它们可以在许多处理器上更快地实现。但是我不知道这些操作是否可以用 QAtomicInt;文档对此非常不清楚。

编辑:wilx 是对的,T 需要是一个平凡可复制的类型

关于c++ - 锁定自由原子状态类 - 是否正确?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10210905/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com