💯💯💯

思路:
一个栈专门用来插入数据
一个栈专门用来出数据

比如如果栈里有5个数据,而要根据队列的特性,出队列肯定出的是队头数据,也就是1,而在栈里,怎么才能将数据1删除掉呢?
我们的做法是:将栈1的数据全部导入到栈2去。这样由原来的数据就倒过来了,那删除栈顶元素即可。

删除栈顶元素之后,我们分析发现,这时的栈2如果再进行删除,就和队列的删除操作是一致的了,不需要再导回去,如果要再删除队头数据2,直接让栈2删除即可。
所有我们发现经过一次导数据之后,栈2就完全可以当成pop数据的。
那我们如果想插入数据,该怎么插入呢?
根据队列特性,我们只能从队尾插入,也就是元素5的后面插入数据。该怎么插呢?将数据插入栈2里?那可不行,将数据插入栈2中后,原来的顺序就乱了。所以我们将数据插入到栈1去,比如我们要插入数据6,7,8.直接插入到栈1即可。

这样就不会影响栈2出数据的顺序了,并且插入的顺序也不受出数据的影响。
唯一需要注意的是,当栈2的数据都删除空了,这时就需要将栈1的数据再导入到栈2中,这样就能接着删除队列中的数据了。

所以我们可以直接定义两个栈,一个栈用来插入数据,一个栈用来删除数据。
typedef struct
{
ST pushst;
ST popst;
} MyQueue;
不过在定义之前我们需要写一个栈的数据结构,因为C语言是没有自己的栈的。
typedef int STData;
typedef struct Stack
{
int* a;
int top;
int capicty;
}ST;
void STInit(ST*ps);//初始化栈表
void STDestroy(ST* ps);//销毁栈表
void STpush(ST* ps,STData x);//压栈,在栈顶压入一个元素
void STpop(ST* ps);//出栈,在栈顶弹出一个元素。
STData STTop(ST* ps);//访问栈顶元素
int STSize(ST* ps);
int STEmpty(ST*ps);//
void STInit(ST* ps)//初始化栈表
{
assert(ps);//判断结构体指针不为NULL;
//一上来可以给栈表初始化容量
ps->a = (STData*)malloc(sizeof(STData) * 4);
if (ps->a == NULL)//判断是否开辟成功
{
perror("malloc");
}
//初始话容量为4
ps->capicty = 4;
ps->top = 0;//top=0 表示的是指向栈顶元素的下一个位置
//top如果为-1,则表示栈顶元素的位置
}
void STDestroy(ST* ps)//销毁栈表
{
assert(ps);
free(ps->a);
ps->a = NULL;
ps->capicty = 0;
ps->top = 0;
}
void STpush(ST* ps, STData x)//压栈,在栈顶压入一个元素--在压入之前也要考虑是否需要增容
{
assert(ps);//断言判断
if (ps->top == ps->capicty)
{
//增容
STData* tmp = (STData*)realloc(ps->a, sizeof(STData) * ps->capicty * 2);
if (tmp == NULL)
{
perror("realloc");
}
ps->a = tmp;
ps->capicty *= 2;
}
ps->a[ps->top] = x;//将元素压入栈顶,一开始top是0,当元素进去后,再让top指向该栈顶元素后一个位置
ps->top++;
}
int STEmpty(ST* ps)
{
assert(ps);
return ps->top == 0;
}
void STpop(ST* ps)//出栈,在栈顶弹出一个元素。
{
assert(ps);
//删除元素之前要检查栈表是否还有元素可删
assert(!STEmpty(ps));//当栈表为NULL是断言
ps->top--;
}
int STSize(ST* ps)//计算栈表长度
{
assert(ps);
return ps->top;
//top的长度就是栈表的长度
}
STData STTop(ST* ps)//访问栈顶元素
{
assert(ps);
return ps->a[ps->top - 1];
}
接下来就是获取一个指向 MyQueue队列的指针。
MyQueue* myQueueCreate() //发现没有传入参数并且返回值是指向栈的指针说明里面是用malloc开辟的内存而不是在栈上开辟的
{
MyQueue *q=(MyQueue*)malloc(sizeof(MyQueue));
if(q==NULL)
{
perror("malloc");
}
STInit(&q->pushst);//一开始需要对两个栈进行初始化
STInit(&q->popst);
return q;
}
入队列,即插入数据,直接在push栈进行插入即可,不用担心顺序什么的,因为当pop栈没有数据时,就会让push栈导数据过来,这时的顺序就符合队列的要求了
void myQueuePush(MyQueue* obj, int x)
{
//只管往pushst里插入即可,不需要管其他
STpush(&obj->pushst,x);
}
出队列呢就指望pop栈即可,不过在出队列之前,我们需要检查一下pop栈里是否为空,如果为空,就需要将push栈里的数据导过来,如果不为空,那就可以进行删除操作了。
int myQueuePop(MyQueue* obj)
{
//需要讨论下,当pop这个栈为空时,需要将push栈中的数据导过来
if(STEmpty(&obj->popst))
{
while(!STEmpty(&obj->pushst))//将push栈中的所有数据都导过去即可
{
STpush(&obj->popst,STTop(&obj->pushst));
STpop(&obj->pushst);//删除push栈里的这个数据
}
}
//走到这里有两种情况,可能push栈里的数据导光了,也可能是pop栈里本来就有数据,不为空
int top=STTop(&obj->popst);//将这个栈顶元素记录下来,最后需要返回
STpop(&obj->popst);
return top;
}
获取队头数据,这个操作与刚刚的删除操作基本一样,只不过该操作不需要将数据删除,直接返回栈顶元素即可,所以大部分代码是一样的。
int myQueuePeek(MyQueue* obj)
{
//这里跟pop数据很像,直接return 栈顶元素即可
//需要讨论下,当pop这个栈为空时,需要将push栈中的数据导过来
if(STEmpty(&obj->popst))
{
while(!STEmpty(&obj->pushst))//将push栈中的所有数据都导过去即可
{
STpush(&obj->popst,STTop(&obj->pushst));
STpop(&obj->pushst);
}
}
//走到这里有两种情况,可能push栈里的数据导光了,也可能是pop栈里本来就有数据,不为空
return STTop(&obj->popst);
}
判断队列是否为空其实很简单,因为该队列是由两个栈构成,当两个栈都为空时,则该队列肯定为空。
bool myQueueEmpty(MyQueue* obj)
{
return STEmpty(&obj->pushst)&&STEmpty(&obj->popst);
}
销毁队列也简单,要想真正的释放该队列,需要理解该队列的结构是如何构成的。
该队列是由两个栈构成,栈是由数组构成。

