当前位置: 首页 > news >正文

用 bsdtar 做 Linux 全系统迁移 - 最省空间、最灵活的Linux系统迁移方式,但需要那么一点点技巧

(首发地址:学习日记 https://www.learndiary.com/2024/03/migrate-linux-with-bsdtar/ )

我们在做 Linux 全系统迁移的时候,可以直接备份磁盘或分区(如 dd ),也可以备份全部文件(如 tar )。前者在硬盘分区一样的情况下比较方便,但备份文件尺寸较大,并且目的磁盘不得小于源磁盘。备份全部文件再恢复的方法比较灵活,备份文件尺寸跟实际文件多少有关,可以很小,但需要额外恢复引导和修改挂载参数,稍微复杂一点。本文在VirtualBox 虚拟机环境中,以把默认安装的 Deepin 20.9 Linux(磁盘 64G)迁移到另一台 16G 磁盘的虚拟机中为示范,演示一下使用 BSD 版 tar,即 bsdtar 作 Linux 全系统迁移的基本步骤。演示视频链接:【用 bsdtar 做 Linux 全系统迁移 - 最省空间、最灵活的Linux系统迁移方式,但需要那么一点点技巧】 https://www.bilibili.com/video/BV1Rj421d7D3/

用 bsdtar 做 Linux 全系统迁移 - 最省空间、最灵活的Linux系统迁移方式,但需要那么一点点技巧

之所以不用 GNU tar 而使用 bsdtar,是因为 bsdtar 保留文件特殊属性方面更全面和简便,具体参见我前面关于 GNU tar 和 bsdtar 在保留文件特殊属性方面的测试日记(参考链接1、2)。

系统最好是在关机状态下备份,避免在备份过程中数据发生变化。

选择跟源系统一样版本或差不多版本的 livecd 启动系统进行备份和恢复,从而完成全系统迁移。我这里使用的是 Deepin 官方的“deepin-live-system-2.0-amd64.iso”。

在源系统上进入 livecd 环境后,我们先在 live 环境中安装 bsdtar 用于备份操作。安装 openssh-server,并为 live 环境的 deepin 用户设置密码用于远程传输备份文件。
命令如下:

deepin@deepin:~$ sudo apt update; sudo apt install bsdtar openssh-server -y
deepin@deepin:~$ sudo passwd deepin
deepin@deepin:~$ 

查看一下源系统的分区情况:

deepin@deepin:~$ lsblk -f /dev/sda
NAME   FSTYPE LABEL     UUID                                 MOUNTPOINT
sda                                                      
├─sda1 vfat   EFI       3726-096F                        
├─sda2 ext4   Boot      536ccefb-fda4-4d7b-b453-234bcc51ed57 /media/deepin/Boot
├─sda3 ext4   Roota     da61922c-599a-4104-8fee-531529ff4a0c /media/deepin/Roota
├─sda4 ext4   Rootb     62675cba-483e-41a9-8490-3a0899547167 /media/deepin/Rootb
├─sda5 ext4   _dde_data 5fb706dc-3c05-4836-9883-2e6afc965aed /media/deepin/_dde_
├─sda6 ext4   Backup    7528aaf1-a3b3-459e-b94e-5f96a5602069 /media/deepin/Backu
└─sda7 swap   SWAP      98a4522b-a822-4a2e-891e-af4e7349c0ac 
deepin@deepin:~$ 

为了简单起见,我们不迁移 Deepin 默认安装时自身用于数据备份恢复的辅助分区和交换分区 sda4、sda6、sda7。先把需要数据迁移的 sda3、sda2、sda1 和 sda5 挂载到 live 环境的 /mnt 目录下。注意,必须按源系统目录层次结构按顺序进行分区挂载,比如:/、/boot、/boot/efi。

deepin@deepin:~$ sudo mount /dev/sda3 /mnt -v
mount: /dev/sda3 mounted on /mnt.
deepin@deepin:~$ sudo mount /dev/sda2 /mnt/boot -v
mount: /dev/sda2 mounted on /mnt/boot.
deepin@deepin:~$ sudo mount /dev/sda1 /mnt/boot/efi -v
mount: /dev/sda1 mounted on /mnt/boot/efi.
deepin@deepin:~$ sudo mount /dev/sda5 /mnt/data -v
mount: /dev/sda5 mounted on /mnt/data.
deepin@deepin:~$ cd /mnt
deepin@deepin:/mnt$ ls
bin   dev   lib    libx32      mnt   recovery  sbin  tmp
boot  etc   lib32  lost+found  opt   root      srv   usr
data  home  lib64  media       proc  run       sys   var
deepin@deepin:/mnt$ sudo df -h
文件系统        容量  已用  可用 已用% 挂载点
udev            2.0G     0  2.0G    0% /dev
tmpfs           395M  5.6M  389M    2% /run
/dev/sr0        385M  385M     0  100% /lib/live/mount/medium
/dev/loop0      303M  303M     0  100% /lib/live/mount/rootfs/filesystem.squashfs
tmpfs           2.0G     0  2.0G    0% /lib/live/mount/overlay
overlay         2.0G  212M  1.8G   11% /
tmpfs           2.0G     0  2.0G    0% /dev/shm
tmpfs           5.0M     0  5.0M    0% /run/lock
tmpfs           2.0G     0  2.0G    0% /sys/fs/cgroup
tmpfs           2.0G  4.0K  2.0G    1% /tmp
tmpfs           395M   16K  395M    1% /run/user/1000
/dev/sda6        11G  7.5G  2.8G   74% /media/deepin/Backup
/dev/sda3        15G  6.7G  7.3G   48% /mnt
/dev/sda4        15G   41M   14G    1% /media/deepin/Rootb
/dev/sda2       1.5G  218M  1.2G   16% /mnt/boot
/dev/sda5        15G  573M   14G    4% /mnt/data
/dev/sda1       300M  9.1M  291M    4% /mnt/boot/efi
deepin@deepin:/mnt$ 

下面对系统进行备份:

deepin@deepin:/mnt$ time sudo bsdtar --exclude=sysbak.btar.gz -czvf sysbak.btar.gz *
a bin
a boot
a boot/vmlinuz-5.15.77-amd64-desktop
(略...)
a var/cache/fontconfig/39a7f1b3-6a3c-4b85-a424-827e510fa6d6-le64.cache-7
a var/cache/fontconfig/288c2344-6f48-412f-b675-b58dc4683ee2-le64.cache-7
a var/cache/fontconfig/8f32e591-21ad-45e9-8f64-16eeb77322bf-le64.cache-7real    5m43.539s
user    3m54.286s
sys     0m59.084s
deepin@deepin:/mnt$ ls -lh sysbak.btar.gz
-rw-r--r-- 1 root root 3.3G 2月  28 16:21 sysbak.btar.gz
deepin@deepin:/mnt$ 

建一个16G磁盘的目的虚拟机,用 deepin livecd 启动系统。

安装 ssh、bsdtar:

deepin@deepin:/mnt$ sudo apt install ssh libarchive-tools -y
deepin@deepin:/mnt$ 

用 gparted 软件创建用于 /boot/efi 的 FAT32 EFI 分区、用于 / 的 EXT4 系统分区、用于 /data 的 EXT4 数据分区。
分区信息如下:

deepin@deepin:~$ sudo parted /dev/sda print
Model: ATA VBOX HARDDISK (scsi)
Disk /dev/sda: 17.2GB
Sector size (logical/physical): 512B/512B
Partition Table: gpt
Disk Flags: Number  Start   End     Size    File system  Name  Flags1      1049kB  316MB   315MB   fat32        EFI   boot, esp2      316MB   13.2GB  12.9GB  ext4         ROOT3      13.2GB  17.2GB  3978MB  ext4         DATA
deepin@deepin:~$ 

创建目录并挂载分区:

deepin@deepin:~$ sudo mount /dev/sda2 /mnt -v
mount: /dev/sda2 mounted on /mnt.
deepin@deepin:~$ sudo mkdir /mnt/{boot/efi,data} -pv
mkdir: 已创建目录 '/mnt/boot'
mkdir: 已创建目录 '/mnt/boot/efi'
mkdir: 已创建目录 '/mnt/data'
deepin@deepin:~$ sudo mount /dev/sda1 /mnt/boot/efi -v
mount: /dev/sda1 mounted on /mnt/boot/efi.
deepin@deepin:~$ sudo mount /dev/sda3 /mnt/data -v
mount: /dev/sda3 mounted on /mnt/data.
deepin@deepin:~$ cd /mnt
deepin@deepin:/mnt$ ls
boot  data  lost+found
deepin@deepin:/mnt$ 

用 scp 把源系统创建的备份文件通过网络传输到目的系统挂载的 / 分区中。

deepin@deepin:/mnt$ sudo scp deepin@192.168.1.19:/mnt/sysbak.btar.gz ./
The authenticity of host '192.168.1.19 (192.168.1.19)' can't be established.
ECDSA key fingerprint is SHA256:cLph0YP2SqmuXTbDMTZmAoedyhPNiQBmsdG9v6ngzn8.
Are you sure you want to continue connecting (yes/no)? yes
Warning: Permanently added '192.168.1.19' (ECDSA) to the list of known hosts.
deepin@192.168.1.19's password: 
sysbak.btar.gz                                100% 3320MB  87.6MB/s   00:37  
deepin@deepin:/mnt$ 

在目的系统中恢复备份系统:

deepin@deepin:/mnt$ time sudo bsdtar -xzvf sysbak.btar.gz
x bin
x boot/
x boot/vmlinuz-5.15.77-amd64-desktop
x boot/initrd.img-5.15.77-amd64-desktop
(略...)
x var/cache/fontconfig/39a7f1b3-6a3c-4b85-a424-827e510fa6d6-le64.cache-7
x var/cache/fontconfig/288c2344-6f48-412f-b675-b58dc4683ee2-le64.cache-7
x var/cache/fontconfig/8f32e591-21ad-45e9-8f64-16eeb77322bf-le64.cache-7real    1m20.898s
user    0m23.647s
sys     0m29.094s
deepin@deepin:/mnt$ 

挂载各种虚拟文件系统:

deepin@deepin:/mnt$ for i in /dev /dev/pts /proc /sys /sys/firmware/efi/efivars /run; do sudo mount -B $i /mnt$i -v; done
mount: /dev bound on /mnt/dev.
mount: /dev/pts bound on /mnt/dev/pts.
mount: /proc bound on /mnt/proc.
mount: /sys bound on /mnt/sys.
mount: /sys/firmware/efi/efivars bound on /mnt/sys/firmware/efi/efivars.
mount: /run bound on /mnt/run.
deepin@deepin:/mnt$ 

进入 chroot 环境修改 /etc/fstab 分区挂载参数,重装引导(参考链接5),更新 grub 菜单:

deepin@deepin:/mnt$ sudo chroot .
root@deepin:/# lsblk -f /dev/sda
NAME FSTYPE FSVER LABEL UUID                                 FSAVAIL FSUSE% MOUNTPOINTS
sda                                                                 
├─sda1
│    vfat   FAT32       017D-4B18                             290.4M     3% /boot/efi
├─sda2
│    ext4   1.0         66f14f2f-405b-4abb-b33f-76a48406e4f6      1G    86% /
└─sda3ext4   1.0         c0ffaf7c-5a55-4b99-ba09-e81f3dd4e0b4    2.9G    15% /data
root@deepin:/# sudo cp /etc/fstab /etc/fstab.orig -v
'/etc/fstab' -> '/etc/fstab.orig'
root@deepin:/# vim /etc/fstab # 或者在 livecd 环境中执行 sudo gedit /mnt/etc/fstab 修改
root@deepin:/# cat /etc/fstab
# /dev/sda2
UUID=66f14f2f-405b-4abb-b33f-76a48406e4f6       /               ext4           rw,relatime      0 1# /dev/sda1
UUID=017D-4B18          /boot/efi       vfat            rw,relatime,fmask=0022,dmask=0022,codepage=437,iocharset=iso8859-1,shortname=mixed,utf8,errors=remount-ro       0 2# /dev/sda3
UUID=c0ffaf7c-5a55-4b99-ba09-e81f3dd4e0b4       /data           ext4           rw,relatime      0 2# /swapfile
# /swapfile       none            swap            defaults,pri=-2 0 0/data/home /home none defaults,bind 0 0
/data/opt /opt none defaults,bind 0 0
/data/root /root none defaults,bind 0 0
/data/var /var none defaults,bind 0 0
root@deepin:/# grub-install /dev/sda
get rootb uuid error: failed to get "rootb" uuid
正在为 x86_64-efi 平台进行安装。
安装完成。没有报告错误。
root@deepin:/# update-grub
get rootb uuid error: failed to get "rootb" uuid
正在生成 grub 配置文件 ...
找到主题:/boot/grub/themes/deepin-fallback/theme.txt
Found background image: /boot/grub/themes/deepin-fallback/background.jpg
找到 Linux 镜像:/boot/vmlinuz-5.15.77-amd64-desktop
找到 initrd 镜像:/boot/initrd.img-5.15.77-amd64-desktop
Found  image: /boot/vmlinuz-5.15.77-amd64-desktop
找到 initrd 镜像:/boot/initrd.img-5.15.77-amd64-desktop
Adding boot menu entry for EFI firmware configuration
完成
root@deepin:/# 

重启系统:

root@deepin:/# exit
deepin@deepin:/mnt$ sudo reboot

使用 bsdtar 进行系统迁移的基本步骤演示完毕。实际中的系统迁移可能还会涉及修改网络配置,安装新硬件驱动等其他问题。这些都需要根据实际情况处理。

参考链接:
1、Linux tar 保留文件特殊属性使用小结:https://www.learndiary.com/2024/03/gnu-tar/
2、bsdtar 归档程序在保留文件特殊属性上比 GNU tar 更全面和简便:https://www.learndiary.com/2024/03/bsdtar/
3、Backup Your System with TAR:https://help.ubuntu.com/community/BackupYourSystem/TAR
4、使用 bsdtar 完整备份/还原 Linux 系统:https://www.mivm.cn/linux-full-backup-with-bsdtar
5、Grub EFI Reinstall:https://wiki.debian.org/GrubEFIReinstall

相关文章:

用 bsdtar 做 Linux 全系统迁移 - 最省空间、最灵活的Linux系统迁移方式,但需要那么一点点技巧

(首发地址:学习日记 https://www.learndiary.com/2024/03/migrate-linux-with-bsdtar/ ) 我们在做 Linux 全系统迁移的时候,可以直接备份磁盘或分区(如 dd ),也可以备份全部文件(如…...

【模拟string函数的实现】

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 目录 前言 模拟string函数的实现 浅拷贝 深拷贝 vs和g下string结构的说明 总结 前言 模拟string函数的实现 浅拷贝 深拷贝 总结 前言 世上有两种耀眼的光芒&#…...

智能合约开发基础知识:最小信任机制、智能合约、EVM

苏泽 大家好 这里是苏泽 一个钟爱区块链技术的后端开发者 本篇专栏 ←持续记录本人自学两年走过无数弯路的智能合约学习笔记和经验总结 如果喜欢拜托三连支持~ 专栏的前面几篇详细了介绍了区块链的核心基础知识 有兴趣学习的小伙伴可以看看http://t.csdnimg.cn/fCD5E关于区块…...

程序人生——Java泛型和反射的使用建议

目录 引出泛型和反射建议93:Java的泛型是类型擦除的建议94:不能初始化泛型参数和数组建议95:强制声明泛型的实际类型 建议96:不同的场景使用不同的泛型通配符建议97:警惕泛型是不能协变和逆变的 建议98:建议…...

JavaSE-----认识异常【详解】

目录 一.异常的概念与体系结构: 1.1异常的概念: 1.2一些常见的异常: 1.3异常的体系结构: 1.4异常的分类: 二.异常的处理机制: 2.1 抛出异常: 2.2异常的捕获: 2.3try-catch-&…...

【机器学习300问】34、决策树对于数值型特征如果确定阈值?

还是用之前的猫狗二分类任务举例(这个例子出现在【机器学习300问】第33问中),我们新增一个数值型特征(体重),下表是数据集的详情。如果想了解更多决策树的知识可以看看我之前的两篇文章: 【机器…...

计算机二级(Python)真题讲解每日一题:《绘制雪花》

在横线处填写代码,完成如下功能‪‬‪‬‪‬‪‬‪‬‮‬‪‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‭‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‭‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬…...

Rust 的 Arc<Mutex<T>> 的用法示例源代码

在 Rust 中&#xff0c;Arc<Mutex<T>> 是一种组合类型&#xff0c;它结合了 Arc&#xff08;原子引用计数&#xff09;和 Mutex&#xff08;互斥锁&#xff09;。Arc 用于在多个所有者之间共享数据&#xff0c;而 Mutex 用于确保在任意时刻只有一个线程可以访问被保…...

【NR 定位】3GPP NR Positioning 5G定位标准解读(十六)-UL-AoA 定位

前言 3GPP NR Positioning 5G定位标准&#xff1a;3GPP TS 38.305 V18 3GPP 标准网址&#xff1a;Directory Listing /ftp/ 【NR 定位】3GPP NR Positioning 5G定位标准解读&#xff08;一&#xff09;-CSDN博客 【NR 定位】3GPP NR Positioning 5G定位标准解读&#xff08;…...

如何理解闭包

闭包是编程语言中一个重要的概念,特别是在函数式编程中常常会遇到。以下是对闭包的理解: 1. 定义: 闭包是一种函数,它引用了在其定义范围之外的自由变量(非全局变量),并且这些引用的变量在函数被调用时仍然保持活跃状态。2. 构成: 闭包通常由两部分组成:内部函数(函…...

python知识点总结(一)

这里写目录标题 一、什么是WSGI,uwsgi,uWSGI1、WSGI2、uWSGI3、uwsgi 二、python中为什么没有函数重载&#xff1f;三、Python中如何跨模块共享全局变量?四、内存泄露是什么?如何避免?五、谈谈lambda函数作用?六、写一个函数实现字符串反转&#xff0c;尽可能写出你知道的所…...

【Poi-tl Documentation】区块对标签显示隐藏改造

前置说明&#xff1a; <dependency><groupId>com.deepoove</groupId><artifactId>poi-tl</artifactId><version>1.12.1</version> </dependency>模板&#xff1a; 删除行表格测试.docx 改造前测试效果 package run.siyuan…...

第十四届蓝桥杯 三国游戏

一开始的思路就是想着暴力&#xff0c;但是呢&#xff0c;如果真的用暴力一个一个列的话&#xff0c;连30%的数据都搞定不了&#xff0c;所以这里需要考虑别的办法。 这道题的思路就是贪心。 我们这样想&#xff1a;既然要满足至少一个国X>YZ&#xff0c;那么我们何不变成…...

数据结构——通讯录项目

1.通讯录的介绍 顺序表是通讯录的底层结构。 通讯录是将顺序表的类型替换成结构体类型来储存用户数据&#xff0c;通过运用顺序表结构来实现的。 用户数据结构&#xff1a; typedef struct PersonInfo {char name[12];char sex[10];int age;char tel[11];char addr[100]; }…...

学点Java打小工_Day4_数组_冒泡排序

1 数组基本概念 程序算法数据结构 算法&#xff1a;解决程序的流程步骤 数据结构&#xff1a;将数据按照某种特定的结构来存储 设计良好的数据结构会导致良好的算法。 ArrayList、LinkedList 数组是最简单的数据结构。 数组&#xff1a;存放同一种类型数据的集合&#xff0c;在…...

内存分配方式?

内存分配方式主要有三种&#xff1a; 静态存储区分配&#xff1a;这种方式在程序编译的时候就已经分配好内存&#xff0c;并且这块内存在程序的整个运行期间都存在。全局变量和静态变量通常就是在静态存储区分配的。这种分配方式效率高&#xff0c;因为内存在程序开始执行前就已…...

2024/3/17周报

文章目录 摘要Abstract文献阅读题目引言模型架构编码器和解码器堆栈AttentionPosition-wise Feed-Forward NetworksEmbeddings and SoftmaxPositional Encoding 实验数据实验结果 深度学习TransformerEncoderDecoder 总结 摘要 本周阅读了Transformer的开山之作《Attention Is…...

函数连续性和Lipschitz连续性

摘要&#xff1a; 直观上&#xff0c;Lipschitz连续性的含义是函数图像的变化速度有一个全局的上限&#xff0c;即函数的增长速率不会无限增加。这种性质确保了函数在任何地方都不会过于陡峭&#xff0c;有助于分析函数的行为&#xff0c;并且在优化、动力系统理论、机器学习等…...

Qt 鼠标滚轮示例

1.声明 void wheelEvent(QWheelEvent *event) override;2.实现&#xff08;方便复制、测试起见用静态变量&#xff09; #include <mutex> void MainWindow::wheelEvent(QWheelEvent *event) {static QLabel *label new QLabel("Zoom Level: 100%", this);st…...

【Unity】进度条和血条的三种做法

前言 在使用Unity开发的时候&#xff0c;进度条和血条是必不可少的&#xff0c;本篇文章将简单介绍一下几种血条的制作方法。 1.使用Slider Slider组件由两部分组成&#xff1a;滑动区域和滑块。滑动区域用于显示滑动条的背景&#xff0c;而滑块则表示当前的数值位置。用户可…...

多人聊天室 (epoll - Linux网络编程)

文章目录 零、效果展示一、服务器代码二、客户端代码三、知识点1.connect()2.socket()3.bind()4.send()5.recv() 四、改进方向五、跟练视频 零、效果展示 一个服务器作为中转站&#xff0c;多个客户端之间可以相互通信。至少需要启动两个客户端。 三个客户端互相通信 一、服务…...

vite配置

"vite": "^5.1.4" resolve.alias&#xff1a;配置别名 1、执行npm install -D types/node 或者 yarn add types/node -D 2、以下配置代表访问src时可以用“”代替 resolve: {alias: {"": path.resolve(__dirname, "./src"),},}, 使…...

服务器生产环境问题解决思路

游戏服务器开发节奏比较快,版本迭代很频繁,有一些项目甚至出现了周更新(每周准时停服更新维护)。由于功能开发时间短,研发人员本身技术能力等原因,线上出现bug很常见。笔者经历过的游戏项目,一年到头没几次更新不出现bug的(当然,配置问题也算bug)。那当出现bug,我们…...

鸿蒙Harmony应用开发—ArkTS声明式开发(容器组件:Column)

沿垂直方向布局的容器。 说明&#xff1a; 该组件从API Version 7开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 子组件 可以包含子组件。 接口 Column(value?: {space?: string | number}) 从API version 9开始&#xff0c;该接口…...

LLM之RAG实战(三十)| 探索RAG语义分块策略

在LLM之RAG实战&#xff08;二十九&#xff09;| 探索RAG PDF解析解析文档后&#xff0c;我们可以获得结构化或半结构化的数据。现在的主要任务是将它们分解成更小的块来提取详细的特征&#xff0c;然后嵌入这些特征来表示它们的语义&#xff0c;其在RAG中的位置如图1所示&…...

软件测试-------Web(性能测试 / 界面测试 / 兼容性测试 / 安全性测试)

Web&#xff08;性能测试 / 界面测试 / 兼容性测试 / 安全性测试&#xff09; 一、Web性能测试&#xff1a;&#xff08;压力测试、负载测试、连接速度测试&#xff09;1、压力测试&#xff1a;      并发测试 &#xff08;如500人同时登录邮箱&#xff09; 2、负载测试…...

工欲善其事,必先利其器,Markdown和Mermaid的梦幻联动(2)

该文章Github地址&#xff1a;https://github.com/AntonyCheng/typora-notes/tree/master/chapter03-mermaid 在此介绍一下作者开源的SpringBoot项目初始化模板&#xff08;Github仓库地址&#xff1a;https://github.com/AntonyCheng/spring-boot-init-template & CSDN文…...

STM32基础--使用寄存器点亮流水灯

GPIO 简介 GPIO 是通用输入输出端口的简称&#xff0c;简单来说就是 STM32 可控制的引脚&#xff0c;STM32 芯片的 GPIO 引脚与外部设备连接起来&#xff0c;从而实现与外部通讯、控制以及数据采集的功能。STM32 芯片的 GPIO被分成很多组&#xff0c;每组有 16 个引脚&#xf…...

代码随想录训练营Day25:● 216.组合总和III ● 17.电话号码的字母组合

216.组合总和III 题目链接 https://leetcode.cn/problems/combination-sum-iii/description/ 题目描述 思路 自己写的效率会慢一些&#xff0c;而且没有用到剪枝 class Solution {List<List<Integer>> list new ArrayList<>();List<Integer> lis…...

SwiftUI的 特性 - ViewModify

SwiftUI的 特性 - ViewModify 记录一下SwiftUI的 特性 - ViewModify的使用方式 可以通过viewModify来管理视图的样式&#xff0c;结合extension来完成封装达到解偶效果 import SwiftUI/// 我们可以通过viewModify来管理视图的样式&#xff0c;来达到解偶效果 struct DefaultB…...