【C++ 】学习问题及补充

一.自定义类型不初始化直接就赋值,比如string类会怎么样

vectr<string>里已经给每个string对象已经分配好空间,为什么不初始化再赋值会报错

在C++中,std::string类是一个动态字符串类,它内部管理着一个字符数组,用于存储字符串内容。当你创建一个std::string对象时,如果没有指定初始长度,编译器默认会给它分配足够的内存来容纳后续可能添加的字符。如果你已经为std::string对象分配了空间,但没有初始化就直接赋值,这通常不会导致编译错误,因为std::string内部会自动处理这种情况。

然而,不初始化就赋值可能会带来一些潜在的问题:

  1. 未定义行为尽管编译器可能允许你这样做,但这可能导致未定义的行为。因为你没有明确地初始化字符串,其内部状态是不确定的,这意味着它的长度、容量以及所指向的内存可能是随机的。在这种情况下,对字符串的操作(例如,尝试访问其中的字符)可能会产生不可预知的结果。

  2. 效率问题:虽然std::string会根据需要自动调整大小,但频繁的动态调整可能会降低程序性能。每次调用push_backassign等方法时,如果当前的内部缓冲区不足以容纳新的字符,std::string可能需要重新分配更大的内存区域,并复制旧的数据到新位置,这是一个相对耗时的过程。

  3. 内存泄漏的风险:如果你手动分配了一个字符数组(比如使用new char[size]),然后试图将这个数组与std::string关联起来,而没有正确地初始化std::string,那么可能会导致内存泄漏。因为std::string会在析构时释放其管理的内存,但如果它不知道内存是在哪里分配的,就不能正确地执行这一操作。

为了避免这些问题,推荐的做法始终是初始化std::string对象,即使你不提供初始值,也可以传递一个空字符串给构造函数:

std::string myString;  // 初始化为空字符串
myString = "Hello, World!";  // 现在可以安全地赋值

或者,如果你知道字符串的预期最大长度,可以显式指定:

std::string myString(100);  // 分配100个字符的空间,但字符串长度仍为0
myString.assign("Hello, World!");  // 安全地赋值,不需要动态调整大小

这样可以确保你的代码更稳定且高效。


自定义类型不初始化直接就赋值,比如string类会怎么样

如果自定义类型没有被初始化就直接赋值,那么其成员变量的值将是不确定的,也就是未定义行为。对于string类,如果没有被初始化就直接赋值,它的值将是随机的,可能是一些垃圾值,也可能是一个空字符串。这种行为是不可预测的,会导致程序出现不可预期的错误。

建议在使用自定义类型时,一定要先进行初始化,以保证程序的正确性和可靠性。

为什么没有被初始化就直接赋值,它的值将是随机的

计算机中,内存是由一系列的字节构成的,当变量在内存中被定义时,它会被分配一块内存空间。如果这个变量没有被显式初始化,那么这个内存空间中就会存在垃圾数据,这些数据是来自于此前使用这段内存空间的程序留下的信息,因此这个变量的值就是随机的,可能是任何值。所以在使用变量之前必须对其进行初始化操作,否则可能会导致程序出现不可预期的错误。


变量的声明和定义

(摘自)【C/C++面试必备】声明和定义的区别

 声明

声明是用来告诉编译器变量的名称和类型,而不分配内存。

extern int var; // 声明
 
extern int ble =10; // 定义
 
typedef int INT; // 声明
 
struct Node; // 声明

上面代码中,语句 extern int var;表示 var 在别的文件中已经定义,提示编译器遇到此变量时在其它模块中寻找其定义。语句 extern int ble = 10;表示定义了变量 ble,这一点需要注意。

注意:即使是 extern ,如果给变量赋值了,就是定义了。

定义

定义是为了给变量分配内存,可以为变量赋初值。

注意:全局变量或静态变量初始值为0,局部变量初始化为随机值。

在 C/C++ 中,变量的声明和定义区别并不大,定义和声明往往是同时发生,变量定义时,会根据变量类型分配空间,如下所示:

int value ; //声明 + 定义
 
