【C++篇】C++11新特性总结1
目录
1,C++11的发展历史
2,列表初始化
2.1C++98传统的{}
2.2,C++11中的{}
2.3,C++11中的std::initializer_list
3,右值引用和移动语义
3.1,左值和右值
3.2,左值引用和右值引用
3.3,引用延长生命周期
3.4,左值和右值的参数匹配
3.5,移动语义
为什么需要移动语义?
移动构造函数与移动赋值运算符
使用场景
常见问题与缺陷
代码示例:自定义类的移动语义
3.6,右值引用和移动语义在传参中的提效
3.7,引用折叠
3.8,完美转发
1,C++11的发展历史
C++11是C++的第二个主要版本,并且是从C++98起的最重要更新。C++11是C++编程语言的一个重要版本,于2011年正式发布。它引入了许多新特性和改进,极大地增强了 C++ 的功能和易用性。下面介绍它的一些主要特性:
2,列表初始化
2.1C++98传统的{}
在C++98中一般数组和结构体支持使用{}初始化。
struct point
{
int x;
int y;
};
int main()
{
int arr1[5] = {1,2,3,4,5};
point p = { 1,2 };
return 0;
}
2.2,C++11中的{}
struct point
{
int x;
int y;
};class Date
{
public:
Date(int year = 1, int month = 1, int day = 1)
:_year(year)
, _month(month)
, _day(day)
{
cout << "Date(int year int month , int day)" << endl;
}
Date(const Date& d)
:_year(d._year)
,_month(d._month)
,_day(d._day)
{
cout << "Date(const Date& d)" << endl;
}
private:
int _year;
int _month;
int _day;
};
- C++11后想统一初始化的方式,试图一切对象皆可使用{}初始化,{}初始化也叫列表初始化。
- 内置类型支持,自定义类型也支持,自定义类型本质是类型转换,中间会产生临时对象,经过编译器优化后变成直接构造。
//C++11
//内置类型支持{}初始化
int x = { 2 };//自定义类型
//本质是{2025,1,1}构造出临时对象,再拷贝给d1,但是编译器优化为直接用{2025,1,1}构造d1
Date d1 = { 2025,1,1 };//这里的d2引用的是{2024,7,2}的临时对象
const Date& d2 = { 2024,7,2 };//需要注意的是C++98支持单参数时类型转换,也可以不用加{}
Date d3 = { 2025 };
Date d4 = 2025;
- {}初始化可以省略=
//可以省略掉=
int x2{ 2 };
point p1{ 1,2 };
Date d6{ 2025,1,2 };
const Date& d7{ 2024,8,15 };
- C++11的列表初始化在许多场景下会带来不少的便利,如容器push/insert多参数构造的对象时,用{}会很方便。
vector<Date> v;
//有名对象传参
v.push_back(d6);//匿名对象传参
v.push_back(Date(2025, 1, 2));//比起有名对象和匿名对象,{}初始化更有性价比
v.push_back({ 2025,1,2 });
2.3,C++11中的std::initializer_list
- 上面的初始化已经很方便,但是对于一个容器的初始化来说,还是不太方便。比如一个vector对象,我们想用N个值去初始化,那么我们需要实现多个构造函数才能支持:vector<int> v1={1,2,3},vector<int> v2={1,2,3,4},vector<int> v3={1,2,3,4,5};
- C++11库中提供了一个std::initializer_list ,这个类的本质是底层开一个数组,将数据拷贝过来,std::initializer_list 中有两个指针分别指向数组的开始和结束。
- 这时只要我们的容器支持一个std::initializer_list的构造函数,就可以支持多个值的{x1,x2,x3......}的初始化。STL中的 容器支持多个值构成的{x1,x2,x3,......}的初始化,就是通过底层支持std::initiaalizer_list的构造实现的。如下图list和vector的构造函数中都增添了支持std::initializer_list的构造函数。


vector<int> v1 = { 1,2,3,4,5 };
vector<int> v2 = { 1,2,3,4,5,6 };//这里pair对象的{}初始化和map的initializer_list构造结合到一起了
map<string, string> dict = { {"sort","排序 "},{"string","字符串"} };
3,右值引用和移动语义
C++98中就有引用的语法,而C++11中新增了右值引用的语法特性,之前的引用叫做左值引用。无论左值引用还是右值引用,都可以理解为是在给变量取别名。
3.1,左值和右值
- 左值是一个表示数据的表达式(如变量名或解引用 的指针),一般是持久状态,存储在内存中,我们可以获取它的地址。左值可以出现在赋值符号的左边,也可以是在右边。如果左值用const修饰,就不能给它赋值,但可以取它的地址。
- 右值也是一个表示数据的表达式,要么是常量或者是临时对象等,右值可以出现在赋值符号的右边,但不能出现在左边,右值不能取地址。
- 左值的英文简写为lvalue,右值的英文简写为rvalue。传统认为它们分别是left value、right value 的缩写。现代C++中,lvalue被解释为loactor value的缩写,可意为存储在内存中、有明确存储地址可以取地址的对象,而rvalue被解释为read value,指的是那些可以提供数据值,但是不可以寻址,例如:临时变量,常量,存储于寄存器中的变量等,也就是说左值和右值的核心区别就是能否取地址。
//左值,可以取地址
//以下均为左值
int* p = new int(0);
int b = 1;
const int c = b;
*p = 10;
string s("1111111");
s[0] = 'x';double x = 1.1, y = 2.2;
//右值,不能取地址
//以下几个均为右值
10;
x + y;
string("111111");
3.2,左值引用和右值引用
- Type& r1=x,Type&& r2=y。其中第一个语句就是左值引用 ,本质是给左值取别名。同理第二个语句就是给右值引用,本质是给右值取别名。
- 左值引用不能直接引用 右值,需加上const修饰。
//左值,可以取地址
//以下均为左值
int* p = new int(0);
int b = 1;
const int c = b;
*p = 10;
string s("1111111");
s[0] = 'x';double x = 1.1, y = 2.2;
//右值,不能取地址
//以下几个均为右值
10;
x + y;
string("111111");
//左值引用,给左值取别名
int& r1 = b;
int*& r2 = p;
int& r3 = *p;
string& r4 = s;
char& r5 = s[0];
//左值引用不能直接引用右值,需加上cosnt
const int& rx1 = 10;
const double& rx2 = x + y;
const string& rx3 = string("111111");
- 右值引用不能直接引用 左值,但可以引用move(左值)。
//左值,可以取地址
//以下均为左值
int* p = new int(0);
int b = 1;
const int c = b;
*p = 10;
string s("1111111");
s[0] = 'x';double x = 1.1, y = 2.2;
//右值,不能取地址
//以下几个均为右值
10;
x + y;
string("111111");
//右值引用
int&& rr1 = 10;
double&& rr2 = x + y;
string&& rr3 = string("111111");
//右值引用不能直接引用 左值,但可以引用 move(左值)
int&& rrx1 = move(b);
int*&& rrx2 = move(p);
int&& rrx3 = move(*p);
string&& rrx4 = (move)(s);
string&& rrx5 = (string&&)s;
- move是库里面的一个函数模板,本质内部做了强制类型转换,涉及到一些引用折叠的知识

- 需要注意的是,变量表达式都是左值属性,也就意味着一个右值被右值引用绑定后,右值引用变量是一个左值。
//右值引用
int&& rr1 = 10;
double&& rr2 = x + y;
string&& rr3 = string("111111");
//这里要注意的是,rr1的属性是左值,要想被右值引用绑定,除非move一下
int&& a = move(rr1);
3.3,引用延长生命周期
右值引用可用于为临时对象延长生命周期,const的左值引用也能延长临时对象生存期,但这些对象无法被修改。
string s1 = "Test";
//s1+s1生成临时对象
const string& s2 = s1 + s1; //const左值引用延长生命周期
string&& s3 = s1 + s1; //右值引用延长生命周期s3 += "Test";
cout << s3 <<endl;
3.4,左值和右值的参数匹配
- C++98中,在函数的形参部分,我们会用const 修饰左值引用的方式,这样实参在传递左值和右值时都可以匹配。
- C++11后,分别重载左值引用,const左值引用和右值引用作为形参的f函数,那么实参时左值,会调用f(左值引用),实参是const 左值引用时,会调用f(const 左值引用),实参是右值引用时,会调用f(右值引用)。
void f(int& x)
{
cout << "f(int& x)" << endl;
}void f(const int& x)
{
cout << "f(const int& x)" << endl;
}void f(int&& x)
{
cout << "f(int&& x)" << endl;
}int main()
{int x = 1;
const int y = x;
f(x);//调用f(int& x)
f(y);//调用f(const int& x)
f(3);//调用f(int&& x)
f(move(x));//调用f(int&& x)
//右值引用变量是左值属性的
int&& z = 1;
f(z); //调用f(int& x)
f(move(z));//调用f(int&& x)return 0;
}
3.5,移动语义
移动语义是现代编程语言(如C++11及更高版本、Rust)中用于优化资源管理的重要机制。其核心目标是避免不必要的拷贝,通过转移资源所有权(而非复制)提升程序性能,尤其在处理动态内存、文件句柄等资源时效果显著。
为什么需要移动语义?
- 传统拷贝的缺陷:
对于包含动态内存或系统资源的对象(如std::vector,std::string等),拷贝构造函数会深度复制所有数据,导致性能开销。
- 移动语义的优化:
直接窃取临时对象(如右值)的资源,避免复制。
移动构造函数与移动赋值运算符
移动构造函数:接受右值引用参数,转移资源。
class MyClass {
public:// 移动构造函数MyClass(MyClass&& other) noexcept : data_(other.data_) ,size_(other.size_) {other.data_ = nullptr; // 置空原对象,避免重复释放other.size_ = 0;}
private:int* data_;size_t size_;
};
移动赋值运算符:类似移动构造,处理对象赋值。
MyClass& operator=(MyClass&& other) noexcept {if (this != &other) {delete[] data_; // 释放当前资源data_ = other.data_; // 转移资源size_ = other.size_;other.data_ = nullptr;other.size_ = 0;}return *this;
}
使用场景
1,显示触发移动语义
使用std:move将左值转化为右值:
std::vector<int> a = {1, 2, 3};
std::vector<int> b = std::move(a); // a变为空,资源转移给b
注意:被移动后的对象处于有效但未定义状态(通常为空),不可再使用其值。
2,返回值优化
编译器自动优化函数返回的临时对象,避免拷贝:
std::vector<int> createVector() {std::vector<int> v = {1, 2, 3};return v; // 编译器可能直接构造v到调用方,无需移动或拷贝
}
3,STL容器的移动支持
标准库容器(如std::vector、std::string)已实现移动语义:
std::vector<std::string> vec;
std::string s = "data";
vec.push_back(std::move(s)); // 移动s到容器,避免拷贝字符串内容
常见问题与缺陷
1,误用std:move
对局部变量过早移动,导致后续访问未定义
std::vector<int> a = {1, 2, 3};
std::vector<int> b = std::move(a);
std::cout << a.size(); // 未定义行为,a可能为空
2,未实现移动语义的类
-
若类未定义移动操作,编译器可能回退到拷贝(即使使用
std::move)。 -
规则:若用户定义了拷贝构造函数、拷贝赋值或析构函数,编译器不会自动生成移动操作。
3,异常安全
移动操作应标记为noexcept,否则某些容器(如std::vector)可能仍选择拷贝。
代码示例:自定义类的移动语义
#include <iostream>
#include <utility> // for std::moveclass Buffer {
public:Buffer(size_t size) : size_(size), data_(new int[size]) {}// 移动构造函数Buffer(Buffer&& other) noexcept : size_(other.size_), data_(other.data_) {other.size_ = 0;other.data_ = nullptr;}// 移动赋值运算符Buffer& operator=(Buffer&& other) noexcept {if (this != &other) {delete[] data_;data_ = other.data_;size_ = other.size_;other.data_ = nullptr;other.size_ = 0;}return *this;}~Buffer() { delete[] data_; }private:size_t size_;int* data_;
};int main() {Buffer a(100);Buffer b = std::move(a); // 调用移动构造函数Buffer c(200);c = std::move(b); // 调用移动赋值运算符return 0;
}
3.6,右值引用和移动语义在传参中的提效
C++11以后,STL容器的push和insert接口增加了右值引用版本。
以vector容器的push_back为例:

当实参是一个左值时,继续调用拷贝构造进行拷贝;当实参是一个右值时,容器内部调用移动构造,提高效率。
3.7,引用折叠
- C++中不能直接定义引用的引用 ,如int& && r1=i;这样写会直接报错,通过模板或typedef的类型操作可以构成引用的引用。
- 通过模板或typedef中的类型操作构成引用的引用时,这时C++11给出了一个引用折叠的规则:右值引用的右值引用折叠成右值引用,所有其他组合均折叠成左值引用。
int main()
{
typedef int& lref;
typedef int&& rref;
int n = 0;
lref& r1 = n; //r1的类型是int&
lref&& r2 = n; //r2的类型是int&
rref& r3 = n; //r3的类型是int&
rref&& r4 = 3; //r4的类型是int&&
return 0;
}
- Function(T&& t)函数模板程序中,假设实参是int右值,模板参数T推导出是int;实参是int左值,模板参数T推导出是int&,再结合引用折叠规则,就实现了实参是左值,实例化出左值引用版本形参的 Function,实参是右值,实例化出右值引用版本形参的Function。
//根据引用折叠规则,f1实例化后总是一个左值引用
template <class T>
void f1(T& x)
{}
//根据引用折叠规则,f2实例化后可以是一个左值引用,也可以是一个右值引用
template <class T>
void f2(T&& x)
{}
int main()
{//折叠->实例化为 f1(int& x)
f1<int>(n);
//f1(0); //报错
//折叠->实例化为 f1(int& x)
f1<int&>(n);
//f1<int&>(0); //报错
//折叠->实例化为 f1(int& x)
f1<int&&>(n);
//f1<int&&>(0); //报错
//折叠->实例化为 f1(const int& x)
f1<const int&>(n);
f1<const int&>(0);
//折叠->实例化为 f1(const int& x)
f1<const int&&>(n);
f1<const int&&>(0);
//没有折叠->实例化为 f2(int&& x)
//f2<int>(n); //报错
f2<int>(0);
//折叠->实例化为 f2(int& x)
f2<int&>(n);
//f2<int&>(0); //报错
//折叠->实例化为 f2(int&& x)
//f2<int&&>(n); //报错
f2<int&&>(0);
return 0;
}
- 像f2这样的函数模板中,(T&& x)参数看起来是右值引用参数,但是由于引用折叠的规则,他传递左值时就是左值引用,传递右值时就是右值引用,有些地方也把这种函数模板的参数叫做万能引用。
3.8,完美转发
- Function(T&& t)函数模板程序中,传左值实例化以后就是左值引用的Function函数,传右值实例化以后就是右值引用的Function函数。
- 但是我们在前面讲过,一个变量表达式都是左值属性,也就是一个右值被右值引用表达式绑定以后,右值引用变量表达式是左值属性的。也就是说Function函数中的 t 是左值属性的。如果Function函数中调用一个func函数,那么我们把t传给下一层函数func,那么匹配的都是左值引用的func函数。这里想要保持t对象的原属性,就需要使用完美转发实现。

std::forward:用于在转发参数时保持其原始的类别 。- 完美转发forwrad本质是一个函数模板,它主要还是通过引用折叠的方式实现。
void func(int& x)
{
cout << "左值引用" << endl;
}
void func(const int& x)
{
cout << "const 左值引用" << endl;
}
void func(int&& x)
{
cout << "右值引用" << endl;
}
void func(const int&& x)
{
cout << "const 右值引用" << endl;
}
template <class T>
void Function(T&& t)
{
func(t);
}
int main()
{
Function(10); //10是右值 Function(int&& t) 右值
int a;、
Function(a); //a是左值 Function(int& t) 左值Function(move(a)); // Function(int&& t) 右值
const int b = 8;
Function(b); //Function(const int& t) const左值
Function(move(b)); //Function(const int&& t) const 右值
return 0;
}
上面的代码是没有使用 完美转发的场景,运行结果如下:
可以看出,在将t传给下一层的func函数时,匹配的都是左值引用。
经过完美转发后的代码和运行结果:
template <class T>
void Function(T&& t)
{
//func(t);
func(forward<T>(t));
}

相关文章:
【C++篇】C++11新特性总结1
目录 1,C11的发展历史 2,列表初始化 2.1C98传统的{} 2.2,C11中的{} 2.3,C11中的std::initializer_list 3,右值引用和移动语义 3.1,左值和右值 3.2,左值引用和右值引用 3.3,…...
【Nginx + Keepalived 实现高可用的负载均衡架构】
使用 Nginx Keepalived 可以实现高可用的负载均衡架构,确保在某个 Nginx 节点故障时,自动将流量转移到备用节点。以下是详细的实现步骤: 1. 架构概述 Nginx:作为负载均衡器,将流量分发到后端服务器。Keepalived&…...
使用外骨骼灵活远程控制协作机器人案例
外骨骼控制器采用可调节结构,简化了机器人编程,使协作机器人 FR3 的远程控制变得容易。 一、引言 在开发机器人手臂或双臂系统的应用程序时,经常会遇到以下挑战: 1. 使用拖动和示教进行定位的困难:拖动和示教功能通常…...
Centos Stream 10 根目录下的文件夹结构
/ ├── bin -> usr/bin ├── boot ├── dev ├── etc ├── home ├── lib -> usr/lib ├── lib64 -> usr/lib64 ├── lostfound ├── media ├── mnt ├── opt ├── proc ├── root ├── run ├── sbin -> usr/sbin ├── srv ├─…...
python连点器
要实现一个用于抖音点赞的鼠标连点工具,可以通过编程或现有软件实现。以下是两种常见方法(但请注意:频繁自动化操作可能违反平台规则,需谨慎使用): 方法 1:使用现成工具(如 AutoClic…...
STM32G474--Whetstone程序移植(单精度)笔记
1 准备基本工程代码 参考这篇笔记从我的仓库中选择合适的基本工程,进行程序移植。这里我用的是stm32g474的基本工程。 使用git clone一个指定文件或者目录 2 移植程序 2.1 修改Whetstone.c 主要修改原本变量定义的类型,以及函数接口全部更换为单精度…...
Spring Boot 3.4 中 MockMvcTester 的新特性解析
引言 在 Spring Boot 3.4 版本中,引入了一个全新的 MockMvcTester 类,使 MockMvc 测试可以直接支持 AssertJ 断言。本文将深入探讨这一新特性,分析它如何优化 MockMvc 测试并提升测试的可读性。 Spring MVC 示例 为了演示 MockMvcTester 的…...
java 读取sq3所有表数据到objectNode
1.实现效果:将sq3中所有表的所有字段读到objectNode 对象中,兼容后期表字段增删情况,数据组织形式如下图所示: 代码截图: 代码如下: package com.xxx.check.util;import java.sql.*; import java.util.Arr…...
网络计算机的五个组成部分
单个计算机是无法进行通信的。所以需要借助网络。 下面介绍一些在网络里常见的设备。 一、服务器 服务器是在网络环境中提供计算能力并运行软件应用程序的特定IT设备 它在网络中为其他客户机(如个人计算机、智能手机、ATM机等终端设备)提供计算或者应用…...
jakarta EE学习笔记-个人笔记
WebServlet注解:声明一个类为Servlet Target({ElementType.TYPE}) Retention(RetentionPolicy.RUNTIME) Documented public interface WebServlet {// 指定Servlet的影子String name() default ""; // 匹配地址映射(URL)String[] value() default {};// …...
vue3-响应式 toRefs
在Vue 3中,toRefs是一个非常有用的组合式API(Composition API)函数,它主要用于将响应式对象(通常是由reactive创建的对象)转换为单独的响应式引用对象。这样做的好处是可以将这些响应式引用解构出来&#x…...
学习threejs,使用Lensflare模拟镜头眩光
👨⚕️ 主页: gis分享者 👨⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨⚕️ 收录于专栏:threejs gis工程师 文章目录 一、🍀前言1.1 ☘️THREE.Lensflare 二、&…...
redis高级数据结构布隆过滤器
文章目录 背景什么是布隆过滤器Redis 中的布隆过滤器布隆过滤器使用注意事项实现原理空间占用估计 背景 我们在使用新闻客户端看新闻时,它会给我们不停地推荐新的内容,它每次推荐时要去重,去掉那些已经看过的内容。问题来了,新闻…...
mysql 5.7安装
基础环境:centos7.9 创建日志存放目录 mkdir -p /opt/supervisor/log安装相关工具 yum install -y perl net-tools numactl gcc python-devel配置yum源 sudo vim /etc/yum.repos.d/mysql-community.repo [mysql-connectors-community] nameMySQL Connectors Com…...
Golang:精通sync/atomic 包的Atomic 操作
在本指南中,我们将探索sync/atomic包的细节,展示如何编写更安全、更高效的并发代码。无论你是经验丰富的Gopher还是刚刚起步,你都会发现有价值的见解来提升Go编程技能。让我们一起开启原子运算的力量吧! 理解Go中的原子操作 在快…...
微信小程序如何使用decimal计算金额
第三方库地址:GitHub - MikeMcl/decimal.js: An arbitrary-precision Decimal type for JavaScript 之前都是api接口走后端计算,偶尔发现这个库也不错,计算简单,目前发现比较准确 上代码 导入js import Decimal from ../../uti…...
最新Modular公司之MAX和Mojo作者 克里斯·拉特纳简介
Chris Lattner(克里斯拉特纳) 是一位著名的计算机科学家和软件工程师,以其在编程语言、编译器技术和软件开发工具领域的贡献而闻名。以下是关于他的详细介绍: 1. 主要成就 (1)LLVM 项目的创始人 Chris La…...
Redis数据库篇 -- Pipeline
一. 什么是Pipeline 在传统的请求-响应模式中,客户端与服务器之间的通信流程如下: 客户端发送一个命令到服务器。服务器接收命令并执行。服务器将执行结果返回给客户端。客户端接收结果后,发送下一个命令 在这种传统的模式下,…...
爬虫自动化(DrissionPage)
目录 ?一.介绍: 下载DrissionPage,还是我们熟悉的pip: 环境准备: ?二.基本代码: 它对于的导包和类使用: 窗口的设置: 和获取的页面的滑动: 3.进一步认识DrissionPage: 浏览器可以多开…...
常见string库中的函数(C语言超详细)
文章目录 strcspnstrcpystrncpystrcatstrncatstrcmpstrncmpstrchrstrrchrstrstrstrtokstrlenstrnlen strcspn 原型: size_t strcspn(const char *str1, const char *str2);功能: strcspn 会扫描 str1,并返回一个整数,表示 str1 中第一个匹配…...
Java 语言特性(面试系列2)
一、SQL 基础 1. 复杂查询 (1)连接查询(JOIN) 内连接(INNER JOIN):返回两表匹配的记录。 SELECT e.name, d.dept_name FROM employees e INNER JOIN departments d ON e.dept_id d.dept_id; 左…...
VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...
postgresql|数据库|只读用户的创建和删除(备忘)
CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...
第25节 Node.js 断言测试
Node.js的assert模块主要用于编写程序的单元测试时使用,通过断言可以提早发现和排查出错误。 稳定性: 5 - 锁定 这个模块可用于应用的单元测试,通过 require(assert) 可以使用这个模块。 assert.fail(actual, expected, message, operator) 使用参数…...
python如何将word的doc另存为docx
将 DOCX 文件另存为 DOCX 格式(Python 实现) 在 Python 中,你可以使用 python-docx 库来操作 Word 文档。不过需要注意的是,.doc 是旧的 Word 格式,而 .docx 是新的基于 XML 的格式。python-docx 只能处理 .docx 格式…...
3-11单元格区域边界定位(End属性)学习笔记
返回一个Range 对象,只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意:它移动的位置必须是相连的有内容的单元格…...
Springboot社区养老保险系统小程序
一、前言 随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱,社区养老保险系统小程序被用户普遍使用,为方…...
iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈
在日常iOS开发过程中,性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期,开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发,但背后往往隐藏着系统资源调度不当…...
