设为首页 加入收藏

TOP

C++11 lambda表达式(二)
2013-01-01 14:48:16 来源: 作者: 【 】 浏览:997
Tags:11  lambda 表达式

  // move构造函数

  CMyString(CMyString &&s)

  {

  cout 《 "CMyString(CMyString &&s)" 《 endl;

  m_pData = s.m_pData;

  s.m_pData = NULL;

  }

  // 析构函数

  ~CMyString()

  {

  cout 《 "~CMyString()" 《 endl;

  delete [] m_pData;

  m_pData = NULL;

  }

  // 拷贝赋值函数

  CMyString &operator =(const CMyString &s)

  {

  cout 《 "CMyString &operator =(const CMyString &s)" 《 endl;

  if (this != &s)

  {

  delete [] m_pData;

  m_pData = new char[strlen(s.m_pData)+1];

  strcpy(m_pData, s.m_pData);

  }

  return *this;

  }

  // move赋值函数

  CMyString &operator =(CMyString &&s)

  {

  cout 《 "CMyString &operator =(CMyString &&s)" 《 endl;

  if (this != &s)

  {

  delete [] m_pData;

  m_pData = s.m_pData;

  s.m_pData = NULL;

  }

  return *this;

  }

  private:

  char *m_pData;

  };

  复制代码

  可以看到,上面我们添加了move版本的构造函数和赋值函数。那么,添加了move版本后,对类的自动生成规则有什么影响呢?唯一的影响就是,如果提供了move版本的构造函数,则不会生成默认的构造函数。另外,编译器永远不会自动生成move版本的构造函数和赋值函数,它们需要你手动显式地添加。

  当添加了move版本的构造函数和赋值函数的重载形式后,某一个函数调用应当使用哪一个重载版本呢?下面是按照判决的优先级列出的3条规则:

  1、常量值只能绑定到常量引用上,不能绑定到非常量引用上。

  2、左值优先绑定到左值引用上,右值优先绑定到右值引用上。

  3、非常量值优先绑定到非常量引用上。

  当给构造函数或赋值函数传入一个非常量右值时,依据上面给出的判决规则,可以得出会调用move版本的构造函数或赋值函数。而在move版本的构造函数或赋值函数内部,都是直接"移动"了其内部数据的指针(因为它是非常量右值,是一个临时对象,移动了其内部数据的指针不会导致任何问题,它马上就要被销毁了,我们只是重复利用了其内存),这样就省去了拷贝数据的大量开销。

  一个需要注意的地方是,拷贝构造函数可以通过直接调用*this = s来实现,但move构造函数却不能。这是因为在move构造函数中,s虽然是一个非常量右值引用,但其本身却是一个左值(是持久对象,可以对其取地址),因此调用*this = s时,会使用拷贝赋值函数而不是move赋值函数,而这已与move构造函数的语义不相符。要使语义正确,我们需要将左值绑定到非常量右值引用上,C++(www.cppentry.com) 11提供了move函数来实现这种转换,因此我们可以修改为*this = move(s),这样move构造函数就会调用move赋值函数。

  C++(www.cppentry.com)的Lambda表达式在WIN RT的异步编程(www.cppentry.com)中,占有非常重要的作用。但C++(www.cppentry.com)的Lambda表达式又不同于其他语言,比如C#,javascript.本篇旨在讨论C++(www.cppentry.com) Lambda表达式的基本语法和概念,希望大家多多指正。

  首先,我们看一下Lambda表达式的基本构成

   1. 是捕获值列表,2.是传入参数列表,3.可修改标示符,4.错误抛出标示符,5.函数返回值,6.是函数体。

  在。NET 中,我们认为比较标准的Lambda表达式应该是这个样子

  // declaring_lambda_expressions1.cpp

  #include <functional>

  int main()

  {

  // Assign the lambda expression that adds two numbers to an auto variable.

  auto f1 = [] (int x, int y) { return x + y; };

  // Assign the same lambda expression to a function object.

  function<int (int, int)> f2 = [] (int x, int y) { return x + y; };

  f1(3,4);

  复制代码

  }

  f1是一个auto的值,也是function<>这个模板类型,我们可以理解成为一个函数指针。然后我们用f1(3,4)去调用他。

  如果我们想在函数声明的时候就直接执行他,我们可以在Lambda表达式的最后加传入参数,像这样。

  int main()

  {

  using namespace std;

  int n = [] (int x, int y) { return x + y; }(5, 4);

  //assign the return type

  int n = [] (int x, int y) -> int{ return x + y;}(5, 4);

  cout 《 n 《 endl;

  复制代码

  }

  第二个表达式中声明的返回值必须跟随->符号,并且两个必须同时出现。如果返回值唯一的话,我们可以省略->+返回值类型。

  Lambda表达式允许返回值不唯一的情况,但必须指定返回值类型。

  在以上的例子当中,只是常规的Lambda表达式用法,下面我们要说一说捕获值列表。

  捕获值列表,是允许我们在Lambda表达式的函数体中直接使用这些值,捕获值列表能捕获的值是所有在此作用域可以访问的值,包括这个作用域里面的临时变量,类的可访问成员,全局变量。捕获值的方式分两种,一种是按值捕获,一种是按引用捕获。顾名思义,按值捕获是不改变原有变量的值,按引用捕获是可以在Lambda表达式中改变原有变量的值。

  [&] 所有的值都是按引用捕获

  [=] 所有的值都是按值捕获

  如果你不想某些值被按引用或者按值捕获,但其他的值却想那样做的话

  [ &, n ] 除了n 所有的值按引用捕获

  [ = , &n ]除了n所有的值按值捕获

  当然,我们也可以指定某几个值的捕获属性

  [ m, n ]m,n按引用捕获

  [ &m, &n ]m,n按值捕获

  int m = 0, n = 0;

  [=] (int a) mutable { m = ++n + a; }(4);

  [&] (int a) { m = ++n + a; }(4);

  [=,&m] (int a) mutable { m = ++n + a; }(4);

  [&,m] (int a) mutable { m = ++n + a; }(4);

  [m,n] (int a) mutable { m = ++n + a; }(4);

  [&m,&n] (int a) { m = ++n + a; }(4);

  [=] (int a) mutable { m = ++n + a; }(4);

  复制代码

  大家一定好奇为什么这里有很多mutable.在按值引用的情况下,Lambda函数体内部是不能直接修改引用值的。如下面注释代码,是会报错的。这种情况下,我们要在Lambda表达式前加mutable,但是结果m,n 依然没有被修改,维持按值引用的特性。

  int main()

  {

  int m = 0, n = 0;

  // 不加mutable会报错

  //[=] (int a){ m = ++n + a; }(4);

  //[m,n] (int a){ m = ++n + a; }(4);

  [=] (int a) mutable { m = ++n + a; }(4);

  //

  // [=] (int m, int n, int a){m=++n+a; }(m, n, 4);

  // 下面这个函数m,n的值依然会被修改,因为m,n是按引用传入的

  // [=] (int &m, int &n, int a){m=++n+a; }(m, n, 4);

  cout 《 m 《 endl 《 n 《 endl;

  复制代码

  }

  在这个例子中捕获值列表[this]中的this是用来指向这个类的,但[this]只有在类的内部,或者是this指针存在的情况下才能使用。

  class Scale

  {

  public:

  // The constructor.

  explicit Scale(int scale)

  : _scale(scale)

  {

  }

  // Prints the product of each element in a vector object

  // and the scale value to the console.

  void ApplyScale(const vector<int>& v) const

  {

  for_each(v.begin(), v.end(),

  [this](int n) { cout 《 n * _scale 《 endl; });

  }

  private:

  int _scale;

  };

  

        

首页 上一页 1 2 3 4 下一页 尾页 2/4/4
】【打印繁体】【投稿】【收藏】 【推荐】【举报】【评论】 【关闭】 【返回顶部
分享到: 
上一篇c++简单线程封装 下一篇c++中查询硬件与系统信息api

评论

帐  号: 密码: (新用户注册)
验 证 码:
表  情:
内  容: