您的位置:首页 > 其它

创新工场笔试题整理

2011-09-19 10:04 232 查看
转自:http://www.cnblogs.com/passingcloudss/archive/2011/05/04/2034209.html

【作者按】网上搜集的题目,自己整理了一下,写了个解答,所有程序均在VS2010上调试通过!

如果各位看官有更好更高效更巧妙的方法,请不吝指教!

【一】三道题程序题,要求一个小时做完,而且提前交卷有加分。

题目1:把一串英文句子按单词反序输出。如:"good moring" -> "moring good"。

题目2:输入一个正整数N,输出大于N且最接近这个数的素数。

题目3:用数组实现排序二叉树。

【出处】http://mcs.sysu.edu.cn/user/longt/Article_1790

【一解】

题目1:我的想法是把句子的每个字符遍历一遍,找到每个单词首字母的位置,然后再倒过来访问。


存放单词首字母序号的结点定义


句子反序输出

我的解法:

a.整个字符串翻转。

b.以空格隔开的每个子串翻转。

题目2:我的想法是写一个质数判断函数,然后从N+1往后开始寻找。感觉应该有更佳的方法。


判断一个数是否是质数


寻找比N大的最小质数

题目3:我不知道用数组实现“排序二叉树”和“实现二叉树”有什么区别。我觉得如果是用数组实现二叉树的话还是很简单的,就是开一个数组,数组中每个元素存放一个树结点,每个树结点包含有“parent”、“leftchild”和“rightchild”三个int。

以下摘自原文作者:“思路: 一颗树中每个结点只有唯一一个父亲,而数组中每个元素只能存一个值,所以把这个值存为父亲结点,根节点值定-1就好。但是要求实现排序二叉树,所以还需要记录每个结点,因此用个pair<T,T>,最后通过数组pair<T,T> tree[] 来实现。”

【二】请问下面这个程序输出是什么?

#include <stdio.h>

union A
{
int i;
char c[2];
};

int main()
{
A a;
a.c[0] = 10;
a.c[1] = 1;
printf("%d", a.i);
return 0;
}


【出处】http://topic.csdn.net/u/20101012/21/3aa793bb-c8e5-49c7-b72f-ca985b0d5c4e.html?32141

【二解】union A开辟了一个4字节的空间,如果是x86的电脑,那么是按小端格式(Little-endian)存储数据的,那么A.c[0]指向的就是i的最后一个字节。举列来说:如果我给A.i赋值为10,那么实际内存中是这么存储的:

(关于何为小端格式,请参见/article/5594550.html

地址  0    1    2    3

数据  0x0A  0x00  0x00  0x00

c[0]=0x0A,c[1]=0x00,c[2]=0x00,c[3]=0x00;

对于未赋值的字节块,编译器在调试版本(DEBUG)会自动填充0xCC,因此本题涉及的内存实际上是这样的:

地址  0    1    2    3

数据  0x0A  0x01  0xCC  0xCC

最终输出结果应该是0xCCCC010A=-859045622。

这里再重申一下,该结果是在x86电脑的DEBUG模式下的输出!

一个诡异的发现,如果把程序改为:

#include <stdio.h>
union
{
int i;
char c[2];
}A;
void main()
{
A.c[0]=10;
A.c[1]=1;
printf("%d", A.i);
}


输出结果是0x0000010A=266!
引申:通过这道题的方法,我们可以判断CPU是小端格式还是大端格式!

我的解释:
第一个程序:因为局部变量a没有初始化,所以是乱码。
第二个程序:因为A是全局变量,被编译器自动初始化为0,所以输出为266.

【三】笔试题量很小,答题时间1个小时。1道填空题,9道左右选择题,最后一道编程题。

题目1:对于int a = 65536 + 1024 + 8 + 1;int b = f(a);执行后b等于多少?

int f(int x)
{
int c = 0;
while(x != 0)
{
x = x & (x-1);
c++;
}
return c;
}


题目2:5个骰子,六个面分别标有1~6,现在将五个同时随机投掷,五个点数之和为下面哪个点的概率最大?
(A)14  (B)15  (C)17  (D)20

题目3:请问c等于多少?

unsigned long c = 0;
char a = 0x48;
char b = 0x52;
c = b<<8 | a;


题目4:编程题

A、B两个量杯,容量分别为M升、N升,现在要用A和B给另一个量杯C盛水K升,C量杯足够大,备用水无限。编程输出每一个步骤三个杯子中的水量。比如:输出(0,0,0), (M,0,0)等。

【出处】http://polaris1119.iteye.com/blog/769423

【三解】

题目1:这个程序就是求二进制数中1的个数,所以答案是4。

题目2:

思路一:先算每个骰子掷出点数的期望值。每面的概率都是1/6,单个骰子的期望为

1*1/6 + 2*1/6 + 3*1/6 + 4*1/6 + 5*1/6 + 6*1/6 = 3.5

5个骰子相互独立,期望就是3.5*5=17.5,17或18为最接近的点数,选(C)。

思路二:掷出的点数最小为5,即1+1+1+1+1只有一种可能;掷出的点数最大为30,即6+6+6+6+6也只有一种可能。

点数的分布应该是中间大、两头小,并且应该是左右对称的,所以最可能的点数应该是(5+30)/2=17.5,选(C)。

题目3:b<<8是把b向左位移8位,相当于把b乘以2的8次方,得到0x5200;

再和a位或,由于0x5200的后8位是0,位或运算等效于加法运算,得到c=0x5248=21064。

引申:如果c不是定义成unsigned long,而是一个小字节的类型,比如c定义成char型,那么c=0x48。

附注:数学运算会将操作数提升到int型,如果《4byte的话。

题目4: 这道题比较复杂,我们先来分析一下思路。首先,不失一般性,不妨设M>N,如果我们能实现 K1=K%N 升,那么只需再用B量杯给C量杯加上 (int)K/N 杯即可。

设M和N的最大公约数为L,于是,不管我们对M和N进行几次和运算或差运算,得到的数仍是L的倍数。也就是说,不管怎样操作,使用A量杯和B量杯,我们只能得到“L的倍数”升。注意,这只是说,能实现的一定是L的倍数,但不一定每个L的倍数都能实现。

另一方面,根据数论中的Bezout等式,存在整数a和b,使得aM+bN=L成立。

事实上,假如我们找到了一组满足a0*M+b0*N=L的(a0,b0),那么对于任意整数t,a=a0+t*N/L,b=b0-t*M/L都能满足aM+bN=L。

所以,我们一定能找到一组a>0、b<0,使aM+bN=L,那么我们只需往C量杯中倒入a杯A量杯的水,再倒走(-b)杯B量杯的水,C量杯中就能得到L升水!

结合上面两方面,我们得到,C量杯能实现K升水 当且仅当 K1是L的自然数倍。

具体实现的思路如下:先求出M和N的最大公约数L,判断K1是否能够整除L;然后不断给C倒入M升水(中途如果C水量超过N,就倒走N升水),直到C水量为K1;最后再给C中倒入若干次N升水即可。

具体代码如下:


打印每次操作


辗转相除法求最大公约数


解法主函数
事实上,这个实现方法不是很完美,比如M=5,N=4,K=10,本来只需用2杯A量杯的水即可;但是按照上述算法,是倒入一杯A量杯的水,随后立即倒走一杯B量杯的水,如此往复10次,得到10升,这显然是不明智的。其实可以在程序中加以改进,只需在每准备倒出一杯B量杯的水时判断一下:现在C中的水量离K升是不是差M的整数倍即可。


解法主函数(改进)
这样的算法就比较完美了!

【四】一些小题目

题目1:路由器与交换机的区别。

题目2:进程与线程的差别:

A、操作系统只调度进程,不调度线程

B、线程共享内存地址空间,进程不共享

C、线程间可以共享内存数据,但进程不可以

D、进程间可以通过IPC通信,但线程不可以

题目3:下面排序算法的时间复杂度不是O(nlogn)的是:

A、二分法插入排序  B、快速排序  C、归并排序  D、堆排序

题目4:利用KMP算法实现字符串匹配。

题目5:某工作有5道工序,某个工作不能在最后做,请问有多少种工作情况。

题目6:内存中有3页,初始为空,页面走向为4,3,2,1,4,3,5,4,3,2,1,5,分别使用先进先出,最近最少使用,理想页面置换算法,请问缺页次数是多少?

题目7:TCP具有但UDP不具有的特点不包括:

A、对上层应用而言,收到数据包的顺序与对方发送的顺序一致

B、源IP、目的IP均相同的数据包经过同样的路由路径

C、传输过程中个别数据包丢失,接收端存在检测机制

D、传输数据前必须使用握手方式建立连接

题目8:下面程序的输出结果是什么?

#include <stdio.h>
int func(int n)
{
if(n==0)
return 3;
else
return (func(n-2)+func(n-1));
}
void main()
{
printf("%d\n", func(13));
}


【出处】http://hi.baidu.com/g882/blog/item/5142211f3d1d2efae0fe0b94.html

【四解】

题目1:路由器和交换机的区别

[1]工作层次不同:交换机是工作在OSI/RM开放体系结构的数据链路层(即第二层),而路由器工作在OSI模型的网络层(即第三层)。

[2]转发所依据的对象不同:交换机是利用物理地址或者说MAC地址来确定转发数据的目的地址;而路由器则是利用不同网络的ID号(即IP地址)来确定数据转发的地址。

[3]传统的交换机只能分割冲突域,不能分割广播域;而路由器可以分割广播域。

[4]路由器提供了防火墙的服务,而交换机则没有。

参考:http://network.51cto.com/art/200806/78135.htm

题目2:

选项A:线程有两种传统的控制模式:用户级线程和内核级线程,前者对于操作系统内核是不可见的,而对于后者,内核了解每一个作为可调度实体的线程。所以内核级线程可被操作系统调度。A错!

选项B:正确!

选项C:进程间虽然不能共享内存地址空间,但是把内存数据复制一份,达到共享数据的目的。 C错!

选项D:进程间通过IPC通信,线程间可以直接读写进程数据段(如全局变量)来进行通信。D错!

题目3:各种排序复杂度总结如下:

O(n^2):冒泡排序、选择排序、插入排序

O(nlogn):堆排序、快速排序

O(n):归并排序

本题选择(A)。

题目4:略。

题目5:没有限制“某个工作不能在最后做”的情况下共有 5!=120 种工作方案,去除“某个工作在最后做”的情况 4!=24 种工作方案,最后得到96种工作方案。

题目6:

(1)先进先出:4,3,2,1(miss),4(miss),3(miss),5(miss),4,3,2(miss),1(miss),5。

缺页次数6次。

(2)最近最少使用:4,3,2,1(miss),4(miss),3(miss),5(miss),4,3,2(miss),1(miss),5(miss)。

缺页次数7次。

(3)理想页面置换算法:4,3,2,1,4,3,5,4,3,2,1,5,
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: