Add support for writing a[i] = v instead of a.set(i, v) in the pvector class.

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-09-12 11:12:02 -07:00
parent f972cc9aae
commit 74b8a4f0ac
2 changed files with 49 additions and 10 deletions

View file

@ -61,7 +61,7 @@ bool operator==(pvector<T> v1, pvector<T> v2) {
return true; return true;
} }
static void driver(unsigned max_sz, unsigned max_val, unsigned num_ops, double push_freq, double copy_freq) { static void driver(unsigned max_sz, unsigned max_val, unsigned num_ops, double updt_freq, double copy_freq) {
std::vector<int> v1; std::vector<int> v1;
pvector<int> v2; pvector<int> v2;
pvector<int> v3; pvector<int> v3;
@ -79,14 +79,23 @@ static void driver(unsigned max_sz, unsigned max_val, unsigned num_ops, double p
lean_assert(v3[idx] == v1[idx]); lean_assert(v3[idx] == v1[idx]);
} }
} }
if (f < push_freq) { if (f < updt_freq) {
if (v1.size() >= max_sz) if (v1.size() >= max_sz)
continue; continue;
int a = std::rand() % max_val; int a = std::rand() % max_val;
if (!empty(v2) && rand()%2 == 0) {
unsigned idx = rand() % size(v2);
v1[idx] = a;
v2[idx] = a;
v3[idx] = a;
lean_assert(v1[idx] == v2[idx]);
lean_assert(v1[idx] == v3[idx]);
} else {
v1.push_back(a); v1.push_back(a);
v2.push_back(a); v2.push_back(a);
v3 = push_back(v3, a); v3 = push_back(v3, a);
lean_assert(back(v3) == a); lean_assert(back(v3) == a);
}
} else { } else {
if (v1.size() == 0) if (v1.size() == 0)
continue; continue;
@ -173,6 +182,22 @@ static void tst5() {
} }
} }
static void tst6() {
pvector<int> v;
v.push_back(0);
v.push_back(1);
pvector<int> v2 = v;
v2[0] = 10;
lean_assert(v2[0] == 10);
lean_assert(v[0] == 0);
v[0] = 5;
lean_assert(v2[0] == 10);
lean_assert(v[0] == 5);
v2[1] = 20;
lean_assert(v2[0] == 10);
lean_assert(v2[1] == 20);
}
#ifdef PVECTOR_PERF_TEST #ifdef PVECTOR_PERF_TEST
static void perf_vector(unsigned n) { static void perf_vector(unsigned n) {
std::vector<int> q; std::vector<int> q;
@ -250,6 +275,7 @@ int main() {
tst3(); tst3();
tst4(); tst4();
tst5(); tst5();
tst6();
#ifdef PVECTOR_PERF_TEST #ifdef PVECTOR_PERF_TEST
tst_perf1(); tst_perf1();
tst_perf2(); tst_perf2();

View file

@ -229,7 +229,7 @@ public:
\brief Access specified element \brief Access specified element
\pre i < size() \pre i < size()
*/ */
T const & operator[](unsigned i) const { T const & get(unsigned i) const {
lean_assert(i < size()); lean_assert(i < size());
cell const * it = m_ptr; cell const * it = m_ptr;
unsigned cost = 0; unsigned cost = 0;
@ -240,7 +240,7 @@ public:
if (const_cast<pvector*>(this)->update_quota_on_read(cost)) if (const_cast<pvector*>(this)->update_quota_on_read(cost))
return to_push(it).m_val; return to_push(it).m_val;
else else
return operator[](i); // representation was changed, \c it may have been deleted return get(i); // representation was changed, \c it may have been deleted
} }
break; break;
case cell_kind::PopBack: case cell_kind::PopBack:
@ -250,14 +250,14 @@ public:
if (const_cast<pvector*>(this)->update_quota_on_read(cost)) if (const_cast<pvector*>(this)->update_quota_on_read(cost))
return to_set(it).m_val; return to_set(it).m_val;
else else
return operator[](i); // representation was changed, \c it may have been deleted return get(i); // representation was changed, \c it may have been deleted
} }
break; break;
case cell_kind::Root: case cell_kind::Root:
if (const_cast<pvector*>(this)->update_quota_on_read(cost)) if (const_cast<pvector*>(this)->update_quota_on_read(cost))
return to_root(it).m_vector[i]; return to_root(it).m_vector[i];
else else
return operator[](i); // representation was changed, \c it may have been deleted return get(i); // representation was changed, \c it may have been deleted
} }
it = static_cast<delta_cell const *>(it)->m_prev; it = static_cast<delta_cell const *>(it)->m_prev;
cost++; cost++;
@ -344,6 +344,19 @@ public:
} }
} }
class ref {
pvector & m_vector;
unsigned m_idx;
public:
ref(pvector & v, unsigned i):m_vector(v), m_idx(i) {}
ref & operator=(T const & a) { m_vector.set(m_idx, a); return *this; }
operator T const &() const { return m_vector.get(m_idx); }
};
T const & operator[](unsigned i) const { return get(i); }
ref operator[](unsigned i) { return ref(*this, i); }
class iterator { class iterator {
friend class pvector; friend class pvector;
pvector const & m_vector; pvector const & m_vector;