请教一道面试题 模拟收银台

shortgun 2008-06-05 03:13:19
某超市的收银处有n个柜台(n>=1),且至少有一个为实
习柜台,且该柜台总在最后(例如一共8个柜台,则8号
柜台为实习柜台)

一般柜台每处理一件商品的时间为1分钟
实习柜台每处理一件商品的时间为2分钟

结帐时默认客户为两种
1.挑选最短队伍结帐的客户 (A类客户)
2.挑选队伍末尾结帐商品数最少的客户,既当前所有队
列中即将在他前面结帐的商品数最少(B类客户)


现输入为:
任意柜台数n 及矩阵(客户及购物信息)

例如:
3
A 1 3
B 2 5
A 2 3
A 2 9
B 5 6

表示:
当前有3个柜台(第一行表示柜台数)及客户信息:
一共有五个客户,某A类客户在第1分钟到的,他有3件商
品,
某B类客户在第2分钟到的,他有5件商品。。。依次类推

对于第二分钟同时到到达的:
B 2 5
A 2 3
A 2 9
处理规则为:
如果A B两类客户同一时间到达,先处理A类客户
如果两个A/B类客户同时到达且商品数相同,处理先输
入的数据
如果两个A/B类客户同事到达且商品数不同,先处理商
品数少的客户


输出结果要求为:总共需要多少分钟,将这些客户全部
处理完成。
...全文
269 11 打赏 收藏 转发到动态 举报
写回复
用AI写文章
11 条回复
切换为时间正序
请发表友善的回复…
发表回复
冷月清晖 2008-06-05
  • 打赏
  • 举报
回复
[Quote=引用 4 楼 shortgun 的回复:]
不行,忘了说了这个题目不能用多线程.
[/Quote]

真正的系统一般都是多线程程序,

消息机制总可以用吧,

否则只能用最土的办法了,遍历?
leelittlelong 2008-06-05
  • 打赏
  • 举报
回复
简单的想法:

每个柜台用3个变量来记录。 int nTime, int nPerson. int nTime.

然后依次读入输入的数据,(读入的时候也要分情况考虑)。
shortgun 2008-06-05
  • 打赏
  • 举报
回复
不行,忘了说了这个题目不能用多线程.
冷月清晖 2008-06-05
  • 打赏
  • 举报
回复
可以用多线程,最好用线程池来模拟
sss__www 2008-06-05
  • 打赏
  • 举报
回复

∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞

力争成为中国最大的架构师群联盟,架构师技术交流群:62402336正式开放!!!

已经上传的顶级软件产品的架构分析,本群资料仅供研究学习,不得商用!!!
google 、
eBay、
Youtube、
淘宝等
......
技术文章包括:
《自己动手写操作系统》
《搜索引擎-原理、技术与系统》
《企业应用架构模式》
......
重要的RUP实例
设计模式精解
......
资料陆续上传中
∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞∽∝∞
shortgun 2008-06-05
  • 打赏
  • 举报
回复
自己顶一下,大家讨论下怎么设计这个问题.
冷月清晖 2008-06-05
  • 打赏
  • 举报
回复
好吧。。。

#include "stdio.h"
#include "stdlib.h"
#include "header/bank_sim.h"

extern EventList ev; /* 事件表 */
extern Event en; /* 事件 */
extern size_t TotalTime, CustNum; /* 累计客户逗留时间,客户数 */

int main( void )
{
if ( !bank_open() ) {
return EXIT_FAILURE; /* 初始化失败,退出 */
}
while ( ev->next ) {
DelFirst(); /* 删除事件表第一个节点,并把值赋给en */
if ( en.NType == ARRIVAL ) {
if ( !CustArr() ) {
return EXIT_FAILURE;
}
} else if ( !CustDepart() ) {
return EXIT_FAILURE;
}
}
printf( "Average Stay Time:%.2f minutes\n", (float)TotalTime / CustNum );

puts( "Press ENTER to quit..." );
getchar();
DestroyAll();

return EXIT_SUCCESS;
}


#ifndef BANK_SIM_H
#define BANK_SIM_H

#include <stddef.h>

#define COUNTER 8 /* 银行共有 8 个营业窗口 */
#define ARRIVAL COUNTER /* 客户到达事件 */

typedef struct {
size_t OccurTime, NType; /* 事件发生时刻;事件类型 */
} Event; /* 链表数据元素类型 */

typedef struct Enode {
Event event;
struct Enode *next;
} Enode, *EventList;

typedef struct {
size_t ArrTime, Duration; /* 到达时刻;办理时间 */
} QElem; /* 队列数据元素类型 */

typedef struct Qnode {
QElem Cust; /* 客户记录 */
struct Qnode *next;
} Qnode, *Qptr;

typedef struct {
Qptr front, rear; /* 队头指针,队尾指针 */
} LinkQueue;

int bank_open( void ); /* 初始化操作 */
int CustArr( void ); /* 处理客户到达事件 */
int CustDepart( void ); /* 处理客户离开事件 */
void DelFirst( void ); /* 删除事件表第一个节点,并把值赋给en */
int DeQueue( LinkQueue * ); /* 出队 */
void DestroyAll( void ); /* 销毁链表和队列 */
void DestroyList( EventList ); /* 销毁链表 */
void DestroyQueue( LinkQueue * ); /* 销毁队列 */
int EnQueue( LinkQueue *, QElem ); /* 进队 */
EventList InitList( void ); /* 创建链表 */
int InitQueue( LinkQueue * ); /* 建立空队列 */
size_t Minimun( void ); /* 求长度最短队列 */
int OrderInsert( EventList, Event ); /* 插入事件表 */
size_t QueueLength( LinkQueue ); /* 计算队列长度 */

#endif


#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "../header/bank_sim.h"

#define CLOSETIME 480 /* 每天工作 8 小时( 60 * 8 = 480 ) */
#define MAX_TIME 30 /* 事务处理时间不超过 30 分钟 */
#define INTERVAL 5 /* 每 5 分钟至少来一位客人 */

EventList ev; /* 事件表 */
Event en; /* 事件 */
LinkQueue q[COUNTER]; /* 客户队列 */
QElem customer; /* 客户记录 */
size_t TotalTime, CustNum; /* 累计客户逗留时间,客户数 */

/* Begin of bank_open 05-9-17 00:40 */
int bank_open( void ) /* 初始化操作 */
{
size_t i;

srand( time(NULL) ); /* 生成随机数种子 */

CustNum = TotalTime = 0; /* 初始化累计时间和客户数 */
if ( !( ev = InitList() ) ) { /* 初始化事件链表 */
return 0; /* 创建失败,返回 0 */
}
en.OccurTime = 0;
en.NType = ARRIVAL; /* 设定第一个客户到达事件 */

if ( !OrderInsert(ev, en) ) { /* 插入事件表 */
DestroyList( ev );
return 0; /* 插入失败,返回0 */
}
for ( i = 0; i < COUNTER; ++i ) { /* 置空队列 */
if ( !InitQueue(&q[i]) ) {
DestroyList( ev );
for ( i = 0; q[i].front; ++i ) {
DestroyQueue( &q[i] );
}
return 0; /* 创建队列失败,返回 0 */
}
}

return 1;
} /* End of bank_open */

/* Begin of CustArr 05-9-17 08:20 */
int CustArr( void ) /* 处理客户到达事件 */
{
int durtime, intertime;
QElem e;
size_t i;

++CustNum;
/* 生成随机数 */
durtime = rand() / (RAND_MAX / MAX_TIME + 1) + 1; /* 事务处理时间不超过 MAX_TIME 分钟 */
intertime = rand() / (RAND_MAX / INTERVAL); /* INTERVAL 分钟内有一位客人到达银行,0 表示客户同时到达 */
e.ArrTime = en.OccurTime;
e.Duration = durtime;
en.OccurTime += intertime; /* 下一客户到达时间 */
if ( en.OccurTime < CLOSETIME ) { /* 银行尚未关门,插入事件表 */
if ( !OrderInsert(ev, en) ) { /* 插入失败 */
DestroyAll();
return 0;
}
}

i = Minimun(); /* 求长度最短队列 */
if ( !EnQueue(&q[i], e) ) {
DestroyAll();
return 0;
}

/* 如果此时 i 号窗口只有一位顾客,则马上把他的离开事件插入事件表 */
if ( QueueLength(q[i]) == 1 ) {
en.NType = i;
en.OccurTime = e.ArrTime + durtime;
if ( !OrderInsert(ev, en) ) {
DestroyAll();
return 0;
}
}

return 1;
} /* End of CustArr */

/* Begin of CustDepart 05-9-17 09:00 */
int CustDepart( void ) /* 处理客户离开事件 */
{
Qptr tmp;
size_t i = en.NType;

if ( !DeQueue( &q[i] ) ) { /* 删除排头客户 */
DestroyAll();
return 0;
}

TotalTime = TotalTime + en.OccurTime - customer.ArrTime; /* 累计逗留时间 */
if ( q[i].front != q[i].rear ) {
tmp = q[i].front->next;
customer = tmp->Cust;
en.OccurTime += customer.Duration;
if ( !OrderInsert(ev, en) ) {
DestroyAll();
return 0;
}
}

return 1;
} /* End of CustDepart */

/* Begin of DelFirst 05-9-17 00:45 */
void DelFirst( void ) /* 删除事件表第一个节点,并把值赋给en */
{
EventList tmp = ev->next;

ev->next = tmp->next;
en = tmp->event;

free(tmp);
} /* End of DelFirst */

/* Begin of DeQueue 05-9-17 08:35 */
int DeQueue(LinkQueue *Q) /* 出队 */
{
Qptr tmp;

if ( Q->front == Q->rear ) {
return 0; /* 如果队列已空,返回 0 */
}

tmp = Q->front->next;
customer = tmp->Cust;
Q->front->next = tmp->next;
if ( Q->rear == tmp ) { /* 如果队尾元素被删除,则队尾指针要指向头结点 */
Q->rear = Q->front;
}
free(tmp);

return 1;
} /* End of DeQueue */

/* Begin of DestroyAll 05-9-17 01:15 */
void DestroyAll( void ) /* 销毁链表和队列 */
{
size_t i;

DestroyList( ev );
for ( i = 0; i < COUNTER; ++i ) {
DestroyQueue( &q[i] );
}
} /* End of DestroyAll */

/* Begin of DestroyList 05-9-16 23:00 */
void DestroyList( EventList L ) /* 销毁链表 */
{
EventList T = L;

for ( L = L->next; L; T = L, L = L->next ) {
free( T );
}
free( T );
} /* End of DestroyList */

/* Begin of DestroyQueue 05-9-17 00:30 */
void DestroyQueue(LinkQueue *Q) /* 销毁队列 */
{
while ( Q->front ) {
Q->rear = Q->front->next;
free( Q->front );
Q->front = Q->rear;
}
} /* End of DestroyQueue */

/* Begin of EnQueue 05-9-17 01:20 */
int EnQueue(LinkQueue *Q, QElem e) /* 进队 */
{
Qptr tmp = malloc( sizeof *tmp );

if ( !tmp ) {
return 0; /* 进队失败,返回 0 */
}

tmp->Cust = e;
tmp->next = NULL;
Q->rear->next = tmp;
Q->rear = tmp;

return 1;
} /* End of EnQueue */

/* Begin of InitList 05-9-16 22:30 */
EventList InitList( void ) /* 创建链表 */
{
EventList h;

if ( !( h = malloc(sizeof *h) ) ) { /* 创建失败,返回 NULL */
return NULL;
}
h->next = NULL;

return h;
} /* End of InitList */

/* Begin of InitQueue 05-9-16 22:00 */
int InitQueue(LinkQueue *Q) /* 建立空队列 */
{
Q->rear = Q->front = malloc( sizeof *Q->front );
if ( !Q->front ) { /* 创建失败,返回 0 */
return 0;
}
Q->front->next = NULL;

return 1;
} /* End of InitQueue */

/* Begin of Minimun 05-9-17 01:05 */
size_t Minimun( void ) /* 求长度最短队列 */
{
size_t h = 0, i, j, k;

j = QueueLength(q[0]);
for ( i = 1; i < COUNTER; ++i ) {
k = QueueLength(q[i]);
if ( j > k ) {
j = k;
h = i;
}
}

return h;
} /* End of Minimun */

/* Begin of OrderInsert 05-9-16 23:15 */
int OrderInsert(EventList ev, Event a) /* 插入事件表 */
{
EventList tmp, h = ev, e = ev->next;

if ( !( tmp = InitList() ) ) { /* 分配空间失败,返回 0 */
return 0;
}
/* 插入 */
tmp->event = a;
while ( e && (a.OccurTime > e->event.OccurTime) ) {
h = e;
e = e->next;
}
h->next = tmp;
tmp->next = e;

return 1;
} /* End of OrderInsert */

/* Begin of QueueLength 05-9-17 01:00 */
size_t QueueLength(LinkQueue Q) /* 计算队列长度 */
{
size_t i;

for ( i = 0; Q.front->next; ++i ) {
Q.front = Q.front->next;
}

return i;
} /* End of QueueLength */

本文链接网址http://www.sfcode.cn/soft/004232243.htm
leelittlelong 2008-06-05
  • 打赏
  • 举报
回复
错了 不是两个nTime
一个表示时间 一个表示人 一个表示货物

shortgun 2008-06-05
  • 打赏
  • 举报
回复
[Quote=引用 8 楼 bitxinhai 的回复:]
看看数据结构
课本上的银行队列模拟,
和你的题目很像的!!!
[/Quote]

能告诉我哪个课本瓦? 多谢拉
bitxinhai 2008-06-05
  • 打赏
  • 举报
回复
看看数据结构
课本上的银行队列模拟,
和你的题目很像的!!!
shortgun 2008-06-05
  • 打赏
  • 举报
回复
[Quote=引用 5 楼 leelittlelong 的回复:]
简单的想法:

每个柜台用3个变量来记录。 int nTime, int nPerson. int nTime.

然后依次读入输入的数据,(读入的时候也要分情况考虑)。

[/Quote]

两个nTime?分别表示什么?

65,210

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