释放空间从里到外释放,先释放两个栈空间,再释放队列空间。
void myQueueFree(MyQueue* obj)
{
STDestroy(&obj->pushst);
STDestroy(&obj->popst);
free(obj);
}
typedef int STData;
typedef struct Stack
{
int* a;
int top;
int capicty;
}ST;
void STInit(ST*ps);//初始化栈表
void STDestroy(ST* ps);//销毁栈表
void STpush(ST* ps,STData x);//压栈,在栈顶压入一个元素
void STpop(ST* ps);//出栈,在栈顶弹出一个元素。
STData STTop(ST* ps);//访问栈顶元素
int STSize(ST* ps);
int STEmpty(ST*ps);//
void STInit(ST* ps)//初始化栈表
{
assert(ps);//判断结构体指针不为NULL;
//一上来可以给栈表初始化容量
ps->a = (STData*)malloc(sizeof(STData) * 4);
if (ps->a == NULL)//判断是否开辟成功
{
perror("malloc");
}
//初始话容量为4
ps->capicty = 4;
ps->top = 0;//top=0 表示的是指向栈顶元素的下一个位置
//top如果为-1,则表示栈顶元素的位置
}
void STDestroy(ST* ps)//销毁栈表
{
assert(ps);
free(ps->a);
ps->a = NULL;
ps->capicty = 0;
ps->top = 0;
}
void STpush(ST* ps, STData x)//压栈,在栈顶压入一个元素--在压入之前也要考虑是否需要增容
{
assert(ps);//断言判断
if (ps->top == ps->capicty)
{
//增容
STData* tmp = (STData*)realloc(ps->a, sizeof(STData) * ps->capicty * 2);
if (tmp == NULL)
{
perror("realloc");
}
ps->a = tmp;
ps->capicty *= 2;
}
ps->a[ps->top] = x;//将元素压入栈顶,一开始top是0,当元素进去后,再让top指向该栈顶元素后一个位置
ps->top++;
}
int STEmpty(ST* ps)
{
assert(ps);
return ps->top == 0;
}
void STpop(ST* ps)//出栈,在栈顶弹出一个元素。
{
assert(ps);
//删除元素之前要检查栈表是否还有元素可删
assert(!STEmpty(ps));//当栈表为NULL是断言
ps->top--;
}
int STSize(ST* ps)//计算栈表长度
{
assert(ps);
return ps->top;
//top的长度就是栈表的长度
}
STData STTop(ST* ps)//访问栈顶元素
{
assert(ps);
return ps->a[ps->top - 1];
}
typedef struct
{
ST pushst;
ST popst;
} MyQueue;
MyQueue* myQueueCreate()
{
MyQueue *q=(MyQueue*)malloc(sizeof(MyQueue));
if(q==NULL)
{
perror("malloc");
}
STInit(&q->pushst);
STInit(&q->popst);
return q;
}
void myQueuePush(MyQueue* obj, int x)
{
//只管往pushst里插入即可,不需要管
STpush(&obj->pushst,x);
}
int myQueuePop(MyQueue* obj)
{
//需要讨论下,当pop这个栈为空时,需要将push栈中的数据导过来
if(STEmpty(&obj->popst))
{
while(!STEmpty(&obj->pushst))//将push栈中的所有数据都导过去即可
{
STpush(&obj->popst,STTop(&obj->pushst));
STpop(&obj->pushst);
}
}
//走到这里有两种情况,可能push栈里的数据导光了,也可能是pop栈里本来就有数据,不为空
int top=STTop(&obj->popst);
STpop(&obj->popst);
return top;
}
int myQueuePeek(MyQueue* obj)
{
//这里跟pop数据很像,直接return 栈顶元素即可
//需要讨论下,当pop这个栈为空时,需要将push栈中的数据导过来
if(STEmpty(&obj->popst))
{
while(!STEmpty(&obj->pushst))//将push栈中的所有数据都导过去即可
{
STpush(&obj->popst,STTop(&obj->pushst));
STpop(&obj->pushst);
}
}
//走到这里有两种情况,可能push栈里的数据导光了,也可能是pop栈里本来就有数据,不为空
return STTop(&obj->popst);
}
bool myQueueEmpty(MyQueue* obj)
{
return STEmpty(&obj->pushst)&&STEmpty(&obj->popst);
}
void myQueueFree(MyQueue* obj)
{
STDestroy(&obj->pushst);
STDestroy(&obj->popst);
free(obj);
}
/**
* Your MyQueue struct will be instantiated and called as such:
* MyQueue* obj = myQueueCreate();
* myQueuePush(obj, x);
* int param_2 = myQueuePop(obj);
* int param_3 = myQueuePeek(obj);
* bool param_4 = myQueueEmpty(obj);
* myQueueFree(obj);
*/
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
我有一个涉及多台机器、消息队列和事务的问题。因此,例如用户点击网页,点击将消息发送到另一台机器,该机器将付款添加到用户的帐户。每秒可能有数千次点击。事务的所有方面都应该是容错的。我以前从未遇到过这样的事情,但一些阅读表明这是一个众所周知的问题。所以我的问题。我假设安全的方法是使用两阶段提交,但协议(protocol)是阻塞的,所以我不会获得所需的性能,我是否正确?我通常写Ruby,但似乎Redis之类的数据库和Rescue、RabbitMQ等消息队列系统对我的帮助不大——即使我实现某种两阶段提交,如果Redis崩溃,数据也会丢失,因为它本质上只是内存。所有这些让我开始关注erlang和
我正在尝试使用Curbgem执行以下POST以解析云curl-XPOST\-H"X-Parse-Application-Id:PARSE_APP_ID"\-H"X-Parse-REST-API-Key:PARSE_API_KEY"\-H"Content-Type:image/jpeg"\--data-binary'@myPicture.jpg'\https://api.parse.com/1/files/pic.jpg用这个:curl=Curl::Easy.new("https://api.parse.com/1/files/lion.jpg")curl.multipart_form_
无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD
华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o
本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01 客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02 数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit