C++17中if和switch语句的新特性

      1.从C++17开始,if语句允许在条件表达式里添加一条初始化语句。当仅在if语句范围内需要变量时,使用这种形式的if语句。在if语句的条件表达式里定义的变量将在整个if语句中有效,包括else部分

std::mutex mx;
bool shared_flag = true; // guarded by mx
constexpr int get_value() { return 66; }

int test_if_init()
{
	std::map<std::string, std::string> addrs{
		{"csdn", "https://blog.csdn.net/fengbingchun/"},
		{"github", "https://github.com/fengbingchun"}
	};

	if (auto it = addrs.find("github"); it != addrs.end())
		std::cout << "github:" << it->second << "\n"; // github:https://github.com/fengbingchun

	if (auto it = addrs.find("csdn"); it == addrs.end())
		std::cout << "no find\n";
	else // if中的it变量在else分支中也有效
		std::cout << "csdn:" << it->second << "\n"; // csdn:https://blog.csdn.net/fengbingchun/

	if (auto [it, inserted] = addrs.insert({ "gitee", "https://gitee.com/fengbingchun/test.git" }); !inserted)
		std::cout << "already exists\n";
	else
		std::cout << "inserted successfully: key:" << it->first << ", value:" << it->second << "\n"; // inserted successfully: key:gitee, value:https://gitee.com/fengbingchun/test.git

	if (auto x = get_value(); x == 66)
		std::cout << "x is:" << x << "\n"; // x is:66

	if (std::lock_guard<std::mutex> lock(mx); shared_flag) {
		std::cout << "setting shared_flag to false\n"; // setting shared_flag to false
		shared_flag = false;
	}

	if (auto val1 = addrs.cbegin()->first, val2 = addrs.crbegin()->first; val1 != val2)
		std::cout << "val1:" << val1 << ", val2:" << val2 << "\n"; // val1:csdn, val2:github

	const std::string str{ "if" };
	if (auto keywords = { "if", "for", "while" }; std::any_of(keywords.begin(), keywords.end(), [&str](const char* kw) { return str == kw; }))
		std::cout << "Error:Token must not be a keyword\n"; // Error:Token must not be a keyword

	return 0;
}

      2.从C++17开始,switch语句允许在条件表达式里添加一条初始化语句,其范围仅限于switch语句块。通过使用带初始化的switch语句,我们可以在对条件表达式求值之前初始化一个对象/实体,用法与以上的if相同。

int test_switch_init()
{
	std::random_device rd;
	std::mt19937 mt(rd());
	std::uniform_int_distribution<int> dist(0, 100);

	switch (auto val = dist(mt); val) {
		default:
			std::cout << "val:" << val << "\n"; // val:20
	}
	
	return 0;
}

      3.从C++17开始,你可以在函数模版中使用if constexpr语句做出编译时分支决策,而无需使用(resort)多个函数重载.
      if constexpr语句在编译时求值,编译器仅生成与发送到函数模板的参数类型相匹配的if分支的代码。该功能主要用在模版中,它允许仅编译特定的语句,具体取决于模版类型。
      注意:
      (1).if constexpr和if的唯一区别是:if constexpr在编译时进行判断,而if在运行时进行判断;所以,使用if constexpr的代码在编译完成后,程序的这一部分其实就不会有分支存在。
      (2).通过使用语法if constexpr,编译器可以计算编译期的条件表达式,在编译期决定使用哪部分,其余部分的代码将会被丢弃,但会进行语法检查。所有的static_assert也必须有效,即使所在的分支没有被编译。
      (3).不能在函数体之外使用if constexpr.
      (4).if constexpr不支持短路求值(当&&左侧为false时停止求值,当||左侧为true时停止求值)。
      (5).if constexpr可以在任何函数中使用,而并非仅限于模版。只要条件表达式是编译期的,并且可以转换成bool类型。
      (6).在泛型代码之外使用if constexpr的唯一好处是被丢弃的部分不会成为最终程序的一部分,这将减小生成的可执行程序的大小。

template<typename T>
auto show(T t)
{
	//if (std::is_pointer_v<T>) // show(a) results in compiler error for return *t. show(p) results in compiler error for return t.
	if constexpr (std::is_pointer_v<T>) // this statement goes away for show(a)
		return *t;
	else
		return t;
}

template<typename T>
void print_value(const T& value)
{
	if constexpr (std::is_same_v<T, std::string>)
		std::cout << "type: std::string: value: " << value << ", length: " << value.length() << "\n";
	else if constexpr (std::is_same_v<T, int>)
		std::cout << "type: int: value: " << value << "\n";
	else if constexpr (std::is_same_v<T, float>)
		std::cout << "type: float: value: " << value << "\n";
	else
		std::cout << "unsupported type\n";
}

int test_if_constexpr()
{
	int a = 66;
	int* p = &a;

	std::cout << show(a) << "\n"; // 66
	std::cout << show(p) << "\n"; // 66

	std::string str{ "hello" };
	print_value(str); // type: std::string: value: hello, length: 5
	print_value(a); // type: int: value: 66
	float val{.6f };
	print_value(val); // type: float: value: 0.6
	print_value(p); // unsupported type

	return 0;
}

      执行结果如下图所示:

      GitHubhttps://github.com/fengbingchun/Messy_Test

  • 8
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值