struct Node { // 声明 + 定义
    int left;
    int right;
}; 

变量的声明、定义、赋值和初始化都是在预处理阶段进行的吗


二.std::vector::iterator 为什么不重载运算符>>

std::vector::iterator 是一个迭代器,用于遍历 vector 中的元素。它不重载运算符>>,是因为这个运算符已经被用于输入流。如果重载运算符>>,会导致代码的可读性变差,同时也会带来潜在的歧义。

当我们使用输入流时,我们通常使用运算符>>从输入流中读取数据。如果 std::vector::iterator 也重载了这个运算符,那么就无法确定应该调用哪个运算符。这会导致代码的可读性变差,并且可能会出现难以调试的问题。

因此,std::vector::iterator 没有重载运算符>>。如果需要从输入流中读取 vector 中的元素,可以使用其他方式,例如循环遍历 vector 并逐个读取元素。

这些可以重载:++  *(前置) ==


三.iterator end() { return _finish; }可以写成 iterator& end() { return _finish; }吗?

引用返回的话,你改变这个迭代器的指向,那么容器中的迭代器指向也会跟着改变,就会让容器损坏,所以需要的是一个拷贝行为。

iterator beign 和iterator _start是两个不同的哦。


四.不能给初始化列表动态开辟空间

在 C++ 中,初始化列表用于初始化类的成员变量。如果一个类的成员变量需要动态分配内存,那么在初始化列表中是不能进行动态内存分配的。因为初始化列表是在构造函数执行之前执行的,而动态内存分配必须在程序运行时才能完成,所以无法在初始化列表中进行动态内存分配。

举个例子,假设一个类的成员变量包含一个指向 int 类型数组的指针,如果要在构造函数中动态分配内存给这个数组,可以使用 new 操作符来实现。但是,由于初始化列表是在构造函数之前执行的,所以在初始化列表中不能使用 new 操作符来分配内存。


五.具有常性的变量不能引用

具有常性的变量(const variable)是指其值不能被修改的变量。由于其值不可更改,因此不能将其引用给一个非常量变量,因为这样做就会打破 const 变量的常性。而将一个 const 变量引用给一个常量变量是可以的,因为常量变量的值也不会被修改。

例如,以下代码是不合法的:

const int a = 5;
int &b = a; // 错误!不能将 const 变量引用给非常量变量

而以下代码是合法的:

const int a = 5;
const int &b = a; // 正确!可以将 const 变量引用给常量变量

六.模板函数不允许自动类型转换,但普通函数可以进行自动类型转换

C++ 中的模板函数(函数模板)可以被用于多种不同类型的参数,但是这些参数必须在编译时确定,因此模板函数不允许自动类型转换。相反,普通函数可以在参数和返回值之间进行自动类型转换。

例如,以下是一个模板函数和一个普通函数的示例:

// 模板函数
template <typename T>
T getMax(T a, T b) {
    return (a > b) ? a : b;
}

// 普通函数
int getMax(int a, int b) {
    return (a > b) ? a : b;
}

当你调用 getMax(3, 5) 时,编译器会自动推断出使用普通函数,因为参数是整数类型。但是,当你调用 getMax(3.5, 2.8) 时,编译器将无法确定使用哪个函数,因为参数既可以是 int 类型也可以是 double 类型。在这种情况下,你必须显式地指定类型来调用模板函数。

七.模板满树的声明和定义必须放在一起(比如:vector<T>)

