c 语言能否画一维坐标图,如何用 C 语言画几何图形?

2024-02-24 17:30

本文主要是介绍c 语言能否画一维坐标图,如何用 C 语言画几何图形?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 原问题

*

* * *

* * * * *

* *

* * * * * *

* * * * * * * * * *

* * *

* * * * * * * * *

* * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

复制代码

我才不会给一个平凡解,给别人来交功课。希望可以用另类一些的方法去解决(至少不是老师要求的),可以让同学开拓一下思路。但这个原答案:

main(i){for(i=0;i<288;i++)printf(i%24-23?abs(i%24-11)*24>i|abs((i%24+i/72*3+3)%6-2)>i/24%3?" ":"* ":"\n");}

复制代码

是「压缩」后的结果,同学可能无法理解当中的意义。我尝试在本文简单解析一下。

2. 绘图框架

在计算机图形学中,要合成(synthesize)一个图像(image)时,通常有两种方法:绘画每个形状,填充形状覆盖的像素。

对于图像中每个像素,采样(sample)该像素覆盖了什么形状。

第一种就是光栅化(rasterization)算法,第二种包括光线追踪(ray tracing)、光线步进(ray marching)等算法。

第二种做法可以理解为设计一个数学函数,例如二维的单色图像就可以定义为一个函数

c172805c61c5756d1471e4214adc4b0b.png。这种方式可以用较少的代码画出复杂的形状。如果要输出文本模式,只用两个符号表示图形,可用这个代码框架:

#include

const int w = 20;

const int h = 20;

int f(int x, int y) {

return /*...*/;

}

int main() {

int x, y;

for (y = 0; y < h; y++) {

for (x = 0; x < w; x++)

printf(f(x, y) ? "* " : " ");

puts("");

}

}

复制代码

例如,圆盘(disk)在数学上可定义为一个隐函数

79c76d74c45db6602dce4cc3a3b683e5.png,那么画一个置于画布中心

2c3a3cd00fa47a71b17844fcd0b58d84.png、半径 8 的圆盘只需要定义

ef6825a791bab5807d6a7b0e3e228ba1.png 为:

int f(int x, int y) {

return (x - 10) * (x - 10) + (y - 10) * (y - 10) 8 * 8;

}

复制代码

输出:

*

* * * * * * *

* * * * * * * * * * *

* * * * * * * * * * * * *

* * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * *

* * * * * * * * * * * * *

* * * * * * * * * * *

* * * * * * *

*

复制代码

3. 直角等腰三角形

先考虑半个三角形,可用

07679b88830db91f1d976942ff70a136.png

int f(int x, int y) {

return x y;

}

复制代码

设 w = 23, h = 12,那么输出是这样的:

*

* *

* * *

* * * *

* * * * *

* * * * * *

* * * * * * *

* * * * * * * *

* * * * * * * * *

* * * * * * * * * *

* * * * * * * * * * *

* * * * * * * * * * * *

复制代码

要画出题目中的大三角形,我们可以使用绝对值

ef583e33e26a38605a0d829293471fc9.png

149292746fd8062b8434ee0101fe0176.png 表示对称轴的

9e0a1373ede1e652d762a6e850128f4d.png 坐标:

int f(int x, int y) {

return abs(x - 11) y;

}

复制代码

输出:

*

* * *

* * * * *

* * * * * * *

* * * * * * * * *

* * * * * * * * * * *

* * * * * * * * * * * * *

* * * * * * * * * * * * * * *

* * * * * * * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * * * * *

复制代码

4. 密铺三角形

然后,我们考虑要画一些密铺三角形。首先,我们可把上面画三角形的功能写成一个函数:

int triangle(int x, int y) {

return abs(x) y;

}

复制代码

这个三角形的上顶点是位于

b42d08c1197f464aed4d2d968cac7a21.png ,我们可以通过平移坐标来放置它在不同位置。另外,为了密辅,我们可用取模运算:

int f(int x, int y) {

return triangle(x % 6 - 2, y % 3);

}

复制代码

输出:

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

复制代码

但这种密辅和题目要求的有出入,我们要对每行三角形偏移半个三角形:

int f(int x, int y) {

return triangle((x + y / 3 * 3 + 3) % 6 - 2, y % 3);

}

复制代码

y / 3 是三角形的行数,y / 3 * 3 是对每行三角形偏移半个三角形(包括一个空白后三角形总宽是6),然后再偏移半个三角形。输出:

* * *

* * * * * * * * * * *

* * * * * * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

* * *

* * * * * * * * * * *

* * * * * * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

复制代码

可以看到中间的部分是我们所需的密铺三角形。

5. 交集

要结合上面的大三角形和密铺三角形,非常简单,只需用逻辑与就能实现两个图形的交集:

int f(int x, int y) {

return triangle(x - 11, y) &&

triangle((x + y / 3 * 3 + 3) % 6 - 2, y % 3);

}

复制代码

这里我们重复地使用 triangle 去画这两种三角形,输出:

*

* * *

* * * * *

* *

* * * * * *

* * * * * * * * * *

* * *

* * * * * * * * *

* * * * * * * * * * * * * * *

* * * *

* * * * * * * * * * * *

* * * * * * * * * * * * * * * * * * * *

复制代码

到这一步,可算是大功告成。(唉⋯⋯几分钟能写完的算什么大功)

6. 压缩为了减少代码的字符,我们从两个 for 变成一个。然后置换 x = i % 24, y = i / 24 。

把每行最后一个字符用作换行。所以宽度从 23 改为 24。

因为 main 的第一个参数是 int,可用来声明那个 for 变量。

main(i) {

for (i = 0; i < 24 * 12; i++)

printf(i % 24 != 23 ? f(i % 24, i / 24) ? "* " : " ":"\n");

}

复制代码

4. 人手内联 triangle()。

int f(int x, int y) {

return abs(x - 11) y &&

abs((x + y / 3 * 3 + 3) % 6 - 2) y % 3;

}

复制代码

5. 人手内联 f()。i / 24 / 3 等价于 i / 72。

6. 移除所有非必要的空白符。

main(i){for(i=0;i<288;i++)printf(i%24!=23?abs(i%24-11)i 24&&abs((i%24+i="" 72*3+3)%6-2)i="" 24%3?"*="" ":"="" ":"\n");}="" <="" code="">复制代码7. 把 && 改成 &,因去除了短路求值降低了性能,但省了一个字符。

8. i%24!=23 等价于 i%24-23,省了一个字符。

9. abs(i%24-11)<=i 24="" 可改成="" abs(i%24-11)*24<="i,提高性能。10. 把条件取反,交换后面的字符串,即 cond?a:b 等价于 !cond?b:a。然后德摩根定律 !(c&&d)等价于(!c)||(!d),所以<= 变成="">,& 变成 |。省了 2 个字符。

main(i){for(i=0;i<288;i++)printf(i%24-23?abs(i%24-11)*24>i|abs((i%24+i/72*3+3)%6-2)>i/24%3?" ":"* ":"\n");}

复制代码

此文的前半步部分可了解一下思路,这个压缩部分纯粹娱乐,不要写这样的代码。

相关问题

这篇关于c 语言能否画一维坐标图,如何用 C 语言画几何图形?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

深入理解C语言的void*

《深入理解C语言的void*》本文主要介绍了C语言的void*,包括它的任意性、编译器对void*的类型检查以及需要显式类型转换的规则,具有一定的参考价值,感兴趣的可以了解一下... 目录一、void* 的类型任意性二、编译器对 void* 的类型检查三、需要显式类型转换占用的字节四、总结一、void* 的

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return

MiniGPT-3D, 首个高效的3D点云大语言模型,仅需一张RTX3090显卡,训练一天时间,已开源

项目主页:https://tangyuan96.github.io/minigpt_3d_project_page/ 代码:https://github.com/TangYuan96/MiniGPT-3D 论文:https://arxiv.org/pdf/2405.01413 MiniGPT-3D在多个任务上取得了SoTA,被ACM MM2024接收,只拥有47.8M的可训练参数,在一张RTX

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而