【C++】---STL之vector详解

【C++】---STL之vector详解

  • 一、vector的介绍:
  • 二、vector的成员函数:
    • 1、vector类的构造函数
    • 2、vector的元素访问符
    • 3、vector的迭代器
    • 4、vector的模版
    • 5、vector的拷贝构造
    • 6、vector的容量
      • (1)vector的增容机制
      • (2)reserve()和resize()
      • (3)size()
      • (4)empty()
    • 7、vector的尾插和尾删
    • 8、vector在任意位置插入与删除
      • (1)插入(insert)
      • (2)删除
    • 9、find()
    • 10、swap()
  • 三、迭代器失效
    • (1)什么是迭代器失效?
    • (2)为什么会出现:迭代器失效问题?
    • (3)如何解决迭代器失效?!
    • (4)迭代器失效的样例

在这里插入图片描述

一、vector的介绍:

二、vector的成员函数:

vector定义:表示可以动态改变大小的数组序列容器!( vector实际上就是一个顺序表

vector的特点:

(1)vector像数组一样拥有连续的储存空间来储存元素,因此可以通过下标访问来访问储存的元素。

然而还有一点,它不像数组,那就是它可以动态改变其自身的大小,而数组是静态的,它改变其自身大

小是容器对容量自动处理的。

(2)vector的空间比实际上所需要储存的空间更大一点, vector的尾插尾删的效率更高一点,而在其

他位置的插入删除效率相对较低一点,因为每次的插入删除都会挪动后面的数据。

1、vector类的构造函数

1、参考文档:
在这里插入图片描述
2、
(1)

//构造空的vector
explicit vector(const allocator_type& alloc = allocator_type());

//构造一个vector,有n个元素,每个元素值为val
explicit vector(size_type n, const value_type& val = value_type(),
    const allocator_type& alloc = allocator_type());

//构造一个vector,值为InputIterator的first到last之间的元素
template <class InputIterator>
vector(InputIterator first, InputIterator last,
    const allocator_type& alloc = allocator_type());

//使用x拷贝构造一个vector
vector(const vector& x);

(2)
vector容器的定义元素、构造函数、拷贝构造函数、尾插:

vector<int> v;// 构造一个空的vector容器

	vector<int> v1(3, 5);// 构造一个元素个数为3,每个元素的初始化为5的容器
	vector<int> v2(v1.begin(),v1.end());// 构造一个从v1.begin()到v1.end()的v2容器
	vector<int> v3(v1);// 拷贝构造


	v.push_back(1);// 对v1空容器进行尾插。
	v.push_back(2);
	v.push_back(3);
	v.push_back(4);
	v.push_back(5);

2、vector的元素访问符

#include<iostream>
#include<vector>

using namespace std;


int main()
{

	
	vector<int> v;// 构造一个空的vector容器

	vector<int> v1(3, 5);// 构造一个元素个数为3,每个元素的初始化为5的容器
	vector<int> v2(v1.begin(),v1.end());// 构造一个从v1.begin()到v1.end()的v2容器
	vector<int> v3(v1);// 拷贝构造


	v.push_back(1);// 对v1空容器进行尾插。
	v.push_back(2);
	v.push_back(3);
	v.push_back(4);
	v.push_back(5);

	// 1、通过访问符[ ]来访问vector容器中的元素:
	for (size_t i = 0; i <v.size(); i++)
	{
		cout << v[i] << " ";
	}
	cout << endl;
	return 0;
}

在这里插入图片描述

3、vector的迭代器

1、迭代器是各种容器,无论是连续还是不连续的空间的容器,通用的遍历容器的方式。

(1)可读可写的迭代器:

int main()
{


	vector<int> v;// 构造一个空的vector容器

	vector<int> v1(3, 5);// 构造一个元素个数为3,每个元素的初始化为5的容器
	vector<int> v2(v1.begin(), v1.end());// 构造一个从v1.begin()到v1.end()的v2容器
	vector<int> v3(v1);// 拷贝构造


	v.push_back(1);// 对v1空容器进行尾插。
	v.push_back(2);
	v.push_back(3);
	v.push_back(4);
	v.push_back(5);

	// 2、迭代器遍历vector
	vector<int>::iterator it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		it++;
	}
	cout << endl;


	return 0;
}

在这里插入图片描述
(2)只读的迭代器:
在这里插入图片描述
库里面存在:const迭代器,直接用

	// 2、迭代器遍历vector
	vector<int>::const_iterator it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		it++;
	}
	cout << endl;

(3)反向迭代器:
在这里插入图片描述

可读可写:

	// 2、反向迭代器遍历vector
	vector<int>::reverse_iterator it = v.rbegin();
	while (it != v.rend())
	{
		cout << *it << " ";
		it++;
	}
	cout << endl;

在这里插入图片描述

(4)反向迭代器:只读:


	// 2、反向迭代器遍历vector
	vector<int>::const_reverse_iterator it = v.rbegin();
	while (it != v.rend())
	{
		cout << *it << " ";
		it++;
	}
	cout << endl;

4、vector的模版

template<class T>
void PrintVector(const vector<T>& v)
{
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
}
int main()
{
	// 1、定义一个v1的顺序表,里面的数据都是int类型
	vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);
	// 调用打印函数:printvector
	PrintVector(v1);

	// 2、定义一个v2的顺序表,里面的数据都是char类型,并且把s1里面的字符串内容,初始化时赋给v2
	string s1 = "hello world";

	vector<char> v2(s1.begin(),s1.end());
	// 调用打印函数:printvector
	PrintVector(v2);

	// 3、定义一个v3的顺序表,里面的数据都是double类型
	vector<double> v3;

	v3.push_back(1.1);
	v3.push_back(2.2);
	v3.push_back(3.3);
	v3.push_back(4.4);
	v3.push_back(5.5);
	// 调用打印函数:printvector
	PrintVector(v3);

	return 0;
}

在这里插入图片描述

5、vector的拷贝构造

1、拷贝构造:
在这里插入图片描述
2、样例展示:

	string s2("hello vector");
	vector<string> v4;
	v4.push_back(string(s2));
	PrintVector(v4);

	// 此时v4的内容就是:hello vector
	// 拷贝构造:
	vector<string> v5(v4);// 把v4拷贝构造给v5
	PrintVector(v5);

在这里插入图片描述

6、vector的容量

(1)vector的增容机制

在VS下执行这段代码:

void test_vector3()
{
    size_t sz;
    std::vector<int> foo;
    sz = foo.capacity();
  
    std::cout << "making foo grow:\n";
 
    for (int i = 0; i < 100; i++)
    {
        foo.push_back(i);
        if (sz != foo.capacity())
        {
            sz = foo.capacity();
            std::cout << "capacity changed:" << sz << endl;
        }
    }
}

发现增容的过程是大概按照1.5倍增容的:
在这里插入图片描述
同样的代码在linux下运行,发现是2倍增容的!

(2)reserve()和resize()

1、reserve():

void reserve (size_type n);//开辟n个元素空间
void resize (size_type n, value_type val = value_type());
//开辟n个元素空间,并将每个元素默认初始化为val

如果加上reserve(),那么会提前知道要开多少空间,就提前开好了,避免后面再开空间

void test_vector3()
{
    size_t sz;
    std::vector<int> foo;
    sz = foo.capacity();
    foo.reserve(100);
  
    std::cout << "making foo grow:\n";
 
    for (int i = 0; i < 100; i++)
    {
        foo.push_back(i);
        if (sz != foo.capacity())
        {
            sz = foo.capacity();
            std::cout << "capacity changed:" << sz << endl;
        }
    }
}

2、resize():
在这里插入图片描述
(1)样例演示:

vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);


	PrintVector(v1);// resize()前:

	v1.resize(3);// 将v1顺序表的元素个数缩小为3个
	PrintVector(v1);// resize()后:

	v1.resize(10,0);// 将v1顺序表的元素个数扩大为10个,并且其余多出来的元素初始化为0。
	PrintVector(v1);// resize()后:

在这里插入图片描述

(3)size()

1、计算顺序表中的·元素个数:

	vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);

	cout << v1.size() << endl;

在这里插入图片描述

(4)empty()

1、判断顺序表是否为空:

	vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);

	cout << v1.empty() << endl;

	vector<int> v2;
	cout << v2.empty() << endl;

在这里插入图片描述

7、vector的尾插和尾删

	vector<int> v1;
	v1.push_back(1);//尾插
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);

	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl;

	v1.pop_back();
	v1.pop_back();//尾删2次
	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl;

在这里插入图片描述

8、vector在任意位置插入与删除

(1)插入(insert)

1、函数的用法理解:

在这里插入图片描述

    vector<double> v5;
    //插入4个float元素
    v5.push_back(1.1);
    v5.push_back(2.2);
    v5.push_back(3.3);
    v5.push_back(4.4);
    v5.push_back(5.5);
    
    v5.insert(v5.end(), 6.6);//在v5末尾插入6.6
    PrintVector(v5);
    
    v5.insert(v5.begin(), 2,0);//在v5开头插入2个0
    PrintVector(v5);
 
    vector<double> v6;
    v6.push_back(7.7);
    v6.push_back(8.8);
 
    v5.insert(v5.begin(), v6.begin(),v6.end());//在v5开头插入v6
    PrintVector(v5);

(2)删除

1、erase的使用:
在这里插入图片描述
2、特别要注意:erase可能会导致迭代器失效!!!

iterator erase (iterator position);//删除某一位置元素
iterator erase (iterator first, iterator last);//删除迭代器first和last之间的元素
    v5.erase(v5.begin());//删除v5开头元素
    PrintVector(v5);
 
    v5.erase(v5.begin(), v5.begin()+2);//从v5开头删除2个元素
    PrintVector(v5);

9、find()

1、在迭代器区间内查找元素,find函数实现在algorithm中,可以给所有容器使用,因此要使用find函数,就要include

template <class InputIterator, class T>
   InputIterator find (InputIterator first, InputIterator last, const T& val);//在InputIterator迭代器first和last区间内查找val元素的位置

注意:迭代器区间是左闭右开,因此能取到第一个位置,但取不到最后一个位置

    vector<double>::iterator pos = find(v5.begin(), v5.begin() + 3, 1.1);//在第一个元素和第四个元素(左闭右开,不包含第四个元素)之间查找值为1.1的元素位置
    v5.erase(pos);//删除1.1位置的元素,即删除1.1
 
     PrintVector(v5);

10、swap()

1、样例展示:

vector<int> v1;
	v1.push_back(1);//尾插
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);

	vector<int> v2;
	v2.push_back(3);
	v2.push_back(3);
	v2.push_back(3);
	v2.push_back(3);
	v2.push_back(3);

	v1.swap(v2);

	cout << "v1:";
	for (auto e : v1)
	{
		
		cout << e << " ";
	}
	cout << endl;

	cout << "v2:";
	for (auto e : v2)
	{
		
		cout << e << " ";
	}
	cout << endl;

在这里插入图片描述

三、迭代器失效

(1)什么是迭代器失效?

1、首先明白什么是迭代器?

1.迭代器的主要作用:就是让算法能够不用关心底层的数据结构,它的底层实际上就是一个指针或者是对指针进行了封装

什么是迭代器失效?迭代器变成了“野指针”!

(2)为什么会出现:迭代器失效问题?

2、为什么会存在迭代器失效?

(1)迭代器失效的原因:是因为迭代器指针所指向的空间被销毁了,而使用一块已经被释放的空间。造成的后果就是程序崩溃。

(2)注意:会引起其底层空间操改变的操作都有可能使迭代器失效,比如resize reserve Insert erase push_back等等。

(3)如何解决迭代器失效?!

如何解决迭代器失效?!
在使用前,对迭代器进行重新赋值即可。

1、样例展示:

#include <iostream>
using namespace std;
#include <vector>
 
int main()
{
 	vector<int> v{ 1, 2, 3, 4 };
 	auto it = v.begin();
 	while (it != v.end())
 	{
 		if (*it % 2 == 0)
 		v.erase(it);
 
 		++it;//  It是一个迭代器,对一块已经释放了空间的迭代器指针进行操作必然会引起程序崩溃。要想程序不崩溃,必须要对迭代器进行重新赋值!
 	}
 
 	return 0;
}
 



int main()
{
	 vector<int> v{ 1, 2, 3, 4 };
	 auto it = v.begin();
 	while (it != v.end())
 	{
		if (*it % 2 == 0)
 			it = v.erase(it);// 在迭代器失效前,对迭代器进行重新赋值,即可保证迭代器不会失效。程序不会崩溃。
 		else
 			++it;
 }
 
 return 0;
}

要特别注意,所有容器如果对它底层的空间数据进行改变的话,就有可能引起迭代器失效,常见的有insert和erase。

(4)迭代器失效的样例

1、

#include <iostream>
using namespace std;
#include <vector>
 
int main()
{
 vector<int> v{1,2,3,4,5,6};
 
 auto it = v.begin();
 
 // 将有效元素个数增加到100个,多出的位置使用8填充,操作期间底层会扩容
 // v.resize(100, 8);
 
 // reserve的作用就是改变扩容大小但不改变有效元素个数,操作期间可能会引起底层容量改变
 // v.reserve(100);
 
 // 插入元素期间,可能会引起扩容,而导致原空间被释放
 // v.insert(v.begin(), 0);
 // v.push_back(8);
 
 // 给vector重新赋值,可能会引起底层容量改变
 v.assign(100, 8);
 
 /*
 出错原因:以上操作,都有可能会导致vector扩容,也就是说vector底层原理旧空间被释放掉,
而在打印时,it还使用的是释放之间的旧空间,在对it迭代器操作时,实际操作的是一块已经被释放的
空间,而引起代码运行时崩溃。
 解决方式:在以上操作完成之后,如果想要继续通过迭代器操作vector中的元素,只需给it重新
赋值即可。
 */
 while(it != v.end())
 {
 cout<< *it << " " ;
 ++it;
 }
 cout<<endl;
 return 0;
}

2、

#include <iostream>
using namespace std;
#include <vector>
 
int main()
{
 int a[] = { 1, 2, 3, 4 };
 vector<int> v(a, a + sizeof(a) / sizeof(int));
 
 // 使用find查找3所在位置的iterator
 vector<int>::iterator pos = find(v.begin(), v.end(), 3);
 
 // 删除pos位置的数据,导致pos迭代器失效。
 v.erase(pos);
 cout << *pos << endl; // 此处会导致非法访问
 return 0;
}

erase删除pos位置元素后,pos位置之后的元素会往前搬移,没有导致底层空间的改变,理论上讲迭代 器不应该会失效,但是:如果pos刚好是最后一个元素,删完之后pos刚好是end的位置,而end位置是 没有元素的,那么pos就失效了。因此删除vector中任意位置上元素时,vs就认为该位置迭代器失效 了。

3、

#include <string>
void TestString()

