list::remove_if(???)

xuechengw 2004-10-12 03:54:26
HI ,

I have create the following code, would you please
give me some suggestions on how to write the parameter
of remove_if(), and the condition I want to use is the
element's flag is false?

----------------------------------------------
class PclElem
{
bool flag;
...
}

typedef list<PclElem*> PclElemList;

PclElemList *pcPclElemList = new PclElemList;

// add element to the list and set the flag for each element
...

// remove the element that the flag is false
pcPclElemList->remove_if(????).
----------------------------------------------

With many thanks!
Bst Rgd!
...全文
247 19 打赏 收藏 转发到动态 举报
写回复
用AI写文章
19 条回复
切换为时间正序
请发表友善的回复…
发表回复
pongba 2004-10-17
  • 打赏
  • 举报
回复
throw away the "not_equal_to", use the solution I gave out initially.
xuechengw 2004-10-15
  • 打赏
  • 举报
回复
Thanks!
I will download and study the boost library, that's a big one for me:(
but without boost library, is there any solution?
pongba 2004-10-14
  • 打赏
  • 举报
回复
class PclElem
{
bool flag;

//add mem function here -- actually,did you already have it?
bool getFlag()
{
return flag;
}
};

vector<PclElem*> vec;//maybe you wanna store "PclElem" instead of "PclElem*" here,too...
std::remove_if(vec.begin(),vec.end(),std::not1(std::mem_fun(&PclElem::getFlag)));

//to take advantage of the enhanced 'mem_fun' of boost,you can use:
std::remove_if(vec.begin(),vec.end(),std::not1(boost::mem_fun(&PclElem::getFlag)));
//this fits all,no matter whether you put PclElem* or PclElem in your container!

--
cheers
Jinhao 2004-10-14
  • 打赏
  • 举报
回复
你把他改成 template<typename _Prl>
demo001 2004-10-14
  • 打赏
  • 举报
回复
mark
Jinhao 2004-10-14
  • 打赏
  • 举报
回复
to xuechengw(红桃k')
typedef binder2nd<not_equal_to<_Ty> > _Pr1; 这是多余的! 应该说是VC6的std lib的BUG
xuechengw 2004-10-14
  • 打赏
  • 举报
回复
The parameter type of *std::list::remove_if* is :
---------------------------------------------
typedef binder2nd<not_equal_to<_Ty> > _Pr1;
---------------------------------------------
Thus the solution you all printed are not suitable
and seems fit to the *std::remove_if* function.

Would you please provide me some other solutions, With
many thank!~
xuechengw 2004-10-14
  • 打赏
  • 举报
回复
TO grooving(向结贴的兄弟们敬礼!) ,
The following compile time error
---------------------------------------------------------
error C2664: 'remove_if' : cannot convert parameter 1 from 'class is_odd<int>' to 'class std::binder2nd<struct std::not_equal_to<int> >'
No constructor could take the source type, or constructor overload resolution was ambiguous
---------------------------------------------------------
xuechengw 2004-10-14
  • 打赏
  • 举报
回复
TO Jinhao(辣子鸡丁)(短歌马甲 No.0)
the following compile time error
-----------------------------------------------------------
error C2664: 'remove_if' : cannot convert parameter 1 from 'class A<class PclElem>' to 'class std::binder2nd<struct std::not_equal_to<class PclElem> >'
No constructor could take the source type, or constructor overload resolution was ambiguous
-----------------------------------------------------------
pongba 2004-10-14
  • 打赏
  • 举报
回复
attention: there's no way to implement an operator != like this:

bool operator != (PclElem const * p1,PclElem const * p2) //compile error
{
p1->flag!=p2->flag;
}

...remove_if(not_equal_to<PclElem*>());

It seems very attractive.But note the concept the code expresses--it changes the semantic of build in operator != of build in pointer type,so stop it,or anyway the compiler will stop it by throw you a compiler error saying that operator != can't be overloaded for buildin types only!
pongba 2004-10-14
  • 打赏
  • 举报
回复
Since I haven't my own notebook at hand which has boost installed,I just give you some "like this" code,check below:

//first,add operator == and operator != to your PclElem since it's comparable
//then include <boost/lambda/lambda.hpp> //or boost/lambda.hpp,somehow I forgot it:-)

using boost::lambda::_1;

//then use this
...remove_if( *_1 != nouse ); //I'll test it when I came back to my notebook

//for now ,wish you good luck:-)
pongba 2004-10-14
  • 打赏
  • 举报
回复
to xuechengw(红桃k'),my buddy,it seems that you've misunderstood something.

First to all,which is most important,is that your not_equal_to<PclElem*>(...) above actually did a *Pointer Comparation*,which of course always yeilds a *false* because no element in the list get the same address with &nose!

And some other things should be known by you:
Since the Type of elements stored here is PclElem* which is a pointer type,using "not_equal_to" naively will always be an error,which occurs at runtime rather than compile time because two pointers can be compared,which is most puzzling!

So,Are there any solutions?There is ,but More complexed than you may think if you wanna go a shortcut.
Anyway I'll show you below.

Rational : There's absolutely no need for a virtual operator() in a predication functor,because the functor is passed as a template type parameter,or have its type deduced ,compiler knows(and should know) what the predication is.

Virtual functions here just add runtime cost.

Be clear that what you face is *template*,which is a *complete* compile-time-thing,so why do we need a oo approach(ie..via virtual functions) since compiler can get(or deduce) the full type information about our Predication functor?

Actually,as you see from the source code of remove_if,it just treat the predication as a static(not dynamic) type and invoke the operator() on it,not via virtual functions!

Review your 'test' class,and find what mistake you've made: you needn't derive your test from "not_equal_to" which isn't for your derivation but for a simplified in-line construction of the simple operator !=.

Have a break,and think it twice,latter i'll show you some solution.
xuechengw 2004-10-14
  • 打赏
  • 举报
回复
Thanks to all!
I have just get a look at the std library(V2.95.3) at
solaris2.8, and found the parameter type of std::list::remove_if
is
---------------------
typedef template <class _Predicate> _Predicate;
---------------------
, which is diffrent from the one that VC6 provided
---------------------
typedef binder2nd<not_equal_to<_Ty> > _Pr1;
---------------------
Thus the solution that Jinhao(辣子鸡丁)(短歌马甲 No.0) provide
is right under the solaris platform. Sorry!

pongba(世界是泛型的...)'s suggestion is just what I am tring
to learn. Thanks!

But as to the problem I have encountered, *the parameter of
std::list::remove_if*.
(1) definition of std::list::remove_if
--------------------------------------------------
typedef binder2nd<not_equal_to<_Ty> > _Pr1;
void remove_if(__Pr1 _Pr)
{iterator _L = end();
for (iterator _F = begin(); _F != _L; )
if (_Pr(*_F))
erase(_F++);
else
++_F; }
--------------------------------------------------
(2)definition of class binder2nd
-----------------------------------------------------------------
template<class _Bfn>
class binder2nd
: public unary_function<_Bfn::first_argument_type,
_Bfn::result_type> {
public:
binder2nd(const _Bfn& _X,
const _Bfn::second_argument_type& _Y)
: op(_X), value(_Y) {}
result_type operator()(const argument_type& _X) const
{return (op(_X, value)); }
protected:
_Bfn op;
_Bfn::second_argument_type value;
};
-----------------------------------------------------------------
(3) definition of struct not_equal_to
-----------------------------------------------------------------
template<class _Ty>
struct not_equal_to : binary_function<_Ty, _Ty, bool> {
bool operator()(const _Ty& _X, const _Ty& _Y) const
{return (_X != _Y); }
};
-----------------------------------------------------------------
Then my source code is realized as below, but it still dose not
work well, after call remove_if, all of the element have been
removed, but there must be some element with flag is true.
--------------------------------------------------
class test :public not_equal_to<PclElem*>
{
bool operator()(const PclElem* & rhs, const PclElem* &) const
{
return !rhs->flag;
}

};

PclElem nouse;
nouse.flag = true;

pcPclElemList->remove_if(
binder2nd<not_equal_to<PclElem*> >(test(), &nouse));
--------------------------------------------------
I found that in template class not_equal_to, the member
function bool operator()(...) is not a virtual function,
then the class that I have inherited can not override this
function.
If my analysis is right, then how to realize the paremeter
of std::list::remove_if under the MS VC6 enviromnet?
pongba 2004-10-14
  • 打赏
  • 举报
回复
of course,the member cousin of remove_if is more efficient,but that's not the point here,the point is how to build the predicating function (functor).

But anyway,I really made a mistake.
Jinhao 2004-10-14
  • 打赏
  • 举报
回复
i don't know why u don't use std::list<>::remove_if?
it is more efficient!
pongba 2004-10-14
  • 打赏
  • 举报
回复
sorry,I missed <functional> in my code above...
pongba 2004-10-14
  • 打赏
  • 举报
回复
actually,xuechengw,you've asked a pretty good question,some budies just write their functions (or functors) by themselvs,never got a sight that the std header--<functional>--has offered a large amount of facilities for you to express your mind,especially when you compose some functions together.

BTW.It's better to see that boost provides even better(sometimes I would rather say "perfect") facilities for functional programming,ie..enhanced bind ,mem_fun,lambda,So just use them ,you'll find them really amazing!
grooving 2004-10-12
  • 打赏
  • 举报
回复
#include <list>
#include <iostream>

template <class T> class is_odd : public std::unary_function<T, bool>
{
public:
bool operator( ) ( T& val )
{
return ( val % 2 ) == 1;
}
};

int main( )
{
using namespace std;
list <int> c1;
list <int>::iterator c1_Iter, c2_Iter;

c1.push_back( 3 );
c1.push_back( 4 );
c1.push_back( 5 );
c1.push_back( 6 );
c1.push_back( 7 );
c1.push_back( 8 );

cout << "The initial list is c1 =";
for ( c1_Iter = c1.begin( ); c1_Iter != c1.end( ); c1_Iter++ )
cout << " " << *c1_Iter;
cout << endl;

list <int> c2 = c1;
c2.remove_if( is_odd<int>( ) );

cout << "After removing the odd elements, "
<< "the list becomes c2 =";
for ( c2_Iter = c2.begin( ); c2_Iter != c2.end( ); c2_Iter++ )
cout << " " << *c2_Iter;
cout << endl;
}
Jinhao 2004-10-12
  • 打赏
  • 举报
回复
template<typename T>
class A
{
bool flag_;
public:
A(bool flag):flag_(flag)
{}

bool operator()(const T& rhs) const
{
return flag_==rhs.getflag();
}
};

class PclElem
{
bool flag;
public:

bool getflag()const
{ return flag; }
};

int main()
{
list<PclElem> lst;
for(int i=0; i<10; i++)
lst.push_back(PclElem());
cout<<lst.size()<<endl; //输出10
lst.remove_if(A<PclElem>(false));
cout<<lst.size()<<endl; //输出0, 因为stl对所有内建类型的都是以0来初始化,所以10个PclElem实例对象的flag都是false
}
学习并掌握C++2.0(11+14+17+20)的新特性,学习线程及线程池的应用 ---------------------------------------------------给小白学员的3年学习路径及计划技术方面分三块:1.纯开发技术方向2.音视频流媒体专业方向3.项目实战---------------------------------------------------1.纯开发技术方向(1) C++必须要过硬(至少学会10本经典好书)(2) 系统级编程(Windows、Linux),必须特别熟练系统API,灵活运用(3) 框架与工具(Qt、MFC):必须精通其中一种。(4) 架构与设计模式:需要提升一个高度,不再是简单的编码,而是思维模式。(5) 驱动级别(如果有兴趣,可以深入到驱动级:包括Windows、Linux)(6) 最好学习点Java+Html+javascript等WEB技术。2.音视频流媒体专业方向(1) 音视频流媒体基础理论:   必须认真学会,否则看代码就是看天书(2) 编解码方向:精通h.264,h.265(hevc), 包括理论和各个开源库(ffmpeg,libx264,libx265,...)。(3) 直播方向:  精通各种直播协议(rtsp,rtmp,hls,http-flv,...), 钻研各个开源库(live555,darwin,srs,zlmediakit,crtmpserver,...)(4) 视频监控:  理论+开源库(onvif+281818)(EasyMonitor、iSpy、ZoneMinder(web)、...) 3.项目实战(1) Qt项目:  至少要亲手练习10个实战项目(网络服务器、多线程、数据库、图像处理、多人聊天、等等)(2)音视频项目:包括编解码、视频监控、直播等各个方向,都需要亲手实战项目,包括视频服务器、后台管理系统、前端播放器(多端)---------------------------------------------------  第1章 C++11新特性 41). nullptr关键字与新语法 42). auto和decltype类型推导 6 auto讲解 6 auto示例 7 decltype 83). for区间迭代 94). 初始化列表 105). 模板增强 11外部模板 11类型别名模板 12默认模板参数 126). 构造函数 13委托构造 13继承构造 147). Lambda 表达式 158). 新增容器 20std::array 20std::forward_list 21无序容器 22元组 std::tuple 239). 正则表达式 2610). 语言级线程支持 28多线程库简介 2811). 右值引用和move语义 31右值引用和move语义 32转移左值 3412). constexpr 35第2章 C++14新特性 36Lambda 函数 36类型推导 37返回值类型推导(Return type deduction) 37泛型lambda 39[[弃用的]]  [[deprecated]]属性 40二进制数字和数字分隔符 41第3章 C++17新特性 42安装GCC10.2 42安装msys2-x86_64-20200720 42更新镜像 42更新软件库 43安装 MinGW64 等必要的软件 43环境变量Path 43编译命令 43constexpr 44typename 45折叠表达式 47结构化绑定 48条件分支语句初始化 49聚合初始化 50嵌套命名空间 52lambda表达式捕获*this的值 53改写/继承构造函数 54用auto作为非类型模板参数 55__has_include 56fallthrough 57nodiscard 57maybe_unused 58第4章 C++20新特性 59编译命令 59concept 59typename 60explicit 61constinit 62位域变量的默认成员初始化 62指定初始化 63基于范围的for循环初始化 64放宽基于范围的for循环,新增自定义范围方法 65嵌套内联命名空间 66允许用圆括弧的值进行聚合初始化 67unicode字符串字面量 68允许转换成未知边界的数组 68likely和unlikely 69第5章 C++2.0(11/14/17/20)总结与分析 705.1 C语言与C++ 715.2 语言可用性的强化 725.2.1 常量 725.2.2 变量及其初始化 735.2.3 类型推导 745.2.4 控制流 765.2.5 模板 775.2.6 面向对象 815.3 语言运行期的强化 835.3.1 Lambda 表达式 835.3.2 右值引用 865.4 容器 885.4.1 线性容器 885.4.2 无序容器 895.4.3 元组 895.5 智能指针与内存管理 905.5.1 RAII 与引用计数 905.5.2 std::shared_ptr 905.5.3 std::unique_ptr 915.5.4 std::weak_ptr 91第6章 C++2.0多线程原理与实战 93什么是并发 93并发的方式 93为什么使用并发 95线程简介 96创建线程的三种方式 971. 通过函数 972.通过类对象创建线程 993.通过lambda表达式创建线程 101thread线程的使用 101互斥量与临界区 105期物Future 111条件变量 112原子操作 114内存模型 118第7章 C++2.0线程池原理与实战 120线程与线程池的基本原理 1201)、线程 1202)、线程的生命周期 1213)、什么是单线程和多线程 1214)、线程池 1225)、四种常见的线程池 123线程池的架构与流程 123线程池代码实战 125    

24,854

社区成员

发帖
与我相关
我的任务
社区描述
C/C++ 工具平台和程序库
社区管理员
  • 工具平台和程序库社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