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

K8S部署Java项目(Gitlab CI/CD自动化部署终极版)

天行健,君子以自强不息;地势坤,君子以厚德载物。


每个人都有惰性,但不断学习是好好生活的根本,共勉!


文章均为学习整理笔记,分享记录为主,如有错误请指正,共同学习进步。

文章目录

  • 一、平台搭建
    • 1. `K8S`搭建
    • 2. `Gitlab`代码仓库部署
    • 3. `Gitlab Runner`执行工具部署
    • 4. `Harbor`镜像仓库部署
      • 4.1 部署Harbor
      • 4.2 创建仓库项目
  • 二、`Java Springboot`项目上传到Gitlab
    • 1. 项目demo获取
    • 2. pom配置
    • 3. Dokcerfile和.gitlab-ci.yml配置
    • 4. Dokcerfile和.gitlab-ci.yml拓展信息
      • 4.1 Dockerfile文件配置
      • 4.2 .gitlab-ci.yml文件配置
    • 5. 上传项目到Gitlab
  • 三、CI功能打包、构建镜像
    • 1. 流程介绍
    • 2. 打包阶段
    • 3. 构建镜像阶段
    • 4. Harbor查看镜像
  • 四、部署镜像(项目)
    • 1. yaml文件配置
    • 2. 将文件上传到服务器
    • 3. 创建命名空间
    • 4. 部署yaml文件
      • 4.1 部署pvc
      • 4.2 部署deploy
      • 4.3 部署svc
    • 5. 查看服务
      • 5.1 查看pvc
      • 5.2 查看deploy
      • 5.3 查看svc
      • 5.4 查看pod
      • 5.5 查看logs
      • 5.6 查看describe
  • 五、访问验证
    • 1. 服务所在节点IP
    • 2. deploy配置的端口
    • 3. Java项目接口URL
    • 4. 完整访问URL
    • 5. 访问服务接口
      • 5.1 curl访问
      • 5.2 浏览器访问
  • 六、拓展(尚未成功,继续研究,仅供参考)
    • 1. 配置Ingress
    • 2. 访问服务
  • 七、报错汇总
    • 1. pod启动失败的查看命令
    • 2. 报错内容汇总


一、平台搭建

在K8S集群中,配合Gitlab、Harbor实现自动化部署Java springboot服务
首先需要搭建K8S集群、Gitlab及Gitlab Runner、Harbor,如已部署或有相同平台可忽略

1. K8S搭建

首先你要有一个K8S集群,如已搭建可跳过,没有可以参考链接搭建
K8S搭建(centos)完整版
k8s搭建(ubuntu)详细演示完整一篇

2. Gitlab代码仓库部署

用于管理项目代码和使用CI自动化工具
如已有可用环境可忽略,如无可参考安装使用
K8S部署GitLab(详细完整版)

3. Gitlab Runner执行工具部署

用于执行自动化脚本实现自动化打包、构建镜像等操作
如已有可用环境可忽略,如无可参考安装使用
gitlab runner 安装、注册、配置、使用(Docker部署)

4. Harbor镜像仓库部署

4.1 部署Harbor

用于存储自动化构建的镜像,后续部署服务会从仓库拉取
如已有可用环境可忽略,如无可参考安装使用
K8S部署Harbor镜像仓库(含离线安装包harbor-offline-installer国内下载链接)
K8S部署Harbor(三部曲之一:配置)
K8S部署Harbor(三部曲之二:部署)
K8S部署Harbor(三部曲之三:使用)

4.2 创建仓库项目

创建harbor镜像仓库项目用于存放推送的项目镜像,项目名为k8s-demo
在这里插入图片描述

二、Java Springboot项目上传到Gitlab

需要提前准备好Java项目,并上传到Gitlab代码仓库

1. 项目demo获取

如果没有现成的springboot项目可以从下面链接获取
下载:Java项目-基于Gitlab CI/CD功能实现自动化部署(在k8s中部署)
其中主要包含了如下文件:

  • 一个简单的请求接口用于后续验证部署是否成功
  • 构建镜像用的Dockerfile配置文件,构建项目镜像使用的文件
  • Gitlab的.gitlab-ci.yml配置文件,该文件用于执行自动化部署脚本等操作
  • 后续部署需要用到的yaml文件sb-pvc.yaml、sb-dplm.yaml、sb-svc.yaml、以及配置域名的sb-igs.yaml
  • 一些Java项目相关的文件

项目截图:
在这里插入图片描述

2. pom配置

需要在pom中添加maven插件依赖、maven插件并定义jar包名称(需要与springboot版本保持一致)

    <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><version>2.6.3</version></dependency></dependencies><build><!--配置后最终生成的jar包名称就是该元素中的名称指定package打包后生成的jar包文件名称为app.jar,package打包后会在target文件夹中看到app.jar文件--><finalName>app</finalName><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><version>2.6.3</version><executions><execution><goals><goal>repackage</goal></goals></execution></executions></plugin></plugins></build>

3. Dokcerfile和.gitlab-ci.yml配置

可根据Java项目情况参考文章配置这两个文件的内容
我是用的两个文件的模板连接如下:
三个都一样,推荐使用第一个
Dockerfile和.gitlab-ci.yml文件模板(标准版)
Dockerfile和.gitlab-ci.yml文件模板(精简无注释)
Dockerfile和.gitlab-ci.yml文件模板(含详细注释)

4. Dokcerfile和.gitlab-ci.yml拓展信息

下面是拓展介绍,可忽略

4.1 Dockerfile文件配置

Dockerfile文件配置参考:Dockerfile文件参数配置和使用

4.2 .gitlab-ci.yml文件配置

.gitlab-ci.yml文件配置参考:.gitlab-ci.yml文件参数配置和使用
.gitlab-ci.yml文件参数汇总及解释参考:.gitlab-ci.yml文件参数

5. 上传项目到Gitlab

配置文件配置好之后就可以将项目上传到Gitlab代码仓库中了
可参考下方链接文章,虽然是gitee但与gitlab相似
通过GIT将本地项目上传到gitee
当然前面搭建Gitlab的文章中也有介绍如何上传本地项目到Gitlab

三、CI功能打包、构建镜像

在上面我们将代码上传到Gitlab上以后即可触发CI/CD流程

1. 流程介绍

Gitlab检测到.gitlab-ci.yml文件会调用Gitlab Runner执行该文件中的脚本,进行流程化作业部署
执行过程会根据yml文件中的阶段进行,流程如下:

  • 打包
  • 构建镜像
  • 将镜像推送到镜像仓库
  • 在主节点服务器中执行kubectl命令(即部署yaml文件)部署项目
  • 后续更新镜像,yaml文件会根据镜像地址拉取新的镜像,实现服务自动部署

提交即部署,体验还不错

接下来是在gitlab中自动执行的操作

2. 打包阶段

job1_package执行过程
准备docker执行器—准备环境—仓库资源拉取—执行job的脚本—开始jar打包—下载依赖—打包完成
在这里插入图片描述
此处省略下载过程(略长)。。。。。。
在这里插入图片描述
job中的脚本执行完毕后会输出Job succeed表示这个job已经执行结束

3. 构建镜像阶段

job2_build执行过程
构建镜像阶段:先用docker命令登录harbor仓库–通过Dockerfile文件构建镜像放到本地—推送到harbor镜像仓库—删除本地镜像
在这里插入图片描述
在这里插入图片描述

构建镜像阶段任务完成
表示构建镜像并推送成功

4. Harbor查看镜像

此时可以到harbor镜像仓库中查看是否成功上传,可以看到已经存在
在这里插入图片描述
到此表示gitlabCI任务的打包、构建镜像两个阶段结束
以上两个阶段为gitalb自动化执行的文件脚本,接下来看需要部署yaml文件,但部署yaml只需要一次,后续更新镜像后会自动部署

四、部署镜像(项目)

根据Java项目情况配置三个yaml文件,其中deploy的yaml文件中镜像的拉取地址为上面自动化构建后推送的镜像仓库地址

1. yaml文件配置

可参考文章配置三个yaml文件用于部署Java项目:K8S部署Java项目的yaml配置文件模板

2. 将文件上传到服务器

需要将yaml文件上传到集群中的节点服务器中,我这里是可执行kubectl命令的主节点k8s-master
放到自己创建的文件夹springboot-yaml中
在这里插入图片描述

3. 创建命名空间

在主节点k8s-master中执行kubectl命令
创建命名空间用于管理部署的服务

kubectl create namespace ns-sb

在这里插入图片描述
查看所有命名空间

kubectl get namespace -A

在这里插入图片描述

4. 部署yaml文件

依次部署pvc文件、deploy文件、svc文件

4.1 部署pvc

kubectl apply -f springboot-yaml/sb-pvc.yaml

在这里插入图片描述

4.2 部署deploy

kubectl apply -f springboot-yaml/sb-dplm.yaml

在这里插入图片描述

4.3 部署svc

kubectl apply -f springboot-yaml/sb-svc.yaml

在这里插入图片描述

5. 查看服务

以下查看命令都可尝试在最后加上-o wide参数来查看更为详细的信息
注:由于部署过程中出现了很多问题,所以会删除部署的服务重新部署,截图中有些服务名称变了,请忽略该细节,最终的结果是成功的。然后遇到的问题也都总结在最后一章了,如遇到类似可供参考

5.1 查看pvc

kubectl get pvc -n ns-sb

刚开始时Pending
在这里插入图片描述
绑定后是Bound
在这里插入图片描述

5.2 查看deploy

kubectl get deployment -n ns-sb

在这里插入图片描述

5.3 查看svc

kubectl get svc -n ns-sb

在这里插入图片描述

5.4 查看pod

kubectl get pod -n ns-sb

在这里插入图片描述
还可以加-o wide参数来查看pod所在节点以及集群ip

kubectl get pod -n ns-sb -o wide

在这里插入图片描述

5.5 查看logs

查看pod的logs日志

kubectl logs sb-dplm-898c9564-v8nq7 -n ns-sb

在这里插入图片描述

5.6 查看describe

查看pod的describe描述

kubectl describe pod sb-dplm-898c9564-v8nq7 -n ns-sb

在这里插入图片描述

五、访问验证

部署完成后,进行访问(如果从节点未配置kubectl命令执行权限则需要再主节点进行操作)
IP+端口+接口请求

1. 服务所在节点IP

需要先确定我们部署的项目pod所在的服务器是哪个(可使用第四章3.5.4小节查看pod的第二个命令进行查看)
如查看我的pod所在位置为k8s-worker2节点,该节点的iP为173.33.0.222

2. deploy配置的端口

我们在部署的deploy配置文件中配置的端口参数即nodePort值为30089

3. Java项目接口URL

最后是我们Java项目的接口请求url为test/java

4. 完整访问URL

所以我们访问的url为

173.33.0.222:30089/test/java

5. 访问服务接口

5.1 curl访问

可在命令窗口通过curl执行get请求(接口注解为@GetMapping)

curl 173.33.0.222:30089/test/java

如图输出我们返回的内容就表示服务可正常访问,表示springboot在k8s中部署成功
在这里插入图片描述

5.2 浏览器访问

也可通过浏览器访问

173.33.0.222:30089/test/java

输出返回内容表示服务正常,部署成功
在这里插入图片描述

六、拓展(尚未成功,继续研究,仅供参考)

通常只有开发测试环境才会使用nodePort方式访问,生产环境需要使用更为安全的访问方法(LoadBalncer或者配置ingress)
我们配置ingress使用自己定义的域名进行访问

1. 配置Ingress

和sb-pvc.yaml、sb-dplm.yaml、sb-svc.yaml放在同一位置,我们可以直接使用vim命令创建并编辑Ingress文件

vim springboot-yaml/sb-igs.yaml

依旧可参考K8S部署Java项目的yaml配置文件模板中的ingress配置文件进行编辑,保存后部署
在这里插入图片描述

2. 访问服务

部署后可使用定义的域名进行访问

hs.sb.com/test/java

七、报错汇总

1. pod启动失败的查看命令

如果pod启动状态有问题,可执行以下两个命令进行检查
查看pod描述信息

kubectl describe pod podname -n namespacename

kubectl describe pod sb-dplm-898c9564-qm425 -n ns-sb

查看pod日志信息

kubectl logs podname -n namespacename

kubectl logs sb-dplm-898c9564-qm425 -n ns-sb

2. 报错内容汇总

以下为整个过程中遇到的问题以及解决方法,供大家参考,希望有所帮助

  • /usr/bin/bash: line 136: docker:command not found
    Gitlab CI/CD docker命令报错:/usr/bin/bash: line 136: docker:command not found
  • ERROR: Cannot connect to the Docker daemon at tcp://docker:2375. Is the docker running?
    Gitlab CI/CD ERROR: Cannot connect to the Docker daemon at tcp://docker:2375. Is the docker running?
  • error during connect: Post http://docker:2375/v1.40xxxxxx: dial tcp: lookup docker o
    Gitlab CI/CD报错: error during connect: Post http://docker:2375/v1.40xxxxxx: dial tcp: lookup docker o
  • ERROR: error during connect : Get “http://docker:2375/_ping“: dial tcp:lookup docker on 10
    Gitlab CI ERROR: error during connect : Get “http://docker:2375/_ping“: dial tcp:lookup docker on 10
  • /usr/bin/bash: line 137: kubectl: command not found
    Gitlab CI 报错: /usr/bin/bash: line 137: kubectl: command not found
  • Error: Unable to access jarfile app.jar
    K8S部署Java项目 pod的logs报错为:Error: Unable to access jarfile app.jar
  • Error: A JNI error has occurred, please check your installation and try again
    Error: A JNI error has occurred, please check your installation and try again
  • CrashLoopBackOff
    K8S部署Java项目(Springboot项目)pod状态:CrashLoopBackOff
  • cannot access org.springframework.context.ConfigurableApplicationContext bad class file: /root/.m2/r
    cannot access org.springframework.context.ConfigurableApplicationContext bad class file: /root/.m2/r
  • no main manifest attribute, in app.jar
    no main manifest attribute, in app.jar
  • no main manifest attribute, in app.jar备用方案,请现尝试上一个再用这个
    K8S部署Java项目 pod报错 logs日志内容:no main manifest attribute, in app.jar

感谢阅读,祝君暴富!

相关文章:

K8S部署Java项目(Gitlab CI/CD自动化部署终极版)

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…...

对Redis锁延期的一些讨论与思考

上一篇文章提到使用针对不同的业务场景如何合理使用Redis分布式锁&#xff0c;并引入了一个新的问题 若定义锁的过期时间是10s&#xff0c;此时A线程获取了锁然后执行业务代码&#xff0c;但是业务代码消耗时间花费了15s。这就会导致A线程还没有执行完业务代码&#xff0c;A线程…...

【高德地图】Android高德地图初始化定位并显示小蓝点

&#x1f4d6;第3章 初始化定位并显示小蓝点 ✅第1步&#xff1a;配置AndroidManifest.xml✅第2步&#xff1a;设置定位蓝点✅第3步&#xff1a;初始化定位✅完整代码 ✅第1步&#xff1a;配置AndroidManifest.xml 在application标签下声明Service组件 <service android:n…...

继电器测试中需要注意的安全事项有哪些?

继电器广泛应用于电气控制系统中的开关元件&#xff0c;其主要功能是在输入信号的控制下实现输出电路的断开或闭合。在继电器测试过程中&#xff0c;为了确保测试的准确性和安全性&#xff0c;需要遵循一定的安全事项。以下是在进行继电器测试时需要注意的安全事项&#xff1a;…...

Java向ES库中插入数据报错:I/O reactor status: STOPPED

Java向ES库中插入数据报错&#xff1a;java.lang.IllegalStateException: Request cannot be executed; I/O reactor status: STO 一、问题问题原因 二、解决思路 一、问题 在使用Java向ES库中插入数据时&#xff0c;第一次成功插入&#xff0c;第二次出现以下错误&#xff1a…...

vue3实现页面跳转

有需求是在vue项目中实现点击按钮完成页面跳转。这里不适用a标签&#xff0c;而是用vue自带的vue-router。 首先看一下项目结构 src │ App.vue │ main.js │ ├─router │ index.js │ └─views index.vue content.vue 可以看到&…...

【Linux运维系列】vim操作

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...

Centos服务器部署前后端项目

目录 准备工作1. 准备传输软件2. 连接服务器 部署Mysql1.下载Mysql(Linux版本)2. 解压3. 修改配置4. 启动服务另一种方法Docker 部署后端1. 在项目根目录中创建Dockerfile文件写入2. 启动 部署前端1. 在项目根目录中创建Dockerfile文件写入2. 启动 准备工作 1. 准备传输软件 …...

【初始RabbitMQ】延迟队列的实现

延迟队列概念 延迟队列中的元素是希望在指定时间到了之后或之前取出和处理消息&#xff0c;并且队列内部是有序的。简单来说&#xff0c;延时队列就是用来存放需要在指定时间被处理的元素的队列 延迟队列使用场景 延迟队列经常使用的场景有以下几点&#xff1a; 订单在十分…...

spark为什么比mapreduce快?

spark为什么比mapreduce快&#xff1f; 首先澄清几个误区&#xff1a; 1&#xff1a;两者都是基于内存计算的&#xff0c;任何计算框架都肯定是基于内存的&#xff0c;所以网上说的spark是基于内存计算所以快&#xff0c;显然是错误的 2;DAG计算模型减少的是磁盘I/O次数&…...