{
 string s("hello");
 auto it = s.begin();
 
 // 放开之后代码会崩溃,因为resize到20会string会进行扩容
 // 扩容之后,it指向之前旧空间已经被释放了,该迭代器就失效了
 // 后序打印时,再访问it指向的空间程序就会崩溃
 //s.resize(20, '!');
 while (it != s.end())
 {
 cout << *it;
 ++it;
 }
 cout << endl;
 
 it = s.begin();
 while (it != s.end())
 {
 it = s.erase(it);
 // 按照下面方式写,运行时程序会崩溃,因为erase(it)之后
 // it位置的迭代器就失效了
 // s.erase(it); 
 ++it;
 }

好了,今天的分享就到这里了
如果对你有帮助,记得点赞👍+关注哦!
我的主页还有其他文章,欢迎学习指点。关注我,让我们一起学习,一起成长吧!
在这里插入图片描述

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

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

相关文章

学习笔记Day21:转录组差异分析

转录组差异分析 差异分析难点在于将数据处理成需要的格式 表达矩阵 数值型矩阵-count 行名是symbol 低表达量的基因需要过滤 分组信息 因子&#xff0c;对照组在level第一位 与表达矩阵的列一一对应 项目名称 字符串&#xff08;不要有特殊字符&#xff09; TCGA-XX…

IntelliJ IDEA - Lombok supports: OpenJDK javac, ECJ

问题描述 java: You arent using a compiler supported by lombok, so lombok will not work and has been disabled.Your processor is: com.sun.proxy.$Proxy26Lombok supports: OpenJDK javac, ECJ 解决方案 在 IDEA 设置中 File -> Settings 中找到配置如下&#xff1…

openEuler-23.03下载、安装

一、下载 下载地址&#xff1a;openEuler下载 | 欧拉系统ISO镜像 | openEuler社区官网 下载版本&#xff1a;openEuler-23.03-x86_64-dvd.iso 二、安装 cd /etc/sysconfig/network-scripts/ vi ifcfg-ens-33## 要修改部分 BOOTPROTOstatic## 新增部分 IPADDR192.168.1.128 …

FPGA - ZYNQ 基于Axi_Lite的PS和PL交互

前言 在FPGA - ZYNQ 基于EMIO的PS和PL交互中介绍了ZYNQ 中PS端和PL端交互的开发流程&#xff0c;接下来构建基于基于Axi_Lite的PS和PL交互。 开发流程 Axi_Lite从机 在FPGA - AXI4_Lite&#xff08;实现用户端与axi4_lite之间的交互逻辑&#xff09;中&#xff0c;详解介绍…

【学习笔记】Vue3源码解析:第五部分 - 实现渲染(3)

课程地址&#xff1a;【已完结】全网最详细Vue3源码解析&#xff01;&#xff08;一行行带你手写Vue3源码&#xff09; 第五部分-&#xff1a;&#xff08;对应课程的第36 - 37节&#xff09; 第36节&#xff1a;《处理proxy&#xff0c;方便取值》 1、执行组件中的 render 方…

BootstrapAdmin Net7:基于RBAC的后台管理框架,实现精细化权限管理与多站点单点登录

BootstrapAdmin Net7&#xff1a;基于RBAC的后台管理框架,实现精细化权限管理与多站点单点登录 摘要 随着企业信息化建设的不断深入&#xff0c;后台管理系统在企业运营中扮演着越来越重要的角色。本文介绍了一款基于RBAC&#xff08;Role-Based Access Control&#xff09;的…

腾讯云轻量2核2G4M服务器优惠价格99元一年,多配置报价单

腾讯云轻量2核2G4M服务器优惠价格99元一年&#xff0c;多配置报价单。腾讯云服务器价格表2024年最新价格&#xff0c;轻量2核2G3M服务器61元一年、2核2G4M服务器99元1年&#xff0c;三年560元、2核4G5M服务器165元一年、3年900元、轻量4核8M12M服务器646元15个月、4核16G10M配置…

WEB攻防-ASP安全-ASP后门植入连接

windows2003环境搭建&#xff0c;可参考上一篇WEB攻防-ASP安全-MDB下载-CSDN博客 将aspcms解压到C:\inetpub\wwwroot,创建网站并赋予internet来宾用户权限 配置启用父路径和主页指向 上一篇文章提到&#xff0c;数据库文件后缀为asp、asa会被执行解析&#xff0c;所以当进行访…

Redis中的慢查询日志和监视器

慢查询 添加新日志 在每次执行命令的之前和之后&#xff0c;程序都会记录微妙格式的当前UNIX时间戳&#xff0c;这两个时间戳之间的差就是服务器执行命令所耗费的时长&#xff0c;服务器会将这个时长作为参数之一传给slowlogPushEntryIfNeeded函数&#xff0c;而slowlogPushE…

每日算法之矩阵置零

题目描述 给定一个 m x n 的矩阵&#xff0c;如果一个元素为 0 &#xff0c;则将其所在行和列的所有元素都设为 0 。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,1,1],[1,0,1],[1,1,1]] 输出&#xff1a;[[1,0,1],[0,0,0],[1,0,1]] 示例 2&#xff1a; 输入&#x…

TQZC706开发板教程:编译zynq linux内核2019_R1

您需要下载对应版本的Linux系统文件以及IMG1.3.1镜像文件。为了方便您的操作&#xff0c;本文所使用的所有文件以及最终生成的文件&#xff0c;我都已经整理并放置在本文末尾提供的网盘链接中。您可以直接通过该链接进行下载&#xff0c;无需在其他地方单独搜索和获取。希望这能…

前端三剑客 HTML+CSS+JavaScript ③ HTML标准结构

生活没有任何意义&#xff0c;这就是活着的理由&#xff0c;而且是唯一的理由 —— 24.4.22 一、HTML注释 1.特点 注释的内容会被浏览器所忽略&#xff0c;不会呈现到页面中&#xff0c;但源代码中依然可见 2.作用 对代码进行解释和说明 3.写法 <!-- xxxxx --> <html&…

webgl canvas系列——animation中基本旋转、平移、缩放(模拟冒泡排序过程)

文章目录 ⭐前言⭐canvas绘制图片&#x1f496;状态保存和恢复&#x1f496;移动、旋转、缩放、变形&#x1f496;移动绘制一个渐变的box&#x1f496;旋转&#x1f496;缩放 ⭐模拟冒泡排序过程⭐结束 ⭐前言 大家好&#xff0c;我是yma16&#xff0c;本文分享webgl canvas系…

【MySQL 数据宝典】【磁盘结构】- 002 数据字典

一、数据字典 ( Data Dictionary ) 1.1 背景介绍 我们平时使用 INSERT 语句向表中插入的那些记录称之为用户数据&#xff0c;MySQL只是作为一个软件来为我们来保管这 些数据&#xff0c;提供方便的增删改查接口而已。但是每当我们向一个表中插入一条记录的时候&#xff0c;MyS…

周鸿祎和雷军、马化腾相逢一笑泯恩仇

关注卢松松&#xff0c;会经常给你分享一些我的经验和观点。 马云竟然没有到场&#xff0c;真是遗憾! 前两天工信部和互联网协会联合举办的中国互联网三十周年座谈会上。周鸿祎、雷军、马化腾相逢一笑泯恩仇。 第一条视频&#xff1a; 周鸿祎和马化腾握手言欢&#xff0c…

Mendix是谁?作为致力于企业低代码服务平台的领头羊,它解决了哪些问题?

一、Mendix 成立的背景 Mendix的成立是为了解决软件开发中最大的问题&#xff1a;业务和IT之间的脱节。这一挑战在各个行业和地区都很普遍&#xff0c;很简单&#xff1a;业务需求通常被描述为IT无法正确解释并转化为软件。业务和IT之间缺乏协作的原因是传统的代码将开发过程限…

Vue.js前端开发零基础教学(六)

学习目标 了解什么是路由&#xff0c;能够说出前端后端路由的原理 掌握多种路由的使用方法&#xff0c;能够实现路由的不同功能 掌握Vue Router的安装及基本使用方法 5.1 初始路由 提到路由&#xff08;Route),一般我们会联想到网络中常见的路由器&#xff08;Router),…

30 消息队列

原理 操作系统可以通过页表映射在共享区创建一块共享内存&#xff0c;也可以申请一个队列。A进程和B进程可以向这个队列发送数据块&#xff0c;两个进程接收数据块来通信 函数 申请数据块 参数中的key来自于ftok函数 删除消息队列 同样消息队列也有数据结构管理&#xff…

c#学习入门1

一、环境配置 颜色主题 字体设置 行号设置 二、第一个应用程序 1. 在解决方案下创建一个新项目 第一种注释&#xff1a;两杠注释 第二种注释&#xff1a;星号注释 第三种注释&#xff1a;三杠注释(只有在花括号后面输出才会自动补全&#xff09; 2.控制台输入打印基础语句 输…

java在线问卷调查系统的设计与实现(springboot+mysql源码+文档)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的在线问卷调查系统。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 基于java的在线问卷调查…