面试公司Offer——我的Python求职之路

如何拿到半数面试公司Offer——我的Python求职之路

从八月底开始找工作,短短的一星期多一些,面试了9家公司,拿到5份Offer,可能是因为我所面试的公司都是些创业性的公司吧,不过还是感触良多,因为学习Python的时间还很短,没想到还算比较容易的找到了工作,就把这些天的面试经验和大家分享一下,希望为学习Python找工作的小伙伴们提供些许帮助。

笔者感觉面试最主要的两点:1.项目经验。 2.项目经验和招聘职位相符,这是最主要的,其他的都是锦上添花。

自我介绍

这是一道送分题,万年不变的第一个问题。不过有些小伙伴可能没有太在意,其实这个问题已经在面试官心中决定了你的去留意向。自我介绍的主要结构:个人基本信息 + 基本技术构成 + 项目经验(具体项目以及在项目中的负责部分)+ 自我评价,其中的原则就是紧紧围绕招聘岗位的需求做介绍。在此之前要做好准备工作,看看招聘方具体需要什么方向的研发工程师。目前针对Python,拉勾上的招聘多为自动化测试平台的设计与开发、数据的挖掘与清洗。单纯的web开发好像还没有,所以web方向的同学注意,多和运维以及自动化方面靠拢。

二段式询问

在面试的过程当中,在面试官提出问题的时候,往往会就问题本身引申出较深层次的问题。比如:你使用过with语句吗?我的回答是:with语句经常适用于对资源进行访问的场合,确保在访问的过程中不管是否发生异常都会指执行必要的清理操作,比如文件的自动关闭以及线程中锁的自动获取与释放。面试官紧接着问,那你知道为什么with语句能够使文件正确关闭,一下子把我问闷了,只能依稀记得with语句会开辟出一块独立环境来执行文件的访问,类似沙盒机制。面试官对这个答案不置可否,算是勉强通过了。所以知其然更要知其所以然。在平时的学习中,多问一个为什么,面试的时候就不会太被动。

不要给自己挖坑

确保你在回答面试官的过程中,回答中的每个知识点都了然于胸,不然被问住,是很难堪的。我在回答web安全问题时,顺嘴说了SQL注入,面试官说既然提到了SQL注入,那么你讲讲它的原理及解决方法吧!丢脸的是我竟然把XSS跨站注入攻击和SQL注入搞混了,场面也是有点尴尬。所以斟酌你说的每一句话,聪明点的同学还可以引导面试官,让他问出自己想要被问的问题。

必问到Redis,高并发解决办法

面试了好多家公司,必然问道Redis了解多少,高并发的解决办法。笔者回答的都不是很好。

这一年你学习了什么新的技能

这是面试官在考察你是否对于新鲜技术抱有极大热忱。面试我的面试官无一例外都问到了这个问题。他们都希望能找一个不断学习,开括创新的年轻人。多浏览最新的技术资讯,选择一方面自己感兴趣的领域。

你会选择创业公司还是像BAT那样的大公司,为什么?

当然是看招聘方属于哪一个公司啦,不过问这种问题的一般都是创业公司。答案无非是:挑战大,享受挑战;创业公司具有无限成功的可能性,想随公司一起成长;

为什么你要从上一家公司离职?

这也是一个必问问题,找一个比较正当的理由,不要说什么公司零食太多胖了20斤,公司周别附近的外卖都吃腻了,真的别这样说…主要原则就是不要对前公司抱有怨言,BOSS朝令夕改,PM不靠谱什么的,多寻找自身原因:公司发展比较稳定,但我还年轻,希望有更大的挑战和更多的学习机会。像这样就可以。

描述一下你的上一家公司

这个问题问到的几率不太大,不过也还是有三家公司问到过,招聘方主要想从上一家公司的具体经营规模以及主营业务来定位你的水平,知道招聘方的目的就可以从容应答。

技术性问题

非技术性的问题就是以上这么多,作为参考稍加准备,面试的时候就能对答如流。下面讲一下在面试中的技术性问题。个人感觉技术性的问题面试官问的没有特别多,一般考察2-3个,由浅到深。

  1. 简述函数式编程

    在函数式编程中,函数是基本单位,变量只是一个名称,而不是一个存储单元。除了匿名函数外,Python还使用fliter(),map(),reduce(),apply()函数来支持函数式编程。

  2. 什么是匿名函数,匿名函数有什么局限性

    匿名函数,也就是lambda函数,通常用在函数体比较简单的函数上。匿名函数顾名思义就是函数没有名字,因此不用担心函数名冲突。不过Python对匿名函数的支持有限,只有一些简单的情况下可以使用匿名函数。

  3. 如何捕获异常,常用的异常机制有哪些?

    如果我们没有对异常进行任何预防,那么在程序执行的过程中发生异常,就会中断程序,调用python默认的异常处理器,并在终端输出异常信息。

    try...except...finally语句:当try语句执行时发生异常,回到try语句层,寻找后面是否有except语句。找到except语句后,会调用这个自定义的异常处理器。except将异常处理完毕后,程序继续往下执行。finally语句表示,无论异常发生与否,finally中的语句都要执行。

    assert语句:判断assert后面紧跟的语句是True还是False,如果是True则继续执行print,如果是False则中断程序,调用默认的异常处理器,同时输出assert语句逗号后面的提示信息。

    with语句:如果with语句或语句块中发生异常,会调用默认的异常处理器处理,但文件还是会正常关闭。

  4. copy()与deepcopy()的区别

    copy是浅拷贝,只拷贝可变对象的父级元素。 deepcopy是深拷贝,递归拷贝可变对象的所有元素。

  5. 函数装饰器有什么作用(常考)

    装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。有了装饰器,就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。

  6. 简述Python的作用域以及Python搜索变量的顺序

    Python作用域简单说就是一个变量的命名空间。代码中变量被赋值的位置,就决定了哪些范围的对象可以访问这个变量,这个范围就是变量的作用域。在Python中,只有模块(module),类(class)以及函数(def、lambda)才会引入新的作用域。Python的变量名解析机制也称为 LEGB 法则:本地作用域(Local)→当前作用域被嵌入的本地作用域(Enclosing locals)→全局/模块作用域(Global)→内置作用域(Built-in)

  7. 新式类和旧式类的区别,如何确保使用的类是新式类

    为了统一类(class)和类型(type),python在2.2版本引进来新式类。在2.1版本中,类和类型是不同的。

    为了确保使用的是新式类,有以下方法:

    放在类模块代码的最前面 __metaclass__ = type
    从内建类object直接或者间接地继承
    在python3版本中,默认所有的类都是新式类。

  8. 简述__new__和__init__的区别

    创建一个新实例时调用__new__,初始化一个实例时用__init__,这是它们最本质的区别。

    new方法会返回所构造的对象,init则不会.

    new函数必须以cls作为第一个参数,而init则以self作为其第一个参数.

  9. Python垃圾回收机制(常考)

    Python GC主要使用引用计数(reference counting)来跟踪和回收垃圾。在引用计数的基础上,通过“标记-清除”(mark and sweep)解决容器对象可能产生的循环引用问题,通过“分代回收”(generation collection)以空间换时间的方法提高垃圾回收效率。

    1 引用计数

    PyObject是每个对象必有的内容,其中ob_refcnt就是做为引用计数。当一个对象有新的引用时,它的ob_refcnt就会增加,当引用它的对象被删除,它的ob_refcnt就会减少.引用计数为0时,该对象生命就结束了。

    优点:

    简单 实时性 缺点:

    维护引用计数消耗资源 循环引用

    2 标记-清除机制

    基本思路是先按需分配,等到没有空闲内存的时候从寄存器和程序栈上的引用出发,遍历以对象为节点、以引用为边构成的图,把所有可以访问到的对象打上标记,然后清扫一遍内存空间,把所有没标记的对象释放。

    3 分代技术

    分代回收的整体思想是:将系统中的所有内存块根据其存活时间划分为不同的集合,每个集合就成为一个“代”,垃圾收集频率随着“代”的存活时间的增大而减小,存活时间通常利用经过几次垃圾回收来度量。

    Python默认定义了三代对象集合,索引数越大,对象存活时间越长。

  10. Python中的@property有什么作用?如何实现成员变量的只读属性?

    @property装饰器就是负责把一个方法变成属性调用,通常用在属性的get方法和set方法,通过设置@property可以实现实例成员变量的直接访问,又保留了参数的检查。另外通过设置get方法而不定义set方法可以实现成员变量的只读属性。

  11. *args and **kwargs

    *args代表位置参数,它会接收任意多个参数并把这些参数作为元组传递给函数。**kwargs代表的关键字参数,允许你使用没有事先定义的参数名,另外,位置参数一定要放在关键字参数的前面。

  12. 有用过with statement吗?它的好处是什么?具体如何实现?

    with语句适用于对资源进行访问的场合,确保不管使用过程中是否发生异常都会执行必要的“清理”操作,释放资源,比如文件使用后自动关闭、线程中锁的自动获取和释放等。

  13. what will be the output of the code below? explain your answer

    def extend_list(val, list=[]):
        list.append(val)
        return list
    
    list1 = extend_list(10)
    list2 = extend_list(123, [])
    list3 = extend_list('a')
    
    print(list1) # list1 = [10, 'a']
    print(list2) # list2 = [123, []]
    print(list3) # list3 = [10, 'a']
    
    class Parent(object):
        x = 1
    
    class Child1(Parent):
        pass
    
    class Child2(Parent):
        pass
    
    print(Parent.x, Child1.x, Child2.x)  # [1,1,1]
    Child1.x = 2
    print(Parent.x, Child1.x, Child2.x)  # [1,2,1]
    Partent.x = 3
    print(Parent.x, Child1.x, Child2.x)  # [3,2,3]
  14. 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。

    arr = [[1,4,7,10,15], [2,5,8,12,19], [3,6,9,16,22], [10,13,14,17,24], [18,21,23,26,30]]
    
    def getNum(num, data=None):
        while data:
            if num > data[0][-1]:
                del data[0]
                print(data)
                getNum(num, data=None)
            elif num < data[0][-1]:
                data = list(zip(*data))
                del data[-1]
                data = list(zip(*data))
                print(data)
                getNum(num, data=None)
            else:
                return True
                data.clear()
        return False
    
    
    if __name__ == '__main__':
        print(getNum(18, arr))
  15. 获取最大公约数、最小公倍数

    a = 36
    b = 21
    
    def maxCommon(a, b):
        while b: a,b = b, a%b
        return a
    
    def minCommon(a, b):
        c = a*b
        while b: a,b = b, a%b
        return c//a
    
    if __name__ == '__main__':
        print(maxCommon(a,b))
        print(minCommon(a,b))
  16. 获取中位数

    def median(data):
        data.sort()
        half = len(data) // 2
        return (data[half] + data[~half])/2
    
    l = [1,3,4,53,2,46,8,42,82]
    
    if __name__ == '__main__':
        print(median(l))
  17. 输入一个整数,输出该数二进制表示中1的个数。其中负数用补码表示。

    def getOneCount(num):
        if num > 0:
            count = b_num.count('1')
            print(b_num)
            return count
        elif num < 0:
            b_num = bin(~num)
            count = 8 - b_num.count('1')
            return count
        else:
            return 8
    
    if __name__ == '__main__':
        print(getOneCount(5))
        print(getOneCount(-5))
        print(getOneCount(0))

以上就是我面试过程中所被问到的问题,算法题还是比较少的,也只有2家公司要求写算法,数据结构似乎被问到的不是特别多,就问到了一个B+树的结构。数据库问到的是索引相关的优化。稍微有些基础的都能回答上来,但是最好可以深层次的探讨一下。

本文只做抛砖引玉之用,有些见解还不是特别成熟,希望可以为学习Python找工作的伙伴们提供一些帮助,面试过程当中最重要的一点是放平心态,求职过程是双方的,不需要太过紧张,把自己掌握的知识充分表达出来就好。只要你是匹千里马,迟早会被伯乐牵出来遛一遛的。

转载请注明出处~

分类:  Django开发
标签:  python,  面试
好文要顶  关注我  收藏该文   
Data&Truth
关注 - 0
粉丝 - 113
+加关注
20
2
« 上一篇: Python面试题 —— 计算列表中出现最多次的字符
» 下一篇: MySQL for mac使用记录
posted @  2016-09-03 11:05  Data&Truth 阅读( 7491) 评论( 18)  编辑  收藏

  
#1楼   2016-09-03 12:11  牛腩   
支持支持。。。撸码八年。。换了几份工作。。每次都是做到公司倒闭才走人。。叫我描述上一家公司该怎么说!!!
^_^!!!!!
支持(4) 反对(0)
  
#2楼 [ 楼主2016-09-03 12:13  Data&Truth   
@ 牛腩
我也是做到公司都倒闭了/(ㄒoㄒ)/~~无一幸免
支持(1) 反对(0)
  
#3楼   2016-09-03 15:34  一个非常有趣的故事✨   
这才是干货满满、~
支持(0) 反对(0)
  
#4楼   2016-09-03 16:16  我当道士那儿些年   
:)看来我已经猜到你是谁了,分析的确是很到位,而且长相很重要
支持(0) 反对(0)
  
#5楼 [ 楼主2016-09-03 16:40  Data&Truth   
@ 我当神仙那儿些年
哈哈,我觉得程序员也得打扮的干净得体,程序员不能成为不修边幅的理由吧O(∩_∩)O~顺便问一句,您是?
支持(1) 反对(0)
  
#6楼   2016-09-04 01:58  WrYcF   
太棒了,这些都是鸡血。
支持(0) 反对(0)
  
#7楼   2016-09-04 10:02  Mosson—li   
十月初就要找工作了,也不知道怎么面试,这个博客是及时雨啊 !博主在说一下面试的详细呗,第一次投简历找工作 有点紧张啊
支持(0) 反对(0)
  
#8楼   2016-09-04 10:05  任意球   
自我介绍这里觉得挺难说的,自己不是个好说话的人
支持(1) 反对(0)
  
#9楼   2016-09-04 10:22  __Neo   
@ 任意球
自我介绍其实是最简单的部分,因为是完全可以由你自己控制的。而且如果你的某段经历和求职的职位吻合,你还可以着重讲一下那个方面。按照现在的话说就是带一带面试的节奏。有些面试官会说你先来个x分钟的自我介绍。不要管什么时间限制,面试的核心就是要和面试官聊得来,展示出自己的实力。

面试不是面试官为了考倒你。他是通过面试找到适合和自己一起工作的未来同事,从这点出发来考虑,你就不会虚自我介绍了
支持(3) 反对(0)
  
#10楼   2016-09-04 10:48  任意球   
@ __Neo
引用 @任意球
自我介绍其实是最简单的部分,因为是完全可以由你自己控制的。而且如果你的某段经历和求职的职位吻合,你还可以着重讲一下那个方面。按照现在的话说就是带一带面试的节奏。有些面试官会说你先来个x分钟的自我介绍。不要管什么时间限制,面试的核心就是要和面试官聊得来,展示出自己的实力。

面试不是面试官为了考倒你。他是通过面试找到适合和自己一起工作的未来同事,从这点出发来考虑,你就不会虚自我介绍了
谢谢提点,开始有点方向了
支持(0) 反对(0)
  
#11楼   2016-09-04 11:17  codegay   
楼主没说在哪找的工作。北京之类的吧?
2线城市都很少有招python的了。
支持(0) 反对(0)
  
#12楼 [ 楼主2016-09-04 11:44  Data&Truth   
@ Mosson—li
面试前先把基础知识过一遍,有些东西长时间不用很容易生疏,关于算法题可以去checkio刷一刷。数据结构和网络这一块也需要复习一下。有项目的话,项目最好能上线,没有上线也可以吧github提交上去。

准备工作完成后就可以投简历了,投简历时注意招聘方的要求,如果要求中80%自己都能达到,就可以投简历了,我并不赞成广投简历,每一份简历都需要根据应聘公司做相应的修改。还有注意一点的是观察招聘方要求的方向与你技术是否一致。另外就是选择公司时要想到,这不是你求职的终点,你要让这一家公司成为你下次应聘时简历的一个亮点。

面试的时候衣着得体,提前查好面试地点,计算好时间,不要迟到。面试过程中,知道的就说知道,不知道的也不要瞎猜,面试官和你聊几句就知道你技术薄弱的地方,掩饰也没有用,不过也不用担心,很多面试者都有不会回答的问题,面试官看中的是你解决问题的能力。只要你基础扎实就可以。

最后希望你旗开得胜,找到自己心仪的工作。
支持(0) 反对(0)
  
#13楼   2016-09-05 08:20  euler   
Python
支持(0) 反对(0)
  
#14楼   2016-09-05 09:45  23云恋49枫   
继续继续
支持(0) 反对(0)
  
#15楼   2016-09-22 15:51  小强和他的朋友   
兄弟做python多久了
支持(0) 反对(0)
  
#16楼   2017-01-05 10:23  euler   
Python
支持(0) 反对(0)
  
#17楼   2017-02-26 14:07  hehanlin   
def extend_list(val, list=[]):
list.append(val)
return list

list1 = extend_list(10)
list2 = extend_list(123, [])
list3 = extend_list('a')

print(list1) # list1 = [10, 'a']
print(list2) # list2 = [123]
print(list3) # list3 = [10, 'a']

你好,13题是否您打字打错了?list2,请作者看一下。
支持(0) 反对(0)
  
#18楼   2017-04-17 17:23  张开开   
@ hehanlin
没有错的 你复制运行下就知道了,
我的理解是只传一个参数的时候。list是由系统内存中自动创建的。所以list1 和list3是同一个,传两个参数的时候,会系统会根据第二个参数初始化一个list。应该是一个新的地址。所以只有123。
print id(list1),id(list2),id(list3)
16342392 16340872 16342392

你可能感兴趣的:(面试公司Offer——我的Python求职之路)