实验四——再探类
Complex类就不发出来了,相对于下面而言过于简单。
下面两个实验都做了扩展,并加入了一些自己的想法;
//!!!!!!!!重新写了Graph类(代码跟在旧的下面(hash_type没有变动)图片没有更新因为太麻烦了,想看实际效果的,还是自己运行吧)
提供一个贪吃蛇小游戏博客链接,可以从里面学到很多东西(就算不学习也可以用来消遣啊)
[https://blog.csdn.net/silence1772/article/details/55005008]
Graph类:
main主函数里面没有什么东西,主要都是针对用户体验的优化,还有就是两个系统指令,不过也是为了优化体验
main.cpp:
#include <iostream>
#include "graph.h"
using namespace std;
int main()
{
while(1)
{
system("CLS");
system("color 0F");//颜色复位
char m;
int n;
cout<<"请输入图形的构成字符与尺寸:"<<endl;
cin>>m>>n;
Graph graph(m,n);
graph.draw();
cout<<"你想要对当前图形进行更多操作吗?是(y),否(n)"<<endl;
char ch;
cin>>ch;
if(ch=='n')continue;
else if(ch=='y')
{
cout<<"您可以用方向键调整图形的位置(D),并可以设置图形的前景色和背景色(C)"<<endl
<<"您在操作中可以输入C或D以切换至相应的操作"<<endl
<<"当您需要结束对当前图形的操作时,请输入 B "<<endl;
while(1)
{
if(graph.coto!='B'&&graph.coto!='C'&&graph.coto!='D')cin>>graph.coto;
if(graph.coto=='B')break;
else if(graph.coto=='D')graph.movg();
else if(graph.coto=='C')graph.color();
else cout<<"请输入规定的字符,否则我们无法理解您的意图"<<endl;
}
}
graph.coto='0';
}
return 0;
}
graph头文件也没什么要说的,要解释的都已经注释了
graph.h:
#ifndef GRAPH_H
#define GRAPH_H
#include<cstdlib>
#include <conio.h>
#include <windows.h>
#include <string>
using namespace std;
// 类Graph的声明
class Graph {
public:
Graph(char ch, int n); // 带有参数的构造函数
void draw(); //绘制图形
void recolordraw(string a,string b); //重绘图形
void removgdraw(int* a);
void color(); //设置前景色与背景色
void movg(); //移动图形
char cdef(string a);
int m[2]={0,0};//记录待绘制图形相对于初始位置的移动,即图形状态参数
char coto='0';//用于记录用户对图形的主操作指令
char s[9]={'c','o','l','o','r',' ','0','F','\0'};//字符串数组初始化,用于执行颜色设置
private:
char symbol;
int size;
};
#endif
类实现文件,嗯。。。我是又爱又恨哪
graph.cpp:
#include "graph.h"
#include"hash_type.h"
#include <iostream>
using namespace std;
Graph::Graph(char ch, int n): symbol(ch), size(n) {}
//draw函数完成初始绘图操作
void Graph::draw()
{
system("CLS");//清屏->输出复位
for(int i=size-1;i>=0;i--)
{
for(int m=0;m<i;m++)
{
cout<<" ";
}
for(int j=0;j<2*(size-i)-1;j++)
{
cout<<symbol;
}
cout<<endl;
}
}
//color函数面对用户设置前景色和背景色的操作
void Graph::color()
{
cout<<"现在您可以设置图形的前景色和背景色"
<<"(请先输入背景色,再输入前景色和背景色)"<<endl
<<"并且请不要使前景色和背景色相同(这样的操作是无效的,将保留上一状态)"<<endl;
while(1)
{
string a,b;
cin>>a;
if(a=="B")///////////////////////////////
{ //
coto='B'; //
break; //
} //
else if(a=="D") //保证能够跳出当前函数
{ //
coto='D'; //
break; //
}////////////////////////////////////////
cin>>b;
recolordraw(a,b);
}
}
//movg函数面对用户移动图形的操作
void Graph::movg()
{
cout<<"现在您可以通过方向键控制图形的移动"<<endl;
while(1)
{
if(getch()==0xE0)//getch()读取输入流时不对输入流进行操作,并且读取一部分以确定是虚拟键值,第二次读取一个整型量确定是哪个键,函数原型在 conio.h 里
{
//system("CLS");//清屏指令
switch(getch())
{
case 72:m[1]--;break;//上
case 80:m[1]++;break;//下
case 75:m[0]--;break;//左
case 77:m[0]++;break;//右
}
removgdraw(m);
}
else
{
cin>>coto;
if(coto=='B')break;//和color函数里的注释部分功能一样
else if(coto=='C')break;
}
}
}
//recolordraw函数负责执行对颜色设置的指令,这个比较坑,两个颜色不能一样,否则不执行。因为这个,我反复调试hash函数,最后还是无意间发现这个问题的
void Graph::recolordraw(string a,string b)
{
s[6]=cdef(a);
s[7]=cdef(b);
system("CLS");//清屏
system(s);
removgdraw(m);
}
//removgdraw函数负责执行对移动图像的操作,本来想用数组存储图像,但这样有较大的局限,所以改用数组存储相对位置,从而输出
//虽然能够对左方边界和上方边界实现越界操作,但是右方和下方不容易实现(按照我的方法必须知道右下方的边界值),除非使用控制台指令。
//控制台命令可以反馈输出是否越界并发出警告信息,也可以反馈某个字符的位置,不过这又要增加变量(用于记录边界字符状态),增加程序的复杂度,所以就没有实现这个功能
void Graph::removgdraw(int* a)
{
system("CLS");//清屏
int k=0;
if(a[1]>0)
for(int i=0;i<a[1];i++)cout<<endl;
else k=a[1];
for(int i=size-1+k;i>=0;i--)
{
int p=0;
for(int m=0;m<i+a[0];m++)
{
cout<<" ";
}
if(i+a[0]<0)p=-a[0]-i;
for(int j=p;j<2*(size-i)-1;j++)
{
cout<<symbol;
}
cout<<endl;
}
}
//cedf函数负责将用户输入字符串转换成用于系统指令的字符,应用了hash值(因为C++中switch_case对象不能为string),除非你想使用16个if_else语句
inline char Graph::cdef(string a)
{
char m;
const char *h=a.data();
switch(hash_s(h))
{
case vphash("black"): m='0';break;
case vphash("blue"): m='1';break;
case vphash("green"): m='2';break;
case vphash("light_blue"): m='3';break;
case vphash("red"): m='4';break;
case vphash("purple"): m='5';break;
case vphash("yellow"): m='6';break;
case vphash("white"): m='7';break;
case vphash("gray"): m='8';break;
case vphash("baby_blue"): m='9';break;
case vphash("pale_green"): m='A';break;
case vphash("light_green"): m='B';break;
case vphash("reddish"): m='C';break;
case vphash("mauve"): m='D';break;
case vphash("faint_yellow"): m='E';break;
case vphash("light_white"): m='F';break;
default: m='0';break;
}
return m;
}
new main.cpp:
hide()函数是从网上找的,主要就是隐藏光标的显示(但是手动改变窗口大小时,会失效)
#include <iostream>
#include "graph.h"
using namespace std;
HANDLE hout=GetStdHandle(STD_OUTPUT_HANDLE);
COORD coord;
//隐藏光标 函数
void hide()
{
CONSOLE_CURSOR_INFO cursor_info={1,0};
SetConsoleCursorInfo(hout, &cursor_info);
}
int main()
{
while(1)
{
char m;
int n;
system("mode con cols=120 lines=30");//设置窗口大小
cout<<"请输入图形的构成字符与尺寸:"<<endl;
hide();//隐藏光标
cin>>m>>n;
Graph graph(m,n);
system("CLS");
graph.draw();
cout<<"你想要对当前图形进行更多操作吗?是(y),否(n)"<<endl;
char ch;
cin>>ch;
system("CLS");
graph.draw();
if(ch=='n')continue;
else if(ch=='y')
{
cout<<"您可以用方向键调整图形的位置(D),并可以设置图形的前景色和背景色(C)"<<endl
<<"您在操作中可以输入C或D以切换至相应的操作"<<endl
<<"当您需要结束对当前图形的操作时,请输入 B "<<endl;
while(1)
{
if(graph.coto!='B'&&graph.coto!='C'&&graph.coto!='D')cin>>graph.coto;
if(graph.coto=='B')break;
else if(graph.coto=='D')graph.movg();
else if(graph.coto=='C')graph.color();
else cout<<"请输入指定的字符,否则我们无法理解您的意图"<<endl;
}
}
graph.coto='0';
system("CLS");
system("color 0F");//颜色复位
}
return 0;
}
new graph.h:
加一张Graph类图,方便看:
#ifndef GRAPH_H
#define GRAPH_H
#include<cstdlib>
#include <conio.h>
#include <windows.h>
#include <string>
#include<vector>
using namespace std;
// 类Graph的声明
class Graph {
private:
char symbol;
int size;
public:
Graph(char ch, int n); // 带有参数的构造函数
void draw(); //绘制图形
void redraw();
void graph_clear();
void recolordraw(string a,string b); //重绘图形
void color(); //设置前景色与背景色
void movg(); //移动图形
void SCPo(const int x, const int y);
void CHPo(int m,int y);
void TEB(); //优化体验
char cdef(string a);
char coto='0';//用于记录用户对图形的主操作指令
char s[9]={'c','o','l','o','r',' ','0','F','\0'};//初始化,用于执行颜色设置
vector<vector<int>>direction;//利用vector 容器使能够创建任意大小的图案(当然不能超过缓冲区大小啦)
};
new graph.cpp:
用光标函数代替原先的纯数学式输出和清屏指令,体验更好
实现了上下左右都能越界的操作(其实只是设置窗口初始大小,从而实现越界,如果运行时手动改变窗口大小,就会被看出破绽。。哈哈)
#include "graph.h"
#include"hash_type.h"
#include <iostream>
using namespace std;
//默认构造函数,完成私有变量初始化和点位置初始化
Graph::Graph(char ch, int n): symbol(ch), size(n)
{
int m=size-1;
for(int i=0;i<size;i++)
{
for(int j=i*i,k=m-i;j<(i+1)*(i+1)&&k<m+i+1;j++,k++)
{
direction.push_back(vector<int>());//向 direction 第一层添加一个数组
//向已添加的数组中添加元素
//左上原点坐标(0,0),横纵坐标皆为正值
direction[j].push_back(k);//x,即将输出的点在控制台的横坐标
direction[j].push_back(i);//y,即将输出的点在控制台的纵坐标
}
}
}
//draw函数完成初始绘图操作(隐藏redraw函数)
void Graph::draw()
{
redraw();
}
//re(al)draw函数执行真正绘图的操作
void Graph::redraw()
{
for(int i=0;i<size*size;i++)
{
if(direction[i][0]<0||direction[i][1]<0||direction[i][0]>119||direction[i][1]>28)continue;
//设置了120*30的窗口,但是纵轴只能到第28行,如果设置29行,移动图案一直向下就会看到一个菱形(很奇妙,是不是?)
//设置为30或更高,继续向下首先会出现一堆不规则图案,接着就是很规则的。。。。一条条线
else
{
SCPo(direction[i][0],direction[i][1]);
cout<<symbol;
}
}
cout<<endl;
}
//graph_clear 将之前的图形擦除
void Graph::graph_clear()
{
for(int i=0;i<size*size;i++)
{
if(direction[i][0]<0||direction[i][1]<0||direction[i][0]>119||direction[i][1]>28)continue;
else
{
SCPo(direction[i][0],direction[i][1]);
cout<<" ";
}
}
}
//color函数面对用户设置前景色和背景色的操作
void Graph::color()
{
system("CLS");
redraw();
cout<<"现在您可以设置图形的前景色和背景色"
<<"(请先输入背景色,再输入前景色)"<<endl
<<"并且请不要使前景色和背景色相同(这样的操作是无效的,将保留上一状态)"<<endl;
Sleep(2000);
TEB();
while(1)
{
string a,b;
cin>>a;
if(a=="B")
{
coto='B';
break;
}
else if(a=="D")
{
coto='D';
break;
}
cin>>b;
recolordraw(a,b);
}
}
//movg函数面对用户移动图形的操作
void Graph::movg()
{
system("CLS");
redraw();
cout<<"现在您可以通过方向键控制图形的移动"<<endl;
TEB();
while(1)
{
if(getch()==0xE0)
{
graph_clear();
switch(getch())
{
case 72:CHPo(1,-1);break;//上
case 80:CHPo(1,1);break;//下
case 75:CHPo(0,-1);break;//左
case 77:CHPo(0,1);break;//右
}
redraw();
}
else
{
cin>>coto;
if(coto=='B')break;
else if(coto=='C')break;
}
}
}
//recolordraw函数负责执行对颜色设置的指令
void Graph::recolordraw(string a,string b)
{
s[6]=cdef(a);
s[7]=cdef(b);
system(s);
redraw();
//下面是为了优化体验
TEB();
}
//cedf函数负责将用户输入字符串转换成用于系统指令的字符,增加对大写的支持,同样可以增加对其他名词的支持,不过太多了,就没写
inline char Graph::cdef(string a)
{
char m;
const char *h=a.data();
switch(hash_s(h))
{
case vphash("BLACK"): m='0';
case vphash("black"): m='0';break;
case vphash("BLUE"): m='1';
case vphash("blue"): m='1';break;
case vphash("GREEN"): m='2';
case vphash("green"): m='2';break;
case vphash("LIGHT_BLUE"): m='3';
case vphash("light_blue"): m='3';break;
case vphash("RED"): m='4';
case vphash("red"): m='4';break;
case vphash("PURPLE"): m='5';
case vphash("purple"): m='5';break;
case vphash("YELLOW"): m='6';
case vphash("yellow"): m='6';break;
case vphash("WHITE"): m='7';
case vphash("white"): m='7';break;
case vphash("GRAY"): m='8';
case vphash("gray"): m='8';break;
case vphash("BABY_BLUE"): m='9';
case vphash("baby_blue"): m='9';break;
case vphash("PALE_GREEN"): m='A';
case vphash("pale_green"): m='A';break;
case vphash("LIGHT_GREEN"): m='B';
case vphash("light_green"): m='B';break;
case vphash("REDDISH"): m='C';
case vphash("reddish"): m='C';break;
case vphash("MAUVE"): m='D';
case vphash("mauve"): m='D';break;
case vphash("FAINT_YELLOW"): m='E';
case vphash("faint_yellow"): m='E';break;
case vphash("LIGHT_WHITE"): m='F';
case vphash("light_white"): m='F';break;
default: m='0';break;
}
return m;
}
//S(et)C(onsoleCursor)Po(sition)函数重定位光标位置
void Graph::SCPo(const int x, const int y)
{
COORD position;
position.X = x;
position.Y = y;
SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), position);
}
//CH(ANGE)Po(sition)函数改变每个点的位置
void Graph::CHPo(int m,int y)
{
for(auto &v:direction)
{
v[m]+=y;
}
}
//TEB函数使程序在运行使停顿 1s ,并清屏重绘图形,使其他无关的东西全部消失
void Graph::TEB()
{
Sleep(1000);
system("CLS");
redraw();
}
将hash值计算函数存放为头文件,不知道为什么,将函数放在graph.h里会提示重复定义,没办法,只能新建一个头文件。可能是位置不对吧
hash函数可以自定义,只要能将数据转换为一个hash值,都可以称作hash函数(如果不需要加密的话),但运行的高效性和hash值冲突(两个不同字符串(也就是两个不同的输入数据)计算出的hash值一样)的良好解决才是一个hash函数的精髓。
通常编译器自带hash函数(在functional.h头文件里),但是编译器没办法在编译期对switch_case对象进行hash值计算,也就是说以上方的形式编写是无法通过编译的,那么如果想使用编译器自带hash函数,只能事先把将要匹配的数据转换为hash值(constexpr 常量表达式 ?),再写入case对象中。
所以说,使用上述方法是非常麻烦的(尤其是匹配对象非常多或者可变换的时候 )。
其实只要知道hash值的计算方法,就可以自建一个constexpr递归函数,用此函数计算case对象的hash值就可以通过编译了,即上面的代码
但通常我们是看不到编译软件中的函数原型的,可能是我不知道怎么看?(我唯一 一次看到头文件是因为cout输出格式出现问题,编译器提示在头文件中有记载并可以点开看)
所以只能自己编写或使用经典的算法,其中有BKDRhash(暴雪公司),APhash,DJBhash等等,有兴趣的话可以看看这篇转载博客(原文网址好像失效了)
hash算法总结收集 - CSDN博客
https://blog.csdn.net/Together_CZ/article/details/72903106
下面这个是从网上找的,选择这个不是因为算法有多好,而是自带了constexpr函数(哈哈。。毕竟,递归计算hash值时有些东西我也不懂啊)
hash_type.h:
#ifndef HASH_TYPE_H_INCLUDED
#define HASH_TYPE_H_INCLUDED
#include <string>
using namespace std;
using hash_type=uint64_t;
constexpr hash_type prime = 0x100000001B3ull;
constexpr hash_type basis = 0xCBF29CE484222325ull;
hash_type hash_s(char const* str)
{
hash_type ret{basis};
while(*str)
{
ret ^= *str;
ret *= prime;
str++;
}
return ret;
}
constexpr hash_type vphash(char const* str, hash_type last_value = basis)
{
return *str ? vphash(str+1, (*str ^ last_value) * prime) : last_value;
}
#endif // HASH_TYPE_H_INCLUDED
下面让我们来看看效果(不会录制屏幕,所以没办法制成GIF动图):
初始界面:
输入* 12后并选择继续操作后的界面:
继续,输入 D :
使用方向键移动至顶端并越界:
使用方向键移动至左端并越界:
想换个颜色时输入 C :
换好了!(图中文字仅为注释而添加):
换个颜色继续动:
想结束时输入 B 将回到初始界面。
Fraction类:
分数类就没什么知识点了,主要就是运算符重载的问题
Fraction类图:
main.cpp:
#include <iostream>
#include"Fraction.h"
using namespace std;
int main()
{
Fraction a(5,-2);
Fraction b(-1,-2);
Fraction f(1,2);
Fraction c;
Fraction g;
g=a*b;
show(g);
g=a/b;
show(g);
c=a-b;
compare(a,b);
compare(b,f);
show(c);
trshow(c);
double m;
m=tranf_d(c);
cout<<m<<endl;
m=tranf_d(a);
cout<<m<<endl;
m=tranf_d(b);
cout<<m<<endl;
m=tranf_d(f);
cout<<m<<endl;
m=tranf_d(g);
cout<<m<<endl;
Fraction p(1635,-135);
trshow(p);
return 0;
}
Fraction.h:
#ifndef FRACTION_H_INCLUDED
#define FRACTION_H_INCLUDED
class Fraction
{
private:
int top;
int bottom;
public:
Fraction();
Fraction(int t0,int b0);
Fraction(int t0);
Fraction(const Fraction &t);
~Fraction();
Fraction operator+(const Fraction &a);//使用运算符重载时必须使用'operator'关键字,否则会报错(亲测有效)
Fraction operator-(const Fraction &a);
Fraction operator*(const Fraction &a);
Fraction operator/(const Fraction &a);
friend void show(Fraction a);
friend double tranf_d(Fraction a);
friend void compare(Fraction a,Fraction b);
friend int codx(int x,int y);
};
#endif // FRACTION_H_INCLUDED
Fraction.cpp:
#include<iostream>
#include<cmath>
#include<cstring>
#include"Fraction.h"
using namespace std;
inline int codx(int x,int y)//求最大公约数
{
int t=0;
int a,b;
a=x;
b=y;
while(b!=0)
{
t=a%b;
a=b;
b=t;
}
return a;
}
Fraction::Fraction():top(0),bottom(1){}
Fraction::Fraction(int t0,int b0):top(t0),bottom(b0)
{
//if(bottom==0)
//throw runtime_error("Warning!The bottom can't be 0.");//嗯。。。这个最好现在还是不要加,没啥意义
if(bottom<0||(top<0&&bottom<0))
{
top=0-t0;
bottom=0-b0;
}
}
Fraction::Fraction(int t0):top(t0),bottom(1){}
Fraction::Fraction(const Fraction &t):top(t.top),bottom(t.bottom){}
Fraction::~Fraction(){}
//下面是实现分数类四则运算的具体代码
Fraction Fraction::operator+(const Fraction& a)
{
Fraction c;
int p,q,m;
p=bottom*a.bottom/codx(bottom,a.bottom);
q=p/bottom*top+p/a.bottom*a.top;
m=codx(p,q);//m是必要的,因为1/2+5/2应该存储显示为3/1而不是6/2
c.bottom=p/m;
c.top=q/m;
return c;
}
Fraction Fraction::operator-(const Fraction& a)
{
Fraction c;
int p,q,m;
p=bottom*a.bottom/codx(bottom,a.bottom);
q=p/bottom*top-p/a.bottom*a.top;
m=codx(p,q);
c.bottom=p/m;
c.top=q/m;
return c;
}
Fraction Fraction::operator*(const Fraction& a)
{
Fraction c;
int m,n,q;
m=bottom*a.bottom;
n=top*a.top;
q=codx(m,fabs(n));
c.top=n/q;
c.bottom=m/q;
return c;
}
Fraction Fraction::operator/(const Fraction& a)
{
Fraction c;
int m,n,q;
m=bottom*a.top;
n=top*a.bottom;
q=codx(fabs(m),fabs(n));
c.top=n/q;
c.bottom=m/q;
return c;
}
//这些都是非成员函数,通过friend成为友元使其能够访问类的私有变量
//其实这些函数都可以被定义为成员函数,但是成为成员函数后它们的使用方式对用户不太友好,比如 compare(a,b); 写成 a.compare(b); 显然和我们的习惯不符
//这也是编程语言面向对象的意义所在(数据保护是另一方面)
void show(Fraction a)//输出很揪心,因为C++没有反射,所以无法以实例名输出(如果有人知道怎么操作的话,请在评论中说明,谢谢)
{
cout<<a.top<<'/'<<a.bottom<<endl;
}
//以传统方式输出分数
void trshow(Fraction a)
{
//计算分子分母的位数以确定输出线的长度
int topd=digit(fabs(a.top));
int botd=digit(fabs(a.bottom));
int m=max(topd,botd);
if(a.top<0)
{
cout<<'-';
couof((m-topd)/2," ");
}
else
couof((m-topd)/2+1," ");
cout<<fabs(a.top)<<endl;
couof(m+1,"━");
cout<<endl;
couof((m-botd)/2+1," ");
cout<<a.bottom<<endl;
}
double tranf_d(Fraction a)
{
double m;
m=a.top/a.bottom;//感谢老师指出这个问题,a.top和a.bottom都是整型,必须乘以一个1.0才能转换为double型
return m;
}
void compare(Fraction a,Fraction b)
{
if(a.top*b.bottom==a.bottom*b.top)
cout<<a.top<<'/'<<a.bottom<<'='<<b.top<<'/'<<b.bottom<<endl;
else
cout<<a.top<<'/'<<a.bottom<<
(a.top*b.bottom>a.bottom*b.top?'>':'<')
<<b.top<<'/'<<b.bottom<<endl;
}
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET 9 new features-C#13新的锁类型和语义
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
· 你所不知道的 C/C++ 宏知识
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· 不到万不得已,千万不要去外包
· C# WebAPI 插件热插拔(持续更新中)
· 会议真的有必要吗?我们产品开发9年了,但从来没开过会
· 如何打造一个高并发系统?
· 《SpringBoot》EasyExcel实现百万数据的导入导出