会画画的乌龟

2023-12-06 05:50
文章标签 乌龟 画画

本文主要是介绍会画画的乌龟,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Guile 是一种 Scheme 方言的编译器,我们将这种 Scheme 方言也称为 Guile。Guile 是为增强 GNU 项目的扩展性而开发的。GNU 项目开发者可以将 Guile 解释器嵌入自己的程序中,从而使得自己的程序能够支持脚本扩展。本文取材于 Guile 官方的一篇教程,讲述一个具有绘图功能的 C 程序如何与 Guile 结合以获得脚本扩展能力。

线性插值

两点确定一条直线。假设直线 $C$ 过 $P$ 与 $Q$ 两点,其参数方程为:

$$C(t) = P + t(Q-P)$$

上述方程可变形为:

$$C(t) = (1-t)P + tQ$$

这就是线性插值公式。

gnuplot

可以使用 gnuplot 将线性插值的结果显示出来。gnuplot 是一款命令行交互式绘图软件。用它可以绘制二维与三维的数据或函数图形,也可以用于解决一些数值分析问题,例如曲线/曲面逼近方面的问题。

如果系统是 Linux,并且已安装了 gnuplot,在终端中输入 gnuplot 命令便可进入 gnuplut 命令式交互绘图环境:

$ gnuplotG N U P L O TVersion 5.0 patchlevel 3 (Gentoo revision r0)    last modified 2016-02-21 Copyright (C) 1986-1993, 1998, 2004, 2007-2016Thomas Williams, Colin Kelley and many othersgnuplot home:     http://www.gnuplot.infofaq, bugs, etc:   type "help FAQ"immediate help:   type "help"  (plot window: hit 'h')Terminal type set to 'x11'
gnuplot>

gnuplot 能够绘制参数方程的图形,它所接受的参数方程是基于维度分量的拆分形式。例如,要绘制过点 $P(0.0, 0.0)$ 与 $Q(2.71, 3.14)$ 的直线,可使用下面这条绘图命令:

plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14

不过,当你在 gnuplot 命令式交互绘图环境中输入上述绘图命令时,gnuplot 会抱怨:

undefined variable: t

这是因为 gnuplot 默认开启的是非参数方程形式的绘制模式。使用 set parametric 命令开启参数方程模式,然后便可基于参数方程绘制图形:

set parametric
plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14

结果如下图所示:

线性插值结果的可视化

set parametric 规定,对于单参数方程(可表示曲线),参数为 t,而对于双参数方程(可表示曲面),参数为 uv。注意,set parametric 命令只需使用一次,后续的 plot 命令便都以参数方程模式绘制图形。也就是说,每次使用 plot 命令绘图时,不需要重复执行 set parametric

gnuplot 默认开启了图例说明,即位于图框内部右上方的文字与图例。如果不需要它,可以在 plot 命令中通过参数 notitle 将其关闭:

plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle

结果如下图所示:

关闭图例说明

也许你已经注意到了,图框的实际宽高比(并非图框上的标称宽高比)与窗口的宽高比相等,这是 gnuplot 的默认设定。这意味着,当你拉长或圧扁窗口,图框也会相应的被拉长或圧扁。可使用 set size ratio -1 命令将图框的宽高比限定为标称宽高比:

set size ratio -1
plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle

结果如下图所示:

图框的标称宽高比

图框上标记的坐标刻度 gnuplot 自动生成的,如果我们想限定横向与纵向的坐标范围,例如限定在 [-5, 5] 区间,可使用 set [x|y]range 命令:

set xrange [-5:5]
set yrange [-5:5]
plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle

结果如下图所示:

限定坐标范围

若希望绘制的是以 $P(0.0, 0.0)$ 与 $Q(2.71, 3.14)$ 为端点的直线段,可通过调整参数 t 的取值范围来实现:

set trange [0:1]
plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle

结果如下图所示:

限定参数范围

上面的示例中,只绘制了一条直线。要是连续使用 plot 绘制两条不同的直线会怎样?例如:

plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle
plot (1-t)*0.0 + t*3.14, (1-t)*0.0 + t*2.71 notitle

结果只显示第 2 条 plot 命令的绘图结果。因为 gnuplot 默认会让新的 plot 命令会刷掉旧的 plot 命令的绘图结果。要想实现多条 plot 命令绘图结果的叠加,需要使用 set multiplot 命令开启图形叠加模式:

set multiplot
plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle
plot (1-t)*0.0 + t*3.14, (1-t)*0.0 + t*2.71 notitle

结果如下图所示:

图形叠加

要在限定横向与纵向坐标范围,并且限定参数范围的情况下绘制无图例说明的叠加图形,所需的绘图命令汇总如下:

set multiplot
set parametric
set size ratio -1
set xrange [-5:5]
set yrange [-5:5]
set trange [0:1]
plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle
plot (1-t)*0.0 + t*3.14, (1-t)*0.0 + t*2.71 notitle

管道

如果将上一节最后给出的那段 gnuplot 命令存放在一份文件中, 例如 foo.gp,那么通过管道,将 foo.gp 中的内容传递给 gnuplot,结果会发生什么?

$ cat foo.gp | gnuplot

结果会出现一个转瞬即逝的绘图窗口。

要想让这个绘图窗口持久的存在,要么使用下面的命令:

$ cat foo.gp | gnuplot --persist

要么就在 foo.gp 文件的首部增加以下命令:

set terminal x11 persist

然后:

$ cat foo.gp | gnuplot

在 C 程序中,也可以借助多进程编程与管道通信技术,将绘图命令传递于 gnuplot:

/* foo.c */
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int
main(int argc, char **argv) {int plot_pipe[2];pipe(plot_pipe);if (fork() == 0) {close(plot_pipe[1]);dup2(plot_pipe[0], STDIN_FILENO);execlp("gnuplot", NULL, NULL);} else {char *cmds = "set terminal x11 persist\n""set multiplot\n""set parametric\n""set size ratio -1\n""set xrange [-5:5]\n""set yrange [-5:5]\n""set trange [0:1]\n""plot (1-t)*0.0 + t*2.71, (1-t)*0.0 + t*3.14 notitle\n""plot (1-t)*0.0 + t*3.14, (1-t)*0.0 + t*2.71 notitle\n";close(plot_pipe[0]);FILE *output = fdopen(plot_pipe[1], "w");fprintf(output, "%s", cmds);fflush(output);}exit(EXIT_SUCCESS);
}

上述代码中的 else 分支中的代码,相当于 cat foo.gp | gnuplot 中的 cat foo.gp 部分,而 if 分支中的代码则相当于 gnuplot 部分。之所以能出现这种奇异的效果,归功于 fork 函数。

fork 函数可以从当前正在运行的程序(主进程)中分裂出一个新的正在运行的程序(新进程),这个过程有点像细胞的分裂。对于新进程,fork 函数返回值为 0,而对于主进程,fork 函数的返回值是那个分裂出来的新进程的 ID。由于我们的程序中没有用到新进程的 ID,所以这个问题就不多说了。若对这个话题感兴趣,可以去找 Linux 多进程编程的资料来看。

新进程通过 execlp 函数开启了 gnuplot 进程,然后它就死了,gnuplot 进程取代了它。gnuplot 进程等待我们向它输入绘图命令。但是,我们的主进程与 gnuplot 进程彼此独立,二者需要一种通信机制来传递信息。这种通信机制就是管道。

pipe 函数创建管道。在上例中,plot_pipe 数组便是管道,plot_pipe[0] 是其输入端,plot_pipe[1] 是其输出端。在主进程中,我们向 plot_pipe[1] 写入绘图命令,而 gnuplot 进程则通过读取 plot_pipe[0] 来获得绘图命令。由于主进程用不到 plot_pipe[1],所以需要将其关闭。同理,gnuplot 进程也用不到 plot_pipe[0],所以也需要将其关闭。

dup2 函数用于文件重定向。dup2(plot_pipe[0], STDIN_FILENO) 表示将管道的输入端重定向到系统的标准输入文件(即 stdin)。由于 gnuplot 具备从标准输入文件中获取信息的能力,所以这一切非常默契。

编译并运行这个 C 程序的命令如下:

$ gcc foo.c -o foo
$ ./foo

乌龟

这是一只会画画的乌龟,它爬行的轨迹就是它画的画。这个梗来自早期的一种面向儿童的编程语言——LOGO 语言。孩子们可以通过程序控制一只乌龟的运动,让它画出图案。现在,我们可以用 C 编写一个会画画的乌龟程序,所用的技术与工具在上文中都已经提到了。这真是个冗长的开始,直到此处,我们依然未触及本文的主题。

首先定义乌龟的活动空间:

typedef struct {FILE *plot_pipe;double west;double east;double south;double north;
} Land;static Land *
init_land(double west, double east, double south, double north) {int tube[2];pipe(tube);if (fork() == 0) {close(tube[1]);dup2(tube[0], STDIN_FILENO);execlp("gnuplot", NULL, NULL);return NULL;} else {close(tube[0]);Land *land = malloc(sizeof(Land));land->east = east;land->west = west;land->south = south;land->north = north;land->plot_pipe = fdopen(tube[1], "w");char *cmds = "set terminal x11 persist\n""set multiplot\n""set size ratio -1\n""set parametric\n""set trange [0:1]\n";assert(land->plot_pipe);fprintf(land->plot_pipe, "%s", cmds);fprintf(land->plot_pipe, "set xrange [%lf:%lf]\n", west, east);fprintf(land->plot_pipe, "set yrange [%lf:%lf]\n", south, north);fflush(land->plot_pipe);return land;}
}

然后定义乌龟:

typedef struct {double x;double y;double direction;Land *land;
} Tortoise;static Tortoise *
tortoise_alloc(Land *land) {Tortoise *t = malloc(sizeof(Tortoise));t->x = t->y = t->direction = 0.0;t->land = land;return t;
}static void
tortoise_reset(Tortoise *self) {self->x = self->y = self->direction =  0.0;
}

xy 表示乌龟在 Land 中的位置。direction 表示乌龟前进的方向。land 指向乌龟的活动空间。

乌龟只需要用上文提到的线性插值方法就可以在 gnuplot 图框内绘制出它的行走轨迹。只要给出乌龟爬行轨迹上的两个点,便可用线性插值的办法,通过一组首尾相接的直线段描绘出乌龟的爬行轨迹。我们将最基本的绘图操作定义为 draw_line 函数:

static void
draw_line(Land *land, double x0, double y0, double x1, double y1) {FILE *output = land->plot_pipe;if (x0 < land->west || x0 > land->east) return;if (y0 < land->south || y0 > land->north) return;if (x1 < land->west || x1 > land->east) return;if (y1 < land->south || y1 > land->north) return;fprintf (output,"plot [0:1] (1-t) * %lf + t * %lf, (1-t) * %lf + t * %lf notitle\n",x0, x1, y0, y1);fflush (output);
}

下面代码定义了乌龟的一些基本行为:

static void
tortoise_reset(Tortoise *self) {self->x = self->y = self->direction =  0.0;
}static void
tortoise_turn(Tortoise *self, double degree) {self->direction += M_PI / 180.0 * degree;
}static void
tortoise_forward(Tortoise *self, double distance, bool to_mark) {double newX, newY;newX = self->x + distance * cos (self->direction);newY = self->y + distance * sin (self->direction);if (to_mark) {draw_line (self->land, self->x, self->y, newX, newY);}self->x = newX;self->y = newY;
}

下面试试这个乌龟能不能胜任画图的任务:

static unsigned int
generate_random_seed_in_linux(void) {unsigned int seed;FILE *fs_p = fopen("/dev/urandom", "r");fread(&seed, sizeof(unsigned int), 1, fs_p);fclose(fs_p);return seed;
}
int
main(void) {double r = 1000.0;Land *land = init_land(-r, r, -r, r);Tortoise *t = tortoise_alloc(land);/* 让乌龟随机爬行 */{tortoise_turn(t, 180.0);tortoise_forward(t, 1000, false);tortoise_turn(t, -180.0);srand(generate_random_seed_in_linux());double old_direction = 90.0;for (int i = 0; i < 200; i++) {double direction = rand() % 180;tortoise_forward(t, 30.0, true);tortoise_turn(t, direction - old_direction);old_direction = direction;}}free(t);fclose(land->plot_pipe);return 0;
}

要让上述代码通过编译,需要包含以下头文件:

#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <stdbool.h>
#include <assert.h>
#include <time.h>
#include <unistd.h>

编译命令为:

$ gcc -lm tortoise.c -o tortoise

程序运行结果类似下图(受 gnuplot 渲染机制的限制,绘图速度不是那么快):

乌龟随机爬行路线

C 程序与 Guile 的结合

上文我们所做的事虽然有趣,但它仅仅是个冗长的前奏。现在刚开始步入正题,对于上一节所写的 C 程序,如何将其与 Guile 相结合以获得脚本扩展能力。为了便于清晰完整的呈现主题,现在假设 Land 里只有一只乌龟。也就是说,我们将定义一个全局变量来表示这只乌龟。

Tortoise *lonely_tortoise = NULL;

基于这个全局变量,就可以将上一节所实现的 tortoise_resettortoise_turn 以及 tortoise_forward 这三个函数封装为更简单的形式,使它们能够嵌入 Guile 环境:

static SCM
guile_tortoise_reset(void) {tortoise_reset(lonely_tortoise);return SCM_UNSPECIFIED;
}static SCM
guile_tortoise_turn(SCM scm_degree) {double degree = scm_to_double(scm_degree);tortoise_turn(lonely_tortoise, degree);return SCM_UNSPECIFIED;
}static SCM
guile_tortoise_forward(SCM scm_distance, SCM scm_to_mark) {double distance = scm_to_double(scm_distance);bool to_mark = scm_to_bool(scm_to_mark);tortoise_forward(lonely_tortoise, distance, to_mark);return SCM_UNSPECIFIED;
}

然后为这三个函数登籍造册,让它们以后能接受 Guile 的管理:

static void *
register_functions_into_guile(void *data) {scm_c_define_gsubr("tortoise-reset", 0, 0, 0, guile_tortoise_reset);scm_c_define_gsubr("tortoise-turn", 1, 0, 0, guile_tortoise_turn);scm_c_define_gsubr("tortoise-forward", 2, 0, 0, guile_tortoise_forward);return NULL;
}

register_functions_into_guile 是一个回调函数,需要将其传递给 scm_with_guile 函数,才能完成上述 C 函数在 Guile 环境中的注册:

scm_with_guile (&register_functions_into_guile, NULL);

一旦将 C 函数注册到 Guile 环境,那么在 Guile 解释器运行期间,可以在 Guile 解释器或 Guile 脚本中使用这些函数的名字(例如,tortoise-forward)来调用它们。scm_shell 函数可用于在 C 程序中开启 Guile 解释器:

int
main(int argc, char **argv) {double r = 1000.0;Land *land = init_land(-r, r, -r, r);lonely_tortoise = tortoise_alloc(land);scm_with_guile(register_functions_into_guile, NULL);scm_shell(argc, argv);free(lonely_tortoise);fclose(land->plot_pipe);return 0;
}

上述代码初始化了 land,生成了 lonely_tortoise 的实体,将用于表示乌龟的行为的三个 C 函数注册到了 Guile 环境,然后运行了 Guile 解释器。

要让上述代码编译通过,需要包含 Guile 库的头文件:

#include <libguile.h>

下面是完整的代码:

/* guile-tortoise.c */
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <stdbool.h>
#include <assert.h>
#include <time.h>
#include <unistd.h>
#include <libguile.h>typedef struct {FILE *plot_pipe;double west;double east;double south;double north;
} Land;static Land *
init_land(double west, double east, double south, double north) {int tube[2];pipe(tube);if (fork() == 0) {close(tube[1]);dup2(tube[0], STDIN_FILENO);execlp("gnuplot", NULL, NULL);return NULL;} else {close(tube[0]);Land *land = malloc(sizeof(Land));land->east = east;land->west = west;land->south = south;land->north = north;land->plot_pipe = fdopen(tube[1], "w");char *cmds = "set terminal x11 persist\n""set multiplot\n""set size ratio -1\n""set parametric\n""set trange [0:1]\n";assert(land->plot_pipe);fprintf(land->plot_pipe, "%s", cmds);fprintf(land->plot_pipe, "set xrange [%lf:%lf]\n", west, east);fprintf(land->plot_pipe, "set yrange [%lf:%lf]\n", south, north);fflush(land->plot_pipe);return land;}
}static void
reset_land(Land *land) {fprintf (land->plot_pipe, "clear\n");fflush (land->plot_pipe);
}static void
draw_line(Land *land, double x0, double y0, double x1, double y1) {FILE *output = land->plot_pipe;if (x0 < land->west || x0 > land->east) return;if (y0 < land->south || y0 > land->north) return;if (x1 < land->west || x1 > land->east) return;if (y1 < land->south || y1 > land->north) return;fprintf (output,"plot [0:1] (1-t) * %lf + t * %lf, (1-t) * %lf + t * %lf notitle\n",x0, x1, y0, y1);fflush (output);
}typedef struct {double x;double y;double direction;Land *land;
} Tortoise;static Tortoise *
tortoise_alloc(Land *land) {Tortoise *t = malloc(sizeof(Tortoise));t->x = t->y = t->direction = 0.0;t->land = land;return t;
}static void
tortoise_reset(Tortoise *self) {self->x = self->y = self->direction =  0.0;
}static void
tortoise_turn(Tortoise *self, double degree) {self->direction += M_PI / 180.0 * degree;
}static void
tortoise_forward(Tortoise *self, double distance, bool to_mark) {double newX, newY;newX = self->x + distance * cos (self->direction);newY = self->y + distance * sin (self->direction);if (to_mark) {draw_line (self->land, self->x, self->y, newX, newY);}self->x = newX;self->y = newY;
}static unsigned int
generate_random_seed_in_linux(void) {unsigned int seed;FILE *fs_p = fopen("/dev/urandom", "r");fread(&seed, sizeof(unsigned int), 1, fs_p);fclose(fs_p);return seed;
}/*****************************************************************                           to guile****************************************************************/
Tortoise *lonely_tortoise = NULL;static SCM
guile_tortoise_reset(void) {tortoise_reset(lonely_tortoise);return SCM_UNSPECIFIED;
}static SCM
guile_tortoise_turn(SCM scm_degree) {double degree = scm_to_double(scm_degree);tortoise_turn(lonely_tortoise, degree);return SCM_UNSPECIFIED;
}static SCM
guile_tortoise_forward(SCM scm_distance, SCM scm_to_mark) {double distance = scm_to_double(scm_distance);bool to_mark = scm_to_bool(scm_to_mark);tortoise_forward(lonely_tortoise, distance, to_mark);return SCM_UNSPECIFIED;
}static void *
register_functions_into_guile(void *data) {scm_c_define_gsubr("tortoise-reset", 0, 0, 0, guile_tortoise_reset);scm_c_define_gsubr("tortoise-turn", 1, 0, 0, guile_tortoise_turn);scm_c_define_gsubr("tortoise-forward", 2, 0, 0, guile_tortoise_forward);return NULL;
}int
main(int argc, char **argv) {double r = 1000.0;Land *land = init_land(-r, r, -r, r);lonely_tortoise = tortoise_alloc(land);scm_with_guile(register_functions_into_guile, NULL);scm_shell(argc, argv);free(lonely_tortoise);fclose(land->plot_pipe);return 0;
}

编译上述代码的命令为:

$ gcc `pkg-config --cflags --libs guile-2.0` guile-tortoise.c -o guile-tortoise

运行编译所得程序:

$ ./guile-tortoise
Copyright (C) 1995-2014 Free Software Foundation, Inc.Guile comes with ABSOLUTELY NO WARRANTY; for details type `,show w'.
This program is free software, and you are welcome to redistribute it
under certain conditions; type `,show c' for details.Enter `,help' for help.
scheme@(guile-user)> 

这个程序不仅会为你开启一个 gnuplot 的绘图窗口,同时也会进入 Guile 解释器交互环境。在这个环境里,可以使用 Scheme 语言控制那只孤独的小乌龟进行绘图。例如:

> (tortoise-forward 300 #t)
> (tortoise-turn 90)
> (tortoise-forward 300 #t)
> (tortoise-turn 90)
> (tortoise-forward 300 #t)
> (tortoise-turn 90)
> (tortoise-forward 300 #t)

上述这些重复的绘制『命令』,可在 gnuplot 绘图窗口中交互绘制出一个矩形:

Guile 解释器

复杂的行走

Guile 是个解释器,它可以解释运行 Scheme 语言。如果你对 Scheme 有一定了解,那么便可以用它写脚本,用更复杂的逻辑来控制那只孤独的小乌龟绘制图案。

下面这份脚本可控制小乌龟在不同方位绘制一些正多边形(边数较大时,近似为圆):

;;;; circles.scm
(define (draw-polygon n r)(do ((i 0 (1+ i)))((= i n))(begin(tortoise-forward (* r (sin (* 3.14159 (/ 1 n)))) #t)(tortoise-turn (/ 360.0 n)))))(do ((i 0 (1+ i)))((= i 36))(begin(tortoise-turn 10.0)(draw-polygon 30 800)))

用上一节生成的 guile-tortoise 程序解释运行 circles.scm 脚本:

$ ./guile-tortoise circles.scm

这些正多边形叠加到一起,可展现出复杂的景象:

圆

下面这份 Scheme 脚本可以绘制两朵不同形状的雪花:

;;;; snowflake.scm
(define (koch-line length depth)(if (zero? depth)(tortoise-forward length #t)(let ((sub-length (/ length 3))(sub-depth (1- depth)))(for-each (lambda (angle)(koch-line sub-length sub-depth)(tortoise-turn angle))'(60 -120 60 0)))))(define (snowflake length depth sign)(let iterate ((i 1))(if (<= i 3)(begin(koch-line length depth)(tortoise-turn (* sign -120))(iterate (1+ i))))))(tortoise-turn 90)
(tortoise-forward 250 #f)
(tortoise-turn -90)(snowflake 800 3 1)
(tortoise-turn 180)
(snowflake 800 3 -1)

用 guile-tortoise 程序解释运行 snowflake.scm 脚本:

$ ./guile-tortoise snowflake.scm

所得结果如下图所示:

雪花

总结

对于编程的初学者而言,这篇文章应该是有趣的。它向你展示了,不需要多么复杂的工具和编程技术,只需将功能较为单一的组件通过某些特定的机制组合起来,便可得到一个能够绘制二维图形并且具备脚本扩展功能的程序。这是不是出乎意料?

从一开始,在 gnuplot 中交互绘图,我们需要了解许多 gnuplot 的知识方能绘制线性插值结果。接下来,我们尝试在 C 程序中通过管道,向 gnuplot 输出绘图命令,这样我们可以很方便的使用 C 语言来操纵 gnuplot 了,而且我们在 C 程序中还抽象出一只会画图的小乌龟,通过控制小乌龟的爬行来绘制图形。利用 C 程序操纵 gnuplot 固然可绘制复杂的图案,但是每次要绘制新的图形,不得不改写并重新编译 C 程序。最后,我们在 C 程序中嵌入了 Guile 解释器,然后用 Scheme 来编写绘图脚本,这样可以在保持 C 程序不变的情况下,绘制出复杂的图案。更有趣的是,在使用 Scheme 语言为这个 C 程序编写绘图脚本时,我们已经不觉得 gnuplot 的存在了。

不过,虽然通过嵌入 Guile 解释器能够让程序拥有脚本扩展功能,但是要用好这一功能,需要对 Scheme 语言有所了解。Scheme 语言很简单,尽管要用它来构建实际的程序看起来困难重重,但是我们可以用它来写一些脚本,逐步的掌握它。事实上,我们学习任何一种编程语言,在开始时,用它写实际的程序也是困难重重的。学习的过程就应该像文中的那只孤独的小乌龟那样一步一步的前进,终有所成。

这篇关于会画画的乌龟的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/460672

相关文章

uva 10154 DP 叠乌龟

题意: 给你几只乌龟,每只乌龟有自身的重量和力量。 每只乌龟的力量可以承受自身体重和在其上的几只乌龟的体重和内。 问最多能叠放几只乌龟。 解析: 先将乌龟按力量从小到大排列。 然后dp的时候从前往后叠,状态转移方程: dp[i][j] = dp[i - 1][j];if (dp[i - 1][j - 1] != inf && dp[i - 1][j - 1] <= t[i]

如何使用小乌龟清除认证缓存、还原版本、定位及常用开发工具集成

😀前言 本篇博文是关于如何使用小乌龟清除认证缓存、还原版本、定位及常用开发工具集成,希望你能够喜欢 🏠个人主页:晨犀主页 🧑个人简介:大家好,我是晨犀,希望我的文章可以帮助到大家,您的满意是我的动力😉😉 💕欢迎大家:这里是CSDN,我总结知识的地方,欢迎来到我的博客,感谢大家的观看🥰 如果文章有什么需要改进的地方还请大佬不吝赐教 先在此感谢啦😊 文章目录 如何清除

如何使用 TortoiseGit(小乌龟)进行分支创建、切换与合并以及解决冲突

😀前言 本文将详细介绍如何使用 TortoiseGit(小乌龟)进行分支创建、切换与合并以及解决冲突等操作。TortoiseGit 是一个广泛使用的 Windows 图形化 Git 客户端,其友好的用户界面和丰富的功能使得 Git 操作变得更加直观和便捷。 🏠个人主页:晨犀主页 🧑个人简介:大家好,我是晨犀,希望我的文章可以帮助到大家,您的满意是我的动力😉😉 💕欢迎大家:这里

如何使用 TortoiseGit(小乌龟)进行项目源代码的检出、添加与提交、代码推送与拉取

😀前言 本文详细介绍如何使用 TortoiseGit(小乌龟)进行项目源代码的检出、文件的添加与提交、代码的推送与拉取, 🏠个人主页:晨犀主页 🧑个人简介:大家好,我是晨犀,希望我的文章可以帮助到大家,您的满意是我的动力😉😉 💕欢迎大家:这里是CSDN,我总结知识的地方,欢迎来到我的博客,感谢大家的观看🥰 如果文章有什么需要改进的地方还请大佬不吝赐教 先在此感谢啦😊

小乌龟运动控制-3小乌龟走五角星

目录 第一章 小乌龟划圆圈 第二章 小乌龟走方形 第三章 小乌龟走五角星 文章目录 目录一、创建功能包二、编写Python脚本三、运行ROS节点 一、创建功能包 要让ROS乌龟做五角星运动,需要编写ROS节点来控制乌龟的运动。以下是一个简单的示例: 创建ROS包和节点 首先需要创建一个ROS包,然后在包内创建一个节点。可以使用以下命令创建ROS包和节点: $ ca

小乌龟运动控制-2 小乌龟走方形

目录 第一章 小乌龟划圆圈 第二章 小乌龟走方形 文章目录 目录一、简陋版-乌龟行走方形二、强化版-乌龟行走方形 一、简陋版-乌龟行走方形 常见的简陋的控制乌龟行走方形的方式很简单,例如下面,可以测试下: 我们需要创建一个名为draw_square.py 的 Python 文件,并将其保存在 ~/catkin_ws/src/turtlebot3_draw_circl

NOIP 2010 乌龟棋

题目 描述 小明过生日的时候,爸爸送给他一副乌龟棋当作礼物。 乌龟棋的棋盘是一行N个格子,每个格子上一个分数(非负整数)。棋盘第1格是唯一的起点,第N格是终点,游戏要求玩家控制一个乌龟棋子从起点出发走到终点。 乌龟棋中M张爬行卡片,分成4种不同的类型(M张卡片中不一定包含所有4种类型的卡片,见样例),每种类型的卡片上分别标有1、2、3、4四个数字之一,表示使用这种卡片后,乌龟棋子将向前爬行

智能对话画画音乐创作歌曲三合一系统源码 前后端分离 附带完整的源代码包+搭建部署教程

系统概述 创想无界是一个高度集成的创意生态系统,它巧妙地融合了人工智能的三大核心领域:自然语言处理(NLP)用于智能对话,计算机视觉(CV)实现自动画画,以及音乐生成算法来创作个性化歌曲。该系统的设计初衷是为了打破传统艺术创作的界限,通过技术的力量,让每个人都能成为艺术家,自由探索和表达自我。 代码示例 系统特色功能一览 1. 智能对话引擎     个性化交流:基于深度学习的聊天机器

看看姐姐的生活乌龟

与困难战斗的生活 今天的与困难战斗的生活,真是缩头乌龟,一边读一边暗暗期盼,哈哈,俯视一切的雄心和气概,我又翻到另一课,所以叫三月桃花雪,我一连读了两篇课文,看看姐姐的生活乌龟。 她也面带笑容,作者,汽车模型是完全仿真的,千百年来为人们所传诵,由于我很紧张,万里无云,身上还插着一对翅膀穿着一身的婚纱群,就像一个小天使般,她手拿彩带瓶。 可它也得到了一个新名字笨笨,趁此机会又大声地读起课文来

【用Python画画】六一儿童节画爱心

本文收录于 《Python编程入门》专栏,从零基础开始,分享一些Python编程基础知识,欢迎关注,谢谢! 文章目录 一、前言二、代码示例三、知识点梳理四、总结 一、前言 本文介绍如何使用Python的海龟画图工具turtle,画一个大大的爱心。 什么是Python? Python是由荷兰人吉多·范罗苏姆于1990年代初设计的一门编程语言,应用领域非常广泛,尤其在数