我一直在尝试为一个简单的类实现一个自定义的前向迭代器。这个类是固定数组的一个非常糟糕的抽象(不幸的是我不能改变它)。只能使用索引访问元素。
template <class T>
struct data
{
static const size_t MAX_BUFFER{ 50 };
T* buffer_[MAX_BUFFER] = {};
int currpos_ = 0;
void insert(T *value) {
if (currpos_ < MAX_BUFFER-1)
buffer_[currpos_++] = value;
}
T** at(int i) {
if (i >= currpos_)
return NULL;
return &buffer_[i];
}
~data() {
for (int i=0; i<currpos_; ++i)
delete buffer_[i];
}
int entries() const { return currpos_; }
struct iterator : std::iterator<std::forward_iterator_tag, T*>
{
using reference = typename std::iterator<std::forward_iterator_tag, T*>::reference;
using pointer = typename std::iterator<std::forward_iterator_tag, T*>::pointer;
iterator(data<T> *d, int start) : p{ d }, index{ start } {}
iterator operator++() { if (index < p->entries()) ++index; return *this; }
friend bool operator==(const iterator &d1, const iterator &d2) { return d1.p == d2.p && d1.index == d2.index; }
friend bool operator!=(const iterator &d1, const iterator &d2) { return !(d1== d2); }
reference operator*() { return *(p->at(index)); }
pointer operator->() { return p->at(index); }
data<T> *p;
int index;
};
iterator begin() { return iterator(this, 0); }
iterator end() { return iterator(this, entries()); }
};
我面临的问题是,通过这个接口(interface),我可以使用大多数 STL 标准算法,如 for_each、transform、find_if。例如,假设 d 已经用 new{2}, new{3}, new{4}, new{14}, new{-4}, new{-44}, new{42} 这个代码
for (auto &i : d) std::cout <<*i <<" "; std::cout <<std::endl;
auto res=std::find_if(d.begin(), d.end(), [](auto &i) { return *i == -44;});
if (res != d.end())
std::cout <<**res <<std::endl;
std::transform(d.begin(), d.end(), d.begin(), [](auto &i) {*i *= 2; return i;});
for (auto &i : d) std::cout <<*i <<" "; std::cout <<std::endl;
会正确显示
2 3 4 14 -4 -44 42
-44
4 6 8 28 -8 -88 84
我面临的问题是算法 std::remove_if() 及其排列元素的方式。我添加了一个类似于 vector::erase: 的成员函数:
void remove_range(iterator begin, iterator end)
{
size_t d=std::distance(begin, end);
currpos_ -= d;
}
这当然也应该删除与已删除元素关联的内存。在这个例子中调用它时:
auto new_end = std::remove_if(d.begin(), d.end(), [](auto &r) { return *r > 3; });
d.remove_range(new_end, d.end());
valdring 一直告诉我有 3 个内存泄漏(这是有道理的:有 3 个元素 >= 3)。我试图在 remove_range() 中添加一个删除操作,但是这个解决方案不起作用(程序崩溃)。在调试 session 期间,我打印出 d 的内部状态:
(gdb) p d
$1 = {static MAX_BUFFER = <optimized out>, buffer_ = {0x603010, 0x603030, 0x603050, 0x603070, 0x603090, 0x6030b0, 0x6030d0, 0x0 <repeats 43 times>}, currpos_ = 7}
(gdb) p d
$2 = {static MAX_BUFFER = <optimized out>, buffer_ = {0x603010, 0x603030, 0x603090, 0x6030b0, 0x603090, 0x6030b0, 0x6030d0, 0x0 <repeats 43 times>}, currpos_ = 7}
我可以看到 removed_if() 基本上移动了三个元素(大于 3 个),我认为我有泄漏的原因是因为应该删除的元素被移动了,所以原始指针永远泄漏了。
我的问题是:这里有什么办法可以避免这种泄漏吗?我需要定义一些额外的移动构造函数吗?
最佳答案
您看到的泄漏是由 remove_range 实现引起的,而不是迭代器。
I have added a member function similar to vector::erase ... which of course should also remove the memory associated to the deleted elements.
但事实并非如此。您在容器析构函数中对元素调用 delete,但在 erase 方法中没有对项目的 delete。
关于c++ - C+ 自定义迭代器?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32202595/
我正在尝试设置一个puppet节点,但rubygems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由rubygems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby
我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢
我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby1.9+ 关于ruby-主要:Objectwhenrun
我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin
我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano
我使用Ember作为我的前端和GrapeAPI来为我的API提供服务。前端发送类似:{"service"=>{"name"=>"Name","duration"=>"30","user"=>nil,"organization"=>"org","category"=>nil,"description"=>"description","disabled"=>true,"color"=>nil,"availabilities"=>[{"day"=>"Saturday","enabled"=>false,"timeSlots"=>[{"startAt"=>"09:00AM","endAt"=>
我在用Ruby执行简单任务时遇到了一件奇怪的事情。我只想用每个方法迭代字母表,但迭代在执行中先进行:alfawit=("a".."z")puts"That'sanalphabet:\n\n#{alfawit.each{|litera|putslitera}}"这段代码的结果是:(缩写)abc⋮xyzThat'sanalphabet:a..z知道为什么它会这样工作或者我做错了什么吗?提前致谢。 最佳答案 因为您的each调用被插入到在固定字符串之前执行的字符串文字中。此外,each返回一个Enumerable,实际上您甚至打印它。试试
我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c