Unity通过XXpermission插件获取MANAGE_EXTERNAL_STORAGE权限

最近公司准备用Unity做一个安卓端的文件管理器功能&#xff0c;文件管理器已经做完了。刚开始的时候想要申请一下所有文件权限&#xff0c;发现在Unity里面申请所有文件权限(android.permission.MANAGE_EXTERNAL_STORAGE)相对来说比较麻烦。所以准备写一下文章记录一下如何申请…...

「连载」边缘计算(二十一)02-26:边缘部分源码(源码分析篇)

&#xff08;接上篇&#xff09; DeviceTwin struct组成剖析 该部分对DeviceTwin struct的组成进行剖析。接着devicetwin struct调用链剖析的实例化DeviceTwin struct&#xff08;dt : DeviceTwin{}&#xff09;往下剖析&#xff0c;进入DeviceTwin struct的定义&#xff0c;…...

Unity(第四部)新手组件

暴力解释就是官方给你的功能&#xff1b;作用的对象上面如&#xff1a; 创建一个球体&#xff0c;给这个球体加上重力 所有物体都是一个空物体&#xff0c;加上一些组件才形成了所需要的GameObject。 这是一个空物体&#xff0c;在Scene场景中没有任何外在表现&#xff0c;因为…...

【JS】【Vue3】【React】获取鼠标位置的方法:JavaScript、Vue 3和React示例

目录 使用JavaScript原生方法在Vue 3中获取鼠标位置在React中获取鼠标位置 随着Web应用程序的复杂性不断增加&#xff0c;获取用户交互信息变得越来越重要。其中&#xff0c;获取鼠标位置是一项常见的任务&#xff0c;可以用于实现各种交互效果&#xff0c;如拖拽、悬停提示等。…...

[Docker 教学] 常用的Docker 命令

Docker是一种流行的容器化技术。使用Docker可以将数据科学应用程序连同代码和所需的依赖关系打包成一个名为镜像的便携式工件。因此&#xff0c;Docker可以简化开发环境的复制&#xff0c;并使本地开发变得轻松。 以下是一些必备的Docker命令列表&#xff0c;这些命令将在你下一…...

小程序应用、页面、组件生命周期

引言 微信小程序生命周期是指在小程序运行过程中&#xff0c;不同阶段触发的一系列事件和函数。这一概念对于理解小程序的整体架构和开发流程非常重要。本文将介绍小程序生命周期的概念以及在不同阶段触发的关键事件&#xff0c;帮助开发者更好地理解和利用小程序的生命周期。 …...

Springboot中如何记录好日志

Springboot中如何记录日志 日志体系整体介绍 日志一直在系统中占据这十分重要的地位&#xff0c;他是我们在系统发生故障时用来排查问题的利器&#xff0c;也是我们做操作审计的重要依据。那么如何记录好日志呢&#xff1f;选择什么框架来记录日志&#xff0c;是不是日志打越…...

vm 虚拟机中ubuntu环境配置共享文件夹的方式

1. 在虚拟机设置中启用共享文件夹选项&#xff0c;映射到Windows中具体的目录。 2. 启动虚拟机。 3. 挂在cd #查看cd设备文件 sudo blkid#创建挂载点 sudo mkdir -p /media/cdrom#挂载cd sudo mount /dev/sr0 /media/cdrom#卸载cd sudo umount /media/cdrom 4. 执行完挂载后…...

EMQX Enterprise 5.5 发布:新增 Elasticsearch 数据集成

EMQX Enterprise 5.5.0 版本已正式发布&#xff01; 在这个版本中&#xff0c;我们引入了一系列新的功能和改进&#xff0c;包括对 Elasticsearch 的集成、Apache IoTDB 和 OpenTSDB 数据集成优化、授权缓存支持排除主题等功能。此外&#xff0c;新版本还进行了多项改进以及 B…...

安全架构设计理论与实践

一、考点分布 安全架构概述&#xff08;※※&#xff09;安全模型&#xff08;※※※&#xff09;信息安全整体架构设计网络安全体系架构设计区块链技术&#xff08;※※&#xff09; 二、安全架构概述 被动攻击&#xff1a;收集信息为主&#xff0c;破坏保密性 主动攻击&#…...

项目解决方案:食堂反浪费AI智能监督系统项目解决方案

目录 第一章 项目背景 1.1 国家粮食节约战略要求 1.2 传统管理模式局限性 1.3 AI技术成熟提供解决方案 1.4 先期成效验证 第二章 需求确认 2.1 实时监测与即时预警需求 2.2 多层级数据管理需求 2.3 AI识别精度与适应性需求 2.4 周期性分析报告需求 第三章 需要解决的…...

ExplorerPatcher深度解析:让Windows 11重获经典操作体验

ExplorerPatcher深度解析&#xff1a;让Windows 11重获经典操作体验 【免费下载链接】ExplorerPatcher This project aims to enhance the working environment on Windows 项目地址: https://gitcode.com/GitHub_Trending/ex/ExplorerPatcher ExplorerPatcher是一款功能…...

从基础到进阶:最短路径辅助的快速扫描法(SPAFSM)在复杂介质走时计算中的精度提升实践

1. 快速扫描法(FSM)基础与原理解析 快速扫描法(Fast Sweeping Method, FSM)是地震波走时计算中的经典算法&#xff0c;由Zhao在2005年首次提出。它的核心思想是通过有限差分法求解程函方程(Eikonal equation)&#xff0c;模拟地震波在地下介质中的传播时间。想象一下&#xff0…...

从BD4954到PMOS管:拆解一个真实物联网产品的太阳能充电管理电路,附完整PCB布局建议

从BD4954到PMOS管&#xff1a;拆解一个真实物联网产品的太阳能充电管理电路&#xff0c;附完整PCB布局建议 在低功耗物联网设备的设计中&#xff0c;电源管理系统的可靠性往往决定了产品的成败。我曾参与开发一款户外环境监测终端&#xff0c;设备需要在零下20℃至60℃的温度范…...

蔚蓝档案自动化脚本:从手动肝游到智能托管的技术革命

蔚蓝档案自动化脚本&#xff1a;从手动肝游到智能托管的技术革命 【免费下载链接】blue_archive_auto_script 支持按轴凹总力战, 无缝制造三解, 用于实现蔚蓝档案自动化的程序( Steam已适配 ) 项目地址: https://gitcode.com/gh_mirrors/bl/blue_archive_auto_script 每…...

1.6T 光模块的能效革命

合作核心与产品规格合作双方&#xff1a;光子技术提供商 Sivers Semiconductors 工程制造服务商 Jabil。核心产品&#xff1a;1.6T 线性接收光收发模块。关键技术&#xff1a;集成 Sivers 的高性能分布式反馈激光器。目标应用&#xff1a;下一代超大规模 AI 数据中心的光互连。…...

ZYNQ实战:手把手教你用LWIP实现UDP文件传输到DDR(附完整代码)

ZYNQ LWIP UDP文件传输实战&#xff1a;从协议栈配置到DDR存储的完整实现 在嵌入式系统开发中&#xff0c;网络通信功能已成为现代SoC设计的标配能力。Xilinx ZYNQ系列凭借其ARM处理器与可编程逻辑的完美结合&#xff0c;为开发者提供了灵活高效的网络通信解决方案。本文将深入…...

Unity Shader 梯度噪声 vs 值噪声

▦值噪声Value Noise插值随机标量值&#xff0c;生成速度快但有明显块状感和人工痕迹&#xff0c;适合低端设备或不需要高视觉质量的大面积纹理。◈梯度噪声Perlin Noise基于随机梯度方向插值&#xff0c;各向同性更均匀&#xff0c;平滑自然无方向性伪像&#xff0c;是程序化纹…...

别再被SystemExit: 2搞懵了!Python argparse在Jupyter Notebook里的正确打开方式

别再被SystemExit: 2搞懵了&#xff01;Python argparse在Jupyter Notebook里的正确打开方式 如果你曾在Jupyter Notebook中尝试运行一个包含argparse模块的Python脚本&#xff0c;大概率会遇到那个令人困惑的SystemExit: 2错误。这个看似简单的报错背后&#xff0c;隐藏着命令…...

别再傻傻转存了!5分钟搞懂Base64图片体积计算与优化技巧(附Python/JS代码)

Base64图片体积计算的科学原理与高效优化策略 在当今数字化时代&#xff0c;Base64编码图片作为数据嵌入方案被广泛应用于网页开发、移动应用和数据传输场景。然而&#xff0c;许多开发者对Base64编码后体积膨胀的机制存在误解&#xff0c;导致资源浪费和性能瓶颈。本文将深入解…...