如何实现序列化和反序列化?如何处理对象的生命周期管理?
序列化和反序列化
实现思路:
- 序列化:将对象的状态信息转换为可以存储或传输的格式,通常是字节流。
- 确定要序列化的对象的数据成员。
- 将这些数据成员按照一定的规则(如二进制、文本、JSON、XML 等)编码为字节序列。
- 将生成的字节序列存储到文件或发送到网络。
- 反序列化:将字节流恢复为对象的状态信息。
- 从文件或网络接收字节序列。
- 按照序列化时使用的规则解码字节序列。
- 根据解码后的数据成员创建或更新对象。
#include <iostream>
#include <fstream>
#include <string>class SerializableObject {
private:int data1;double data2;std::string data3;public:SerializableObject(int d1 = 0, double d2 = 0.0, const std::string& d3 = "") : data1(d1), data2(d2), data3(d3) {}// 序列化函数void serialize(const std::string& filename) const {std::ofstream file(filename, std::ios::binary);if (file.is_open()) {// 写入数据成员file.write(reinterpret_cast<const char*>(&data1), sizeof(data1));file.write(reinterpret_cast<const char*>(&data2), sizeof(data2));// 先写入字符串长度size_t len = data3.length();file.write(reinterpret_cast<const char*>(&len), sizeof(len));// 再写入字符串内容file.write(data3.c_str(), len);file.close();} else {std::cerr << "Failed to open file for serialization." << std::endl;}}// 反序列化函数void deserialize(const std::string& filename) {std::ifstream file(filename, std::ios::binary);if (file.is_open()) {// 读取数据成员file.read(reinterpret_cast<char*>(&data1), sizeof(data1));file.read(reinterpret_cast<char*>(&data2), sizeof(data2));// 先读取字符串长度size_t len;file.read(reinterpret_cast<char*>(&len), sizeof(len));// 再读取字符串内容data3.resize(len);file.read(&data3[0], len);file.close();} else {std::cerr << "Failed to open file for deserialization." << std::endl;}}void display() const {std::cout << "data1: " << data1 << ", data2: " << data2 << ", data3: " << data3 << std::endl;}
};int main() {SerializableObject obj(42, 3.14, "Hello, World!");std::string filename = "object.bin";// 序列化obj.serialize(filename);std::cout << "Serialized object: " << std::endl;obj.display();SerializableObject newObj;// 反序列化newObj.deserialize(filename);std::cout << "Deserialized object: " << std::endl;newObj.display();return 0;
}
代码解释:
SerializableObject
类包含三个数据成员:data1
(整数)、data2
(双精度浮点数)和data3
(字符串)。serialize
函数:- 使用
std::ofstream
以二进制模式打开文件。 - 对于
data1
和data2
,使用file.write
将它们的二进制表示写入文件。 - 对于
data3
,先写入字符串的长度,再写入字符串的内容。
- 使用
deserialize
函数:- 使用
std::ifstream
以二进制模式打开文件。 - 对于
data1
和data2
,使用file.read
读取它们的二进制表示。 - 对于
data3
,先读取字符串的长度,再读取字符串的内容。
- 使用
display
函数:打印对象的数据成员。
如何处理对象的生命周期管理?
实现思路:
- 构造函数:用于对象的初始化,可进行资源分配和成员初始化。
- 析构函数:用于对象销毁时释放资源,如释放动态分配的内存、关闭文件等。
- 拷贝构造函数和拷贝赋值运算符:控制对象的拷贝行为,避免浅拷贝导致的资源问题。
- 移动构造函数和移动赋值运算符:实现对象资源的高效移动,避免不必要的拷贝。
#include <iostream>
#include <string>
#include <utility>class ResourceManagingObject {
private:int* data;size_t size;public:// 构造函数ResourceManagingObject(size_t s = 0) : size(s) {if (s > 0) {data = new int[s];for (size_t i = 0; i < s; ++i) {data[i] = i;}} else {data = nullptr;}}// 析构函数~ResourceManagingObject() {delete[] data;}// 拷贝构造函数ResourceManagingObject(const ResourceManagingObject& other) : size(other.size) {if (other.data) {data = new int[size];for (size_t i = 0; i < size; ++i) {data[i] = other.data[i];}} else {data = nullptr;}}// 拷贝赋值运算符ResourceManagingObject& operator=(const ResourceManagingObject& other) {if (this == &other) return *this;delete[] data;size = other.size;if (other.data) {data = new int[size];for (size_t i = 0; i < size; ++i) {data[i] = other.data[i];}} else {data = nullptr;}return *this;}// 移动构造函数ResourceManagingObject(ResourceManagingObject&& other) noexcept : data(other.data), size(other.size) {other.data = nullptr;other.size = 0;}// 移动赋值运算符ResourceManagingObject& operator=(ResourceManagingObject&& other) noexcept {if (this == &other) return *this;delete[] data;data = other.data;size = other.size;other.data = nullptr;other.size = 0;return *this;}void display() const {if (data) {for (size_t i = 0; i < size; ++i) {std::cout << data[i] << " ";}std::cout << std::endl;} else {std::cout << "No data." << std::endl;}}
};int main() {ResourceManagingObject obj1(5);std::cout << "Original object: ";obj1.display();// 拷贝构造ResourceManagingObject obj2 = obj1;std::cout << "Copied object: ";obj2.display();// 移动构造ResourceManagingObject obj3 = std::move(obj1);std::cout << "Moved object: ";obj3.display();std::cout << "Original object after move: ";obj1.display();return 0;
}
代码解释:
ResourceManagingObject
类管理一个动态分配的整数数组。- 构造函数:根据大小分配内存并初始化数组元素。
- 析构函数:释放动态分配的内存。
- 拷贝构造函数和拷贝赋值运算符:深拷贝资源,避免浅拷贝导致的资源共享和潜在的内存问题。
- 移动构造函数和移动赋值运算符:将资源从源对象移动到目标对象,源对象放弃资源所有权。
相关文章:
如何实现序列化和反序列化?如何处理对象的生命周期管理?
序列化和反序列化 实现思路: 序列化:将对象的状态信息转换为可以存储或传输的格式,通常是字节流。 确定要序列化的对象的数据成员。将这些数据成员按照一定的规则(如二进制、文本、JSON、XML 等)编码为字节序列。将生…...

WPF+MVVM案例实战与特效(三十八)- 封装一个自定义的数字滚动显示控件
文章目录 1、运行效果2、案例实现1、功能设计2、页面布局3、控件使用4、运行效果3、拓展:多数字自定义控件1、控件应用4、总结1、运行效果 在Windows Presentation Foundation (WPF)应用程序中,自定义控件允许开发者创建具有特定功能和外观的独特UI元素。本博客将介绍一个名…...
docker安装Redis、docker使用Redis、docker离线安装redis、Redis离线安装
服务器到期了,换了一个新的环境要重搭,就记录一下好了: -----docker在线安装Redis 拉取 Redis 镜像 docker pull redis:6.2 运行 Redis 容器 docker run --name redis -d redis docker run --name redis -d redis:6.2 映射端口 docker run -…...
单目动态新视角合成
目录 单目动态新视角合成 Generative Camera Dolly:Extreme Monocular Dynamic Novel View Synthesis 单目动态新视角合成 Generative Camera Dolly: Extreme Monocular Dynamic Novel View Synthesis Generative Camera Dolly: Extreme Monocular Dynamic Novel View Synth…...

STM32--IO引脚复用
IO引脚复用...

Python字符串及正则表达式(十):字符串常用操作、字符串编码转换
前言:在编程的世界里,字符串无处不在。它们是构建用户界面、存储数据、进行通信的基础元素。无论是财务系统的总账报表、电子游戏的比赛结果,还是火车站的列车时刻表,这些信息最终都需要以文本的形式呈现给用户。这些文本的背后&a…...
前端的Python入门指南(完):错误和异常处理策略及最佳实践
《前端的 Python 入门指南》系列文章: (一):常用语法和关键字对比(二):函数的定义、参数、作用域对比(三):数据类型对比 - 彻底的一切皆对象实现和包装对象异…...
LeetCode 2475 数组中不等三元组的数目
问题描述: 给定一个下标从 0 开始的正整数数组 nums,我们的目标是找出并统计满足下述条件的三元组 (i, j, k) 的数目: 0 < i < j < k < nums.length,这确保了三元组索引的顺序性。nums[i]、nums[j] 和 nums[k] 两…...

【和春笋一起学C++】字符串比较
目录 C语言字符串比较 C语言字符比较 C字符串比较 C语言字符串比较 在C语言中用于比较字符串的函数为strcmp函数,该函数定义在头文件<string.h>中,是一个标准库函数。strcmp函数的工作原理是逐字符比较两个字符串,直到找到不同的字符…...

HTTP 协议报文结构 | 返回状态码详解
注:本文为 “HTTP 历史 | 协议报文结构 | 返回状态码” 相关文章合辑。 未整理去重。 HTTP 历史 wangjunliang 最后更新: 2024/3/16 上午10:29 超文本传输协议(英语:HyperTextTransferProtocol,缩写:HTTP)是 万维网(World Wide Web)的基础协议。自 蒂姆…...

.net winform 实现CSS3.0 泼墨画效果
效果图 代码 private unsafe void BlendImages1(Bitmap img1, Bitmap img2) {// 确定两个图像的重叠区域Rectangle rect new Rectangle(0, 0,Math.Min(img1.Width, img2.Width),Math.Min(img1.Height, img2.Height));// 创建输出图像,尺寸为重叠区域大小Bitmap b…...

LearnOpenGL学习(高级OpenGL - - 实例化,抗锯齿)
实例化 对于在同一场景中使用相同顶点数据的对象(如草地中的草),可以使用实例化(Instancing)技术,用一个绘制函数让OpenGL绘制多个物体,而非循环(Drawcall: N->1)。 …...

大数据与AI:从分析到预测的跃迁
引言:数据时代的新纪元 从每天的社交分享到企业的运营决策,数据早已成为现代社会不可或缺的资源。我们正置身于一个数据爆炸的时代,数以亿计的信息流实时生成,为人类带来了前所未有的洞察能力。然而,数据的价值并不仅限…...

【CC2530开发基础篇】继电器模块使用
一、前言 1.1 开发背景 本实验通过使用CC2530单片机控制继电器的吸合与断开,深入了解单片机GPIO的配置与应用。继电器作为一种常见的电气控制元件,广泛用于自动化系统中,用于控制大功率负载的开关操作。在本实验中,将通过GPIO口…...
C05S07-Tomcat服务架设
一、Tomcat 1. Tomcat概述 Tomcat也是一个Web应用程序,具有三大核心功能。 Java Servlet:Tomcat是一个Servlet容器,负责管理和执行Java Servlet、服务端的Java程序,处理客户端的HTTP请求和响应。Java Server:服务端…...
Java stream groupingBy sorted 实现多条件排序与分组的最佳实践
1. 数据初始化 这一部分代码用于创建 Product 对象并将它们添加到 result 列表中。 // 初始化数据 List<Product> result new ArrayList<>(); List<Product> resp new ArrayList<>();// 添加产品数据 result.add(new Product("手机A", 1…...

JAVA:代理模式(Proxy Pattern)的技术指南
1、简述 代理模式(Proxy Pattern)是一种结构型设计模式,用于为其他对象提供一种代理,以控制对这个对象的访问。通过代理模式,我们可以在不修改目标对象代码的情况下扩展功能,满足特定的需求。 设计模式样例:https://gitee.com/lhdxhl/design-pattern-example.git 2、什…...

爬取Q房二手房房源信息
文章目录 1. 实战概述2. 网站页面分析3. 编写代码爬取Q房二手房房源信息3.1 创建项目与程序3.2 运行程序,查看结果 4. 实战小结 1. 实战概述 本次实战项目旨在通过编写Python爬虫程序,抓取深圳Q房网上的二手房房源信息。我们将分析网页结构,…...

Ansible自动化运维(五) 运维实战
Ansible自动化运维这部分我将会分为五个部分来为大家讲解 (一)介绍、无密钥登录、安装部署、设置主机清单 (二)Ansible 中的 ad-hoc 模式 模块详解(15)个 (三)Playbook 模式详解 …...
K-means算法的python实现
K-means算法步骤 初始化质心:输入初始的质心位置。分配样本:将每个数据点分配到离它最近的质心对应的簇中。更新质心:对每个簇中的所有数据点,计算它们的均值,并将均值更新为新的质心。重复步骤2和3,直到质…...

多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度
一、引言:多云环境的技术复杂性本质 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时,基础设施的技术债呈现指数级积累。网络连接、身份认证、成本管理这三大核心挑战相互嵌套:跨云网络构建数据…...
【网络】每天掌握一个Linux命令 - iftop
在Linux系统中,iftop是网络管理的得力助手,能实时监控网络流量、连接情况等,帮助排查网络异常。接下来从多方面详细介绍它。 目录 【网络】每天掌握一个Linux命令 - iftop工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景…...

Cinnamon修改面板小工具图标
Cinnamon开始菜单-CSDN博客 设置模块都是做好的,比GNOME简单得多! 在 applet.js 里增加 const Settings imports.ui.settings;this.settings new Settings.AppletSettings(this, HTYMenusonichy, instance_id); this.settings.bind(menu-icon, menu…...

ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...
C++八股 —— 单例模式
文章目录 1. 基本概念2. 设计要点3. 实现方式4. 详解懒汉模式 1. 基本概念 线程安全(Thread Safety) 线程安全是指在多线程环境下,某个函数、类或代码片段能够被多个线程同时调用时,仍能保证数据的一致性和逻辑的正确性…...

MFC 抛体运动模拟:常见问题解决与界面美化
在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...

【Linux】Linux 系统默认的目录及作用说明
博主介绍:✌全网粉丝23W,CSDN博客专家、Java领域优质创作者,掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域✌ 技术范围:SpringBoot、SpringCloud、Vue、SSM、HTML、Nodejs、Python、MySQL、PostgreSQL、大数据、物…...

系统掌握PyTorch:图解张量、Autograd、DataLoader、nn.Module与实战模型
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文通过代码驱动的方式,系统讲解PyTorch核心概念和实战技巧,涵盖张量操作、自动微分、数据加载、模型构建和训练全流程&#…...

Java数组Arrays操作全攻略
Arrays类的概述 Java中的Arrays类位于java.util包中,提供了一系列静态方法用于操作数组(如排序、搜索、填充、比较等)。这些方法适用于基本类型数组和对象数组。 常用成员方法及代码示例 排序(sort) 对数组进行升序…...
CppCon 2015 学习:REFLECTION TECHNIQUES IN C++
关于 Reflection(反射) 这个概念,总结一下: Reflection(反射)是什么? 反射是对类型的自我检查能力(Introspection) 可以查看类的成员变量、成员函数等信息。反射允许枚…...