Python实现火柴人的设计与实现
1.引言
火柴人(Stick Figure)是一种极简风格的图形,通常由简单的线段和圆圈组成,却能生动地表达人物的姿态和动作。火柴人不仅广泛应用于动画、漫画和涂鸦中,还可以作为图形学、人工智能等领域的教学和研究工具。本文旨在介绍如何使用Python实现火柴人的设计与绘制,通过编程的方式,让读者了解火柴人背后的基本原理和实现方法。
2.准备工作
在开始实现火柴人之前,你需要确保已经安装了Python环境,并且熟悉基本的Python编程知识。此外,为了绘制图形,我们将使用matplotlib库,这是一个强大的绘图库,适用于生成各种静态、动态和交互式的图表。
你可以通过以下命令安装matplotlib:
bash复制代码
pip install matplotlib
3.基础理论知识
火柴人的绘制主要依赖于几何图形的绘制和变换。具体来说,我们需要:
(1)定义关节:火柴人的关节包括头部、肩膀、肘部、手腕、臀部、膝盖和脚踝等。这些关节可以看作二维或三维空间中的点。
(2)绘制线段:根据关节的位置,绘制连接关节的线段,这些线段构成了火柴人的骨骼。
(3)添加圆形:在头部等关节处添加圆形,以表示关节。
(4)变换与动画:通过变换关节的位置,可以实现火柴人的动作和动画效果。
4.步骤详解
下面,我们将逐步介绍如何使用Python和matplotlib绘制火柴人。
(1)导入库
首先,我们需要导入matplotlib库中的pyplot模块:
import matplotlib.pyplot as plt
import numpy as np
(2)定义关节位置
为了简单起见,我们先在二维平面上定义火柴人的关节位置。这里以一个简单的火柴人站立姿势为例:
# 定义关节位置
head = [0, 1]
torso = [0, 0]
left_shoulder = [-0.5, 0]
left_elbow = [-1, -0.5]
left_hand = [-1, -1]
right_shoulder = [0.5, 0]
right_elbow = [1, -0.5]
right_hand = [1, -1]
left_hip = [-0.5, -0.5]
left_knee = [-1, -1.5]
left_foot = [-1, -2]
right_hip = [0.5, -0.5]
right_knee = [1, -1.5]
right_foot = [1, -2] # 将关节位置存储在一个字典中
joints = { 'head': head, 'torso': torso, 'left_shoulder': left_shoulder, 'left_elbow': left_elbow, 'left_hand': left_hand, 'right_shoulder': right_shoulder, 'right_elbow': right_elbow, 'right_hand': right_hand, 'left_hip': left_hip, 'left_knee': left_knee, 'left_foot': left_foot, 'right_hip': right_hip, 'right_knee': right_knee, 'right_foot': right_foot
}
(3)绘制火柴人
接下来,我们编写一个函数,根据关节位置绘制火柴人:
def draw_stick_figure(joints, ax): # 绘制身体 body_parts = [ ('torso', 'head'), ('torso', 'left_shoulder'), ('left_shoulder', 'left_elbow'), ('left_elbow', 'left_hand'), ('torso', 'right_shoulder'), ('right_shoulder', 'right_elbow'), ('right_elbow', 'right_hand'), ('torso', 'left_hip'), ('left_hip', 'left_knee'), ('left_knee', 'left_foot'), ('torso', 'right_hip'), ('right_hip', 'right_knee'), ('right_knee', 'right_foot') ] for start, end in body_parts: start_pos = np.array(joints[start]) end_pos = np.array(joints[end]) ax.plot([start_pos[0], end_pos[0]], [start_pos[1], end_pos[1]], 'k-') # 绘制头部 circle = plt.Circle(joints['head'], 0.1, color='black', fill=True) ax.add_patch(circle) # 绘制手部(可选) circle = plt.Circle(joints['left_hand'], 0.05, color='black', fill=True) ax.add_patch(circle) circle = plt.Circle(joints['right_hand'], 0.05, color='black', fill=True) ax.add_patch(circle) # 绘制脚部(可选) circle = plt.Circle(joints['left_foot'], 0.05, color='black', fill=True) ax.add_patch(circle) circle = plt.Circle(joints['right_foot'], 0.05, color='black', fill=True) ax.add_patch(circle)
(4)绘制并显示图形
最后,我们创建一个图形对象,调用绘制函数,并显示结果:
def main(): fig, ax = plt.subplots() ax.set_aspect('equal') ax.axis('off') # 关闭坐标轴 draw_stick_figure(joints, ax) plt.show() if __name__ == "__main__": main()
5.常见问题解答
(1)火柴人看起来扭曲或比例不对:这通常是由于关节位置定义不合理或线段连接错误导致的。检查关节位置和连接顺序是否正确。
(2)图形显示不全:确保设置ax.set_aspect('equal'),使得图形按等比例显示。
(3)如何添加动画效果:可以使用matplotlib的FuncAnimation类,通过不断更新关节位置来实现动画效果。
6.成果案例分享
通过上述步骤,你已经成功绘制了一个简单的火柴人。接下来,我们可以尝试更复杂的姿势和动画效果。例如,通过改变关节位置,实现火柴人的跳跃、行走等动作。
下面是一个简单的动画示例,展示火柴人从左到右移动的过程:
import matplotlib.animation as animation def update_position(frame, joints): # 这里我们简单地将火柴人向右移动 translation = 0.1 * frame for key in joints.keys(): joints[key][0] += translation return joints def animate(frame): global joints_anim joints_anim = update_position(frame, joints_anim) ax.clear() ax.set_aspect('equal') ax.axis('off') draw_stick_figure(joints_anim, ax) def main_animation(): fig, ax = plt.subplots() global joints_anim joints_anim = {key: value.copy() for key, value in joints.items()} # 复制初始关节位置 ani = animation.FuncAnimation(fig, animate, frames=100, interval=100) plt.show() if __name__ == "__main__": main_animation()
7.案例代码示例
以下是完整的代码示例,包括所有步骤和注释:
import matplotlib.pyplot as plt
import numpy as np
import matplotlib.animation as animation # 定义关节位置
joints = { 'head': [0, 1], 'torso': [0, 0], 'left_shoulder': [-0.5, 0], 'left_elbow': [-1, -0.5], 'left_hand': [-1, -1], 'right_shoulder': [0.5, 0], 'right_elbow': [1, -0.5], 'right_hand': [1, -1], 'left_hip': [-0.5, -0.5], 'left_knee': [-1, -1.5], 'left_foot': [-1, -2], 'right_hip': [0.5, -0.5], 'right_knee': [1, -1.5], 'right_foot': [1, -2]
} # 将关节位置转换为numpy数组,以便进行数学运算
joints = {key: np.array(value) for key, value in joints.items()} # 绘制火柴人的函数
def draw_stick_figure(joints, ax): # 清除之前的绘图 ax.clear() # 设置坐标轴的比例和限制 ax.set_aspect('equal') ax.set_xlim(-2, 2) ax.set_ylim(-2.5, 1.5) # 定义身体部分和对应的颜色(可选) body_parts = [ ('torso', 'head', 'black'), ('torso', 'left_shoulder', 'black'), ('left_shoulder', 'left_elbow', 'black'), ('left_elbow', 'left_hand', 'black'), ('torso', 'right_shoulder', 'black'), ('right_shoulder', 'right_elbow', 'black'), ('right_elbow', 'right_hand', 'black'), ('torso', 'left_hip', 'black'), ('left_hip', 'left_knee', 'black'), ('left_knee', 'left_foot', 'black'), ('torso', 'right_hip', 'black'), ('right_hip', 'right_knee', 'black'), ('right_knee', 'right_foot', 'black') ] # 绘制火柴人的各个部分 for part in body_parts: start_joint, end_joint, color = part[0], part[1], part[2] if len(part) > 2 else 'black' ax.plot([joints[start_joint][0], joints[end_joint][0]], [joints[start_joint][1], joints[end_joint][1]], color=color, linewidth=2) # 显示网格(可选) ax.grid(True) # 创建图形和坐标轴
fig, ax = plt.subplots() # 初始化函数(用于动画)
def init(): draw_stick_figure(joints, ax) return [] # 返回空列表,因为我们没有需要更新的艺术家对象 # 动画更新函数
def update(frame): # 这里可以添加使火柴人移动或改变姿势的逻辑 # 例如,简单地旋转手臂或腿 # 但为了简化,我们在这里不改变关节位置 draw_stick_figure(joints, ax) return [] # 同样返回空列表 # 创建动画
ani = animation.FuncAnimation(fig, update, frames=100, init_func=init, blit=True, interval=100) # 显示图形
plt.show()
请注意以下几点:
(1)我将关节位置转换为了numpy数组,以便在需要时进行数学运算(虽然在这个简单的例子中并没有用到)。
(2)在draw_stick_figure函数中,我添加了设置坐标轴比例和限制的代码,以及一个可选的网格显示。
(3)在body_parts列表中,我添加了颜色参数,但在这个例子中,我默认使用了黑色。你可以根据需要更改颜色。
(4)在update函数中,我没有改变关节位置,因此火柴人在动画中保持静止。你可以根据需要添加逻辑来改变火柴人的姿势或位置。
(5)我使用了FuncAnimation来创建动画,并设置了100帧和每帧之间的间隔为100毫秒。你可以根据需要调整这些参数。
运行这段代码将显示一个包含静止火柴人的窗口,并且由于动画的设置,它会每隔100毫秒重新绘制一次(尽管看起来是静止的,因为关节位置没有改变,感兴趣的读者朋友可以尝试改变关节位置)。
相关文章:
Python实现火柴人的设计与实现
1.引言 火柴人(Stick Figure)是一种极简风格的图形,通常由简单的线段和圆圈组成,却能生动地表达人物的姿态和动作。火柴人不仅广泛应用于动画、漫画和涂鸦中,还可以作为图形学、人工智能等领域的教学和研究工具。本文…...
衡石分析平台系统分析人员手册-应用模版
应用模板 应用模板使分析成果能被快速复用,节省应用创作成本,提升应用创作效率。此外应用模板实现了应用在不同环境上快速迁移。 支持应用复制功能 用户可以从现有的分析成果关联到新的分析需求并快速完成修改。 支持应用导出为模板功能 实现多个用户…...
Git和SVN
一. Git和SVN的区别 1.1 Git是分布式的,SVN是集中式的 1.2 Git复杂概念多,SVN简单易上手 Git 的命令实在太多了,日常工作需要掌握 add, commit, status, fetch, push, rebase等,若要熟练掌握,还必须掌握 rebase和 m…...
【C语言教程】【常用类库】(十八)宏与预处理 - <stddef.h> 和 <stdbool.h>
18. 宏与预处理 - <stddef.h> 和 <stdbool.h> C语言的宏和预处理指令在程序编译之前就被执行,用于文件包含、符号定义、条件编译等操作。理解和运用宏和预处理可以提高代码的灵活性和可移植性。 18.1 宏定义与条件编译 18.1.1 #define 与参数化宏 #…...
订单超时过期的实现方案的探讨
在我们的业务开发中,会遇到这样一个场景,用户下了一个单,如果超过20分钟不进行支付,订单就要变成已取消状态。 字段设定 订单中需要设定了三个字段:订单是否取消、是否支付、支付超时时间。 订单是否取消会存在&…...
C++中的CRTP
CRTP,全称为 Curiously Recurring Template Pattern(奇异递归模板模式),是一种在C中使用继承和模板技术来实现静态多态和功能复用的惯用法。它使用派生类来模板参数化基类,使得基类能够访问派生类,从而在编…...
go压缩的使用
基础:使用go创建一个zip func base(path string) {// 创建 zip 文件zipFile, err : os.Create("test.zip")if err ! nil {panic(err)}defer zipFile.Close()// 创建一个新的 *Writer 对象zipWriter : zip.NewWriter(zipFile)defer zipWriter.Close()// 创…...
一图解千言,了解常见的流程图类型及其作用
在企业管理、软件研发过程中,经常会需要进行各种业务流程梳理,而流程图就是梳理业务时必要的手段,同时也是梳理的产出。但在不同的情况下适用的流程图又不尽相同。 本文我们就一起来总结一下8 种最常见的流程图类型 数据流程图 数据流程图&…...
【微信小程序_19_自定义组件(1)】
摘要:本文主要介绍了小程序开发中自定义组件的相关知识。包括组件的创建与引用,可在项目根目录创建组件文件夹,生成相应文件,并根据使用频率选择全局或局部引用。还阐述了组件和页面的区别,如组件的.json 文件需声明 “component: true”,.js 文件调用 Component () 函数…...
标准版admin后台页面添加及开发操作流程及注意事项
基础介绍 CRMEB后台管理是基于vue2技术栈进行开发搭建的 Vue Router 使用的是v3版本,mode为history模式 如需修改 mode 请在src/setting.js中修改routerMode 新建页面 新建路由 根据目录结构,需要在src/router/modules中对应模块中,添加对…...
‘perl‘ 不是内部或外部命令,也不是可运行的程序 或批处理文件。
‘perl’ 不是内部或外部命令,也不是可运行的程序 或批处理文件。 明明已经根据教程安装了perl环境,但是在cmd中依赖报该错误,本章教程提供解决办法。 一、激活perl环境 state shell ActiveState-Perl-5.36.0此时输入perl -v 是可以直接输出perl版本号的。 二、找到perl的执…...
如何利用CMMI帮助组织消除低价值流程
CMMI发展到今天,过程中历经了不断的蜕变和升级。从早期的CMM到今天的CMMI3.0,从早期的22个过程域优化组合到今天的20个实践域,从早期隶属的SEI到今天的CMMI研究院,所有的变化都是与时俱进,都是为了提供更好的实践&…...
如何理解线程安全这个概念?
文章目录 为什么需要线程安全?线程安全的实现方式总结推荐阅读文章 线程安全(Thread Safety)是指在多线程环境中,多个线程同时访问某个对象时,不会导致程序出现错误的状态或不一致的结果。简单来说,线程安全…...
代码随想录算法训练营第48天| 739. 每日温度,496.下一个更大元素 I,503.下一个更大元素II
第十一章:图论part01 图论理论基础 大家可以在看图论理论基础的时候,很多内容 看不懂,例如也不知道 看完之后 还是不知道 邻接矩阵,邻接表怎么用, 别着急。 理论基础大家先对各个概念有个印象就好,后面在…...
Qt 支持打包成安卓
1. 打开维护Qt,双击MaintenanceTool.exe 2.登陆进去,默认是添加或移除组件,点击下一步, 勾选Android, 点击下一步 3.更新安装中 4.进度100%,完成安装,重启。 5.打开 Qt Creator,编辑-》Preferences... 6.进…...
PDF工具类源码
PDF-Guru: PDF Guru Anki是一款以PDF为中心的多功能办公学习工具箱软件,包含四大板块功能:PDF实用工具箱、Anki制卡神器、Anki最强辅助、视频笔记神器,软件功能众多且强大,熟练运用可以大幅提高办公和学习效率,绝对是您…...
NirCmd-Gui-Chinese-Introduction
简介 此程序是我的一个练习作品,单纯是为了提升编程水平,次要是为了做一个NirCmd的Gui,其实主要成分还是Gui,核心代码就两三行。 主要是Gui,功能基于nircmd.exe实现,程序本身不提供一些重要的功能。 关于…...
吴恩达深度学习笔记(7)
误差分析: 你运行一个算法代替人类计算,但是没有达到人类的效果,需要手动检查算法中的错误,对模型的一些部分做相应调整,才能更好地提升分类的精度。如果不加分析去做,可能几个月的努力对于提升精度并没有…...
二、数据离线处理场景化解决方案
https://connect.huaweicloud.com/courses/learn/Learning/sp:cloudEdu_?courseNocourse-v1:HuaweiXCBUCNXE147Self-paced&courseType1 1.离线处理方案 **业务场景-安平领域** 业务场景-金融领域 离线批处理常用组件 HDFS:分布式文件系统,为各种…...
算法题总结(十四)——贪心算法(上)
贪心算法 什么是贪心 贪心的本质是选择每一阶段的局部最优,从而达到全局最优。 贪心的套路(什么时候用贪心) 刷题或者面试的时候,手动模拟一下感觉可以局部最优推出整体最优,而且想不到反例,那么就试一试…...
变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析
一、变量声明设计:let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性,这种设计体现了语言的核心哲学。以下是深度解析: 1.1 设计理念剖析 安全优先原则:默认不可变强制开发者明确声明意图 let x 5; …...
rknn优化教程(二)
文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK,开始写第二篇的内容了。这篇博客主要能写一下: 如何给一些三方库按照xmake方式进行封装,供调用如何按…...
定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
GitHub 趋势日报 (2025年06月08日)
📊 由 TrendForge 系统生成 | 🌐 https://trendforge.devlive.org/ 🌐 本日报中的项目描述已自动翻译为中文 📈 今日获星趋势图 今日获星趋势图 884 cognee 566 dify 414 HumanSystemOptimization 414 omni-tools 321 note-gen …...
华为云Flexus+DeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建
华为云FlexusDeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建 前言 如今大模型其性能出色,华为云 ModelArts Studio_MaaS大模型即服务平台华为云内置了大模型,能助力我们轻松驾驭 DeepSeek-V3/R1,本文中将分享如何…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...
永磁同步电机无速度算法--基于卡尔曼滤波器的滑模观测器
一、原理介绍 传统滑模观测器采用如下结构: 传统SMO中LPF会带来相位延迟和幅值衰减,并且需要额外的相位补偿。 采用扩展卡尔曼滤波器代替常用低通滤波器(LPF),可以去除高次谐波,并且不用相位补偿就可以获得一个误差较小的转子位…...
comfyui 工作流中 图生视频 如何增加视频的长度到5秒
comfyUI 工作流怎么可以生成更长的视频。除了硬件显存要求之外还有别的方法吗? 在ComfyUI中实现图生视频并延长到5秒,需要结合多个扩展和技巧。以下是完整解决方案: 核心工作流配置(24fps下5秒120帧) #mermaid-svg-yP…...
ArcPy扩展模块的使用(3)
管理工程项目 arcpy.mp模块允许用户管理布局、地图、报表、文件夹连接、视图等工程项目。例如,可以更新、修复或替换图层数据源,修改图层的符号系统,甚至自动在线执行共享要托管在组织中的工程项。 以下代码展示了如何更新图层的数据源&…...