(不同于一般c++函数,声明可以在头文件,定义可以在cpp文件。

模板函数的声明和定义必须放在一起,否则编译链接会报错。

非要分开写的话,需要在cpp文件中进行模板的显式实例化,但这需要把所有用到的T都列出来。或者在头文件末尾include 这个cpp文件,但这本质和写在一个头文件没有区别,而且由于cpp文件往往没有防止重复include的机制,链接时容易报错重复定义。

参考:

类模板的定义和实现必须在一个文件吗_c++ 接口类 实现和定义不在同一个文件

/* a.hpp */
template <typename T>
T func(T input);
 
/* 必须也放在a.hpp */
template <typename T>
T func(T input)
{
    return input;
}

汉诺塔问题,青蛙跳台,斐波那契数列


汉诺塔问题和青蛙跳台阶问题/汉诺塔和斐波那契数列的递归算法实现

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/646874.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++ | Leetcode C++题解之第111题二叉树的最小深度

题目&#xff1a; 题解&#xff1a; class Solution { public:int minDepth(TreeNode *root) {if (root nullptr) {return 0;}queue<pair<TreeNode *, int> > que;que.emplace(root, 1);while (!que.empty()) {TreeNode *node que.front().first;int depth que…

蓝桥杯物联网竞赛_STM32L071_17_DMA收发 不定长DMA接收

前言&#xff1a; 前面已经说过&#xff0c;由于国赛的代码量的增加&#xff0c;cpu在其他代码的时间块会较省赛大大增加&#xff0c;为了减少对cpu的依赖所以学习DMA收发数据 对于串口中断收发来说串口接收数据无法收取不定长数据所以不好用&#xff0c;而DMA有收集不定长数…

汇编:字符串的输出

在16位汇编程序中&#xff0c;可以使用DOS中断21h的功能号09h来打印字符串&#xff1b;下面是一个简单的示例程序&#xff0c;演示了如何在16位汇编程序中打印字符串&#xff1a; assume cs:code,ds:data ​ data segmentszBuffer db 0dh,0ah,HelloWorld$ //定义字符串 data …

Hadoop概览以及编译hadoop说明

一、Hadoop概述 Hadoop 是一个用于跨计算机集群存储和处理大型数据集的软件框架。它旨在处理大数据&#xff0c;即传统数据库无法有效管理的极其庞大和复杂的数据集。Hadoop不是传统意义上的数据仓库&#xff0c;因为它们的用途不同&#xff0c;架构也不同。Hadoop 是一个跨分布…

Elasticsearch 8.1官网文档梳理 - 十三、Search your data(数据搜索)

Search your data 这里有两个比较有用的参数需要注意一下 Search timeout&#xff1a;设置每个分片的搜索超时时间。从集群级别可以通过 search.default_search_timeout 来设置超时时间。如果在 search.default_search_timeout 设置的时间段内未完成搜索请求&#xff0c;就会…

京东应届生公司内网说了一句‘什么时候被pdd收购‘,结果惨遭辞退

京东应届生公司内网说了一句’什么时候被pdd收购’&#xff0c;结果惨遭公司开除 这个事最近在圈子讨论比较多 前二天&#xff0c;有一个上海交大毕业的应届生&#xff0c;在京东实习了9个月&#xff0c;好不容易转正12天后&#xff0c;只因在内网说了一句话&#xff0c;就被…

【传知代码】无监督动画中关节动画的运动表示-论文复现

文章目录 概述动画技术的演进原理介绍核心逻辑环境配置/部署方式小结 本文涉及的源码可从无监督动画中关节动画的运动表示该文章下方附件获取 概述 该文探讨了动画在教育和娱乐中的作用&#xff0c;以及通过数据驱动方法简化动画制作的尝试。近期研究通过无监督运动转移减少对…

python数据分析-CO2排放分析

导入所需要的package import numpy as np import pandas as pd import matplotlib.pyplot as plt import seaborn as sns import datetime %matplotlib inline plt.rcParams[font.sans-serif] [KaiTi] #中文 plt.rcParams[axes.unicode_minus] False #负号 数据清洗…

汇编原理(二)

寄存器&#xff1a;所有寄存器都是16位&#xff08;0-15&#xff09;&#xff0c;可以存放两个字节 AX,BX,CX,DX存放一般性数据&#xff0c;称为通用寄存器 AX的逻辑结构。最大存放的数据为2的16次方减1。可分为AH和AL&#xff0c;兼容8位寄存器。 字&#xff1a;1word 2Byte…

Mysql中表之间的关系

表之间的关系 一对一、多对一&#xff08;其实就是主从关系&#xff0c;在从表中设置一个外键关联上主表&#xff09;、多对多关系&#xff08;需要一个中间表&#xff0c;设置两个外键&#xff0c;分别关联到两个表的主键&#xff09; 比如订单和商品之间&#xff1a;一个订单…

数据安全不容小觑:.hmallox勒索病毒的防范与应对

一、引言 随着网络技术的飞速发展&#xff0c;网络安全问题日益凸显&#xff0c;其中勒索病毒作为一种极具破坏性的网络攻击手段&#xff0c;已在全球范围内造成了巨大的经济损失和社会影响。在众多勒索病毒中&#xff0c;.hmallox勒索病毒以其狡猾的传播方式和强大的加密能力…

深度学习复盘与小实现

文章目录 一、查漏补缺复盘1、python中zip()用法2、Tensor和tensor的区别3、计算图中的迭代取数4、nn.Modlue及nn.Linear 源码理解5、知识杂项思考列表6、KL散度初步理解 二、处理多维特征的输入1、逻辑回归模型流程2、Mini-Batch (N samples) 三、加载数据集1、Python 魔法方法…

JVM学习-Class文件结构②

访问标识(access_flag) 在常量池后&#xff0c;紧跟着访问标记&#xff0c;标记使用两个字节表示&#xff0c;用于识别一些类或接口层次的访问信息&#xff0c;包括这个Class是类还是接口&#xff0c;是否定义为public类型&#xff0c;是否定义为abstract类型&#xff0c;如果…

打造游戏APP:面向对象编程的实战演练

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、项目背景与架构概览 二、类的设计与实现 三、面向对象编程的实践 四、游戏循环与事件…

WordPress安装memcached提升网站速度

本教程使用环境为宝塔 第一步、服务器端安装memcached扩展 在网站使用的php上安装memcached扩展 第二步&#xff1a;在 WordPress 网站后台中&#xff0c;安装插件「Memcached Is Your Friend」 安装完成后启用该插件&#xff0c;在左侧工具-中点击Memcached 查看是否提示“U…

【Git】版本控制工具——Git介绍及使用

目录 版本控制版本控制系统的主要目标分类小结 分布式版本控制系统——GitGit特点Git与SVN的区别Git的工作机制 Git安装Git 团队协作机制团队内协作跨团队协作远程仓库远程仓库的作用有以下几个方面远程仓库操作流程/团队协作流程 Git分支什么是分支分支的好处 Git的常用命令Gi…

JDBC使用QreryRunner简化SQL查询注意事项

QreryRunner是Dbutils的核心类之一&#xff0c;它显著的简化了SQL查询&#xff0c;并与ResultSetHandler协同工作将使编码量大为减少。 注意事项 1. 使用QreryRunner必须保证实体类的变量名&#xff0c;和sql语句中要查找的字段名必须相同&#xff0c;否则查询 不到数据,会出…

K-means聚类算法详细介绍

目录 &#x1f349;简介 &#x1f348;K-means聚类模型详解 &#x1f348;K-means聚类的基本原理 &#x1f348;K-means聚类的算法步骤 &#x1f348;K-means聚类的优缺点 &#x1f34d;优点 &#x1f34d;缺点 &#x1f348;K-means聚类的应用场景 &#x1f348;K-mea…

公共代理IP和独享代理IP之间的区别?

公共代理IP和独享代理IP在网络应用中扮演着不同的角色&#xff0c;它们之间的区别主要体现在使用方式、性能、安全性以及隐私保护等方面。以下是对这两种代理IP的详细对比和分析。 第一点就是使用的方式以及成本上的不同&#xff0c;公共代理IP&#xff0c;顾名思义&#xff0…

MySQL中, 自增主键和UUID作为主键有什么区别?

首先我们来看看, 存储自增主键和uuid的数据类型 我们知道, mysql中作为主键的通常是int类型的数据, 这个 数据从第一条记录开始, 从1开始主键往后递增, 例如我有100条数据, 那么根据主键排序后, 里面的记录从上往下一次就是1, 2, 3 ... 100, 但是UUID就不一样了, UUID是根据特殊…