C語(yǔ)言愛(ài)心代碼easyx有注釋版
2022-11-26 18:43 作者:那蔚藍(lán)天空 | 我要投稿
?尊重UP原創(chuàng),以下內(nèi)容僅供學(xué)習(xí),請(qǐng)不要用我的相關(guān)代碼在平臺(tái)發(fā)布任何視頻或者專欄。
遇到錯(cuò)誤的可以參考前面兩份源碼的教程(三份源碼)
? ? ? ? ? ? ? ? ? ? ? ? ? ?C語(yǔ)言愛(ài)心代碼
#include<graphics.h>
#include <conio.h>
#include<time.h>
#include<math.h>
#include<stdlib.h>
//愛(ài)心點(diǎn)結(jié)構(gòu)體
struct Point {
double x, y; ? ? //坐標(biāo)
COLORREF color; ?//顏色
};
//顏色數(shù)組
COLORREF colors[7] = { RGB(255,32,83),RGB(252,222,250) ,RGB(255,0,0) ,RGB(255,0,0) ,RGB(255,2,2) ,RGB(255,0,8) ,RGB(255,5,5) };
//COLORREF colors[7] = { RGB(55,132,83),RGB(252,222,250) ,RGB(25,120,130) ,RGB(25,230,40) ,RGB(25,24,112) ,RGB(255,230,128) ,RGB(25,5,215) };
const int xScreen = 1200; ?//屏幕寬度
const int yScreen = 800; ?//屏幕高度
const double PI = 3.1426535159; ? ? ? ? //圓周率
const double e = 2.71828; ?//自然數(shù)e
const double averag_distance = 0.162; ? ?//弧度以0.01增長(zhǎng)時(shí),原始參數(shù)方程每個(gè)點(diǎn)的平均距離
const int quantity = 506; ?//一個(gè)完整愛(ài)心所需點(diǎn)的數(shù)量
const int circles = 210; ?//組成愛(ài)心主體的愛(ài)心個(gè)數(shù)(每個(gè)愛(ài)心會(huì)乘以不同系數(shù))
const int frames = 20; ?//愛(ài)心擴(kuò)張一次的幀數(shù)
Point ?origin_points[quantity]; ?//創(chuàng)建一個(gè)保存原始愛(ài)心數(shù)據(jù)的數(shù)組
Point ?points[circles * quantity]; ? ? ?//創(chuàng)建一個(gè)保存所有愛(ài)心數(shù)據(jù)的數(shù)組
IMAGE images[frames]; ?//創(chuàng)建圖片數(shù)組
//坐標(biāo)轉(zhuǎn)換函數(shù)
double screen_x(double x)
{
x += xScreen / 2;
return x;
}
//坐標(biāo)轉(zhuǎn)換函數(shù)
double screen_y(double y)
{
y = -y + yScreen / 2;
return y;
}
//創(chuàng)建x1-x2的隨機(jī)數(shù)的函數(shù)
int creat_random(int x1, int x2)
{
if (x2 > x1)
return ?rand() % (x2 - x1 + 1) + x1;
else
return 0;
}
//創(chuàng)建愛(ài)心擴(kuò)張一次的全部數(shù)據(jù),并繪制成20張圖片保存
// 1 用參數(shù)方程計(jì)算出一個(gè)愛(ài)心的所有坐標(biāo)并保存在 origin_points 中
// 2 重復(fù)對(duì) origin_points 的所有坐標(biāo)乘上不同的系數(shù)獲得一個(gè)完整的愛(ài)心坐標(biāo)數(shù)據(jù),并保存在 points 中
// 3 通過(guò)一些數(shù)學(xué)邏輯計(jì)算 points 中所有點(diǎn)擴(kuò)張后的坐標(biāo)并繪制,并覆蓋掉原來(lái)的數(shù)據(jù)(循環(huán)20次)
// 4 計(jì)算圓的外層那些閃動(dòng)的點(diǎn),不保存這些點(diǎn)的數(shù)據(jù)(循環(huán)20次)
void creat_data()
{
int index = 0;
//保存相鄰的坐標(biāo)信息以便用于計(jì)算距離
double x1 = 0, y1 = 0, x2 = 0, y2 = 0;
for (double radian = 0.1; radian <= 2 * PI; radian += 0.005)
{
//愛(ài)心的參數(shù)方程
x2 = 16 * pow(sin(radian), 3);
y2 = 13 * cos(radian) - 5 * cos(2 * radian) - 2 * cos(3 * radian) - cos(4 * radian);
//計(jì)算兩點(diǎn)之間的距離 開(kāi)根號(hào)((x1-x2)平方 + (y1-y1)平方)
double distance = sqrt(pow(x2 - x1, 2) + pow(y2 - y1, 2));
//只有當(dāng)兩點(diǎn)之間的距離大于平均距離才保存這個(gè)點(diǎn),否則跳過(guò)這個(gè)點(diǎn)
if (distance > averag_distance)
{
//x1和y1保留當(dāng)前數(shù)據(jù)
//x2和y2將在下一次迭代獲得下一個(gè)點(diǎn)的坐標(biāo)
x1 = x2, y1 = y2;
origin_points[index].x = x2;
origin_points[index++].y = y2;
}
}
index = 0;
for (double size = 0.1; size <= 20; size += 0.1)
{
//用sigmoid函數(shù)計(jì)算當(dāng)前系數(shù)的成功概率
//用個(gè)例子說(shuō)明一下,假設(shè)有100個(gè)點(diǎn)成功概率為 90%,那么就可能會(huì)有90個(gè)點(diǎn)經(jīng)過(guò)篩選保留下來(lái)
// 假設(shè)有100個(gè)點(diǎn)成功概率為 20%,那么就可能會(huì)有20個(gè)點(diǎn)經(jīng)過(guò)篩選保留下來(lái)
double success_p = 1 / (1 + pow(e, 8 - size / 2));
//遍歷所有原始數(shù)據(jù)
for (int i = 0; i < quantity; ++i)
{
//用概率進(jìn)行篩選
if (success_p > creat_random(0, 100) / 100.0)
{
//從顏色數(shù)組隨機(jī)獲得一個(gè)顏色
points[index].color = colors[creat_random(0, 6)];
//對(duì)原始數(shù)據(jù)乘上系數(shù)保存在points中
points[index].x = size * origin_points[i].x + creat_random(-4, 4);
points[index++].y = size * origin_points[i].y + creat_random(-4, 4);
}
}
}
//index當(dāng)前值就是points中保存了結(jié)構(gòu)體的數(shù)量
int points_size = index;
for (int frame = 0; frame < frames; ++frame)
{
//初始化每張圖片寬xScreen,高yScreen
images[frame] = IMAGE(xScreen, yScreen);
//把第frame張圖像設(shè)為當(dāng)前工作圖片
SetWorkingImage(&images[frame]);
//計(jì)算愛(ài)心跳動(dòng)的坐標(biāo)
for (index = 0; index < points_size; ++index)
{
double x = points[index].x, y = points[index].y; ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?//把當(dāng)前值賦值給x和y
double distance = sqrt(pow(x, 2) + pow(y, 2)); //計(jì)算當(dāng)前點(diǎn)與原點(diǎn)的距離
double diatance_increase = -0.0009 * distance * distance + 0.35714 * distance + 5; //把當(dāng)前距離代入方程獲得該點(diǎn)的增長(zhǎng)距離
//根據(jù)增長(zhǎng)距離計(jì)算x軸方向的增長(zhǎng)距離 x_increase = diatance_increase * cos(當(dāng)前角度)
//cos(當(dāng)前角度)= x / distance
double x_increase = diatance_increase * x / distance / frames;
//根據(jù)增長(zhǎng)距離計(jì)算x軸方向的增長(zhǎng)距離 x_increase = diatance_increase * sin(當(dāng)前角度)
//sin(當(dāng)前角度)= y / distance
double y_increase = diatance_increase * y / distance / frames;
//因?yàn)橐陨嫌?jì)算得到的是一整個(gè)過(guò)程的增長(zhǎng)距離,而整個(gè)過(guò)程持續(xù)20幀,因此要除20
//用新的數(shù)據(jù)覆蓋原來(lái)的數(shù)據(jù)
points[index].x += x_increase;
points[index].y += y_increase;
//提取當(dāng)前點(diǎn)的顏色設(shè)置為繪畫(huà)顏色
setfillcolor(points[index].color);
//注意,因?yàn)橐陨纤凶鴺?biāo)是基于數(shù)學(xué)坐標(biāo)的
//因此繪制到屏幕是就要轉(zhuǎn)換為屏幕坐標(biāo)
solidcircle(screen_x(points[index].x), screen_y(points[index].y), 1);
}
//產(chǎn)生外圍閃動(dòng)的點(diǎn)
for (double size = 17; size < 23; size += 0.3)
{
for (index = 0; index < quantity; ++index)
{
//當(dāng)系數(shù)大于等于20,通過(guò)概率為百分之四十,當(dāng)系數(shù)小于20,通過(guò)概率為百分之五
//20作為關(guān)鍵值是因?yàn)閻?ài)心主體的最大系數(shù)就是20
if ((creat_random(0, 100) / 100.0 > 0.6 && size >= 20) || (size < 20 && creat_random(0, 100) / 100.0 > 0.95))
{
double x, y;
if (size >= 20)
{
//用frame的平方的正負(fù)值作為上下限并加減15產(chǎn)生隨機(jī)數(shù)
//用frame的平方的好處是frame越大,外圍閃動(dòng)的點(diǎn)運(yùn)動(dòng)范圍越大
x = origin_points[index].x * size + creat_random(-frame * frame / 5 - 15, frame * frame / 5 + 15);
y = origin_points[index].y * size + creat_random(-frame * frame / 5 - 15, frame * frame / 5 + 15);
}
else
{
//對(duì)于系數(shù)小于20的處理與愛(ài)心點(diǎn)一樣
x = origin_points[index].x * size + creat_random(-5, 5);
y = origin_points[index].y * size + creat_random(-5, 5);
}
//隨機(jī)獲取顏色并設(shè)置為當(dāng)前繪圖顏色
setfillcolor(colors[creat_random(0, 6)]);
//把數(shù)學(xué)坐標(biāo)轉(zhuǎn)換為屏幕坐標(biāo)再進(jìn)行繪制
solidcircle(screen_x(x), screen_y(y), 1);
//需要注意的是,我并沒(méi)有保存這些點(diǎn),因?yàn)檫@些點(diǎn)不需要前一幀的坐標(biāo)數(shù)據(jù)
//只需要當(dāng)前系數(shù)就可繪制出來(lái),因此沒(méi) 必要保存
}
}
}
}
}
int main()
{
initgraph(xScreen, yScreen); ?//創(chuàng)建屏幕
BeginBatchDraw(); ?//開(kāi)始批量繪圖
srand(time(0)); ?//初始化隨機(jī)種子
creat_data(); ?//調(diào)用函數(shù)產(chǎn)生20張圖片
SetWorkingImage(); ?//調(diào)用函數(shù)把工作圖像恢復(fù)為窗口,沒(méi)有添加參數(shù)默認(rèn)為窗口
?//因?yàn)榻酉率怯么翱诓シ艌D片,因此要把繪圖效果設(shè)置為窗口
bool extend = true, shrink = false;
for (int frame = 0; !_kbhit();) ? ? //退出條件為檢測(cè)到按鍵信息
{
putimage(0, 0, &images[frame]); //播放第frame張圖片
FlushBatchDraw(); //刷新批量繪圖
Sleep(20); //延時(shí)20毫秒
cleardevice(); //清除屏幕,用來(lái)播放下一幀圖片
//注意 creat data 產(chǎn)生的只是愛(ài)心擴(kuò)張的20張圖片,并沒(méi)有產(chǎn)生愛(ài)心收縮的圖片
//但是把擴(kuò)張的圖片倒著播放就產(chǎn)生的收縮的效果
//所以下面這個(gè) if else 語(yǔ)句就是決定圖片是正常播放還是倒著播放
if (extend) ?//擴(kuò)張時(shí), ++frame,正常播放
frame == 19 ? (shrink = true, extend = false) : ++frame;
else ? ? ? ? //收縮時(shí), --frame,倒著播放
frame == 0 ? (shrink = false, extend = true) : --frame;
}
EndBatchDraw(); //關(guān)閉批量繪圖
closegraph(); //關(guān)閉繪圖窗口
return 0; //結(jié)束程序
}
標(biāo)簽: