加餐六 | 什么才是所谓的编程能力?如何考察一个人的编程能力?
王争 2020-07-01
00:00
讲述:冯永吉大小:6.91M时长:07:32
在招聘要求里,我们经常看到“要求候选人有扎实的编程能力”。在面试反馈中,我们也经常看到面试官用“编程能力很强”来评价候选人。那到底什么是编程能力呢?如何考察一个人的编程能力呢?又如何提高编程能力呢?今天,我们就编程能力这个话题展开聊一聊。
话不多说,让我们正式开始今天的内容吧!
什么是编程能力?
所谓编程能力,指的是把“逻辑”(或者说“功能”“业务”)翻译成代码的能力。所谓编程能力强,指的是,不仅能编写正确的代码,而且编写代码的速度很快,写出来的代码 bug 很少、性能很好、质量很高。
更加具体点讲,一个编程能力强的人,能熟练使用编程语言、开发类库等工具,思路清晰,面对复杂的逻辑,能够编写出 bug free 的代码,能够合理地利用数据结构和算法编写高效的代码,能够灵活地使用设计思想、原则和模式,编写易读、易扩展、易维护、可复用的高质量代码。
相反,一个编程能力差的人,往往逻辑思维能力比较差,面对复杂的逻辑,编写代码的速度很慢,而且容易考虑不周,写出来的代码 bug 很多,更没有性能意识,不懂得如何分析代码的时间复杂度、空间复杂度,更不懂得如何借助现成的数据结构和算法来优化代码性能。除此之外,写代码的时候,几乎不考虑代码的可读性、可扩展性等质量问题,只追求能运行就可以。
如何考察编程能力?
前面我给出了编程能力的简单的定义,从定义中,我们能很清楚地了解,考察一个人的编程能力的几个要素。总结一下,我觉得主要包含这样三个方面:编程语言,数据结构和算法,设计思想、原则和模式。
考察编程能力,那就离不开写代码。所以,在面试中,我一般都会出道编程题,让候选人写段代码看看,也就是所谓的“白板编程”。白板编程在外企面试中比较流行,国内有些候选人不怎么能接受,特别是工作年限比较长的候选人,一听说要写个代码,就觉得这是在“羞辱”他,觉得不应该从这么基础的东西考起。
不过从我多年的面试经验来看,这种拒绝写代码的“大龄码农”,满嘴“架构、高可用、高并发、分布式”,往往代码写得惨不忍睹。所以,只要是应聘一线技术研发岗的候选人,不管是资深工程师、架构师,还是技术 Leader,我都会要求他现场写一段代码。因为这是最直接、最有效检验这个人基本技术素养的途径。
一般来讲,编程语言都可以快速掌握,所以,我一般都不会把它拎出来单独考察,只是顺带着考察一下就可以了。我会重点考察后两个方面:数据结构和算法,设计思想、原则和模式。但是,要想设计一个题目,既能考察到这两方面的知识,又能在不到 1 个小时的面试时间内完成,还是挺难的。所以,对于这两个方面,我一般都分开来考察。我今天重点讲对数据结构和算法的考察,对于设计思想原则和模式的考察,我后面有文章专门来讲。
对于数据结构和算法的考察,我个人不建议面试题目涉及需要记忆的算法,比如被很多人诟病的面试题:写个快排。没有人会天天背诵快排算法,候选人写不出来也理所应当。如果我们换个问法,比如给候选人讲一下快排的思想,然后让候选人用代码实现,测试候选人的代码翻译能力,我觉得这反倒是一个比较好的面试题。除此之外,我也不建议面试题目涉及特殊解题方法或技巧,比如需要用到线段树、并查集等高级数据结构。毕竟大家在工作中不常用到这些知识,不知道或者忘记了我觉得也很正常。
所以,那种不依赖很强的算法背景和特殊的解题技巧的题目,比较适合用来面试。比如像下面这样的面试题就比较好:“写一个函数将 IPv4 地址字符串(仅包含数字、点、空格)转化成 32 位整数。另外,数字和点之间的空格是合法的,其他情况均为非法地址,要求输出合法地址的 32 位整型结果。”
我觉得这种题目是比较公平的,对于没有刷过题的人来说也很友好,因为它纯粹是在考察候选人的基本编程素质:逻辑思维是否清晰,细节是否考虑全面,是否能写出 bug free 的代码,是否有计算机思维,会关注时间空间复杂度等。
如何提高编程能力?
刚刚我们讲了什么是编程能力,如何考察编程能力,现在,我们讲下如何提高编程能力。实际上,我的两个专栏《数据结构与算法之美》《设计模式之美》,就是为了提高你的编程能力而设计的。《数据结构与算法之美》专栏教你如何写出高性能代码,《设计模式之美》专栏教你如何编写易读、易扩展、易维护的高质量代码。
方向很明确,但是要真的提高编程能力,光学不练肯定是不行的。
对于数据结构和算法的练习,我们推荐你多刷 LeetCode 上的题目。刷题不仅仅能加强你对数据结构和算法的掌握,还能锻炼你的逻辑思维能力、写出 bug free 代码的能力、快速实现复杂逻辑的能力,也能锻炼你的性能意识。所以,刷题不只是为了面试,刷题对这些能力的锻炼,都有助于你在平时的业务开发中写出好的代码。
对于设计思想原则和模式的练习,它就不像算法练习那样有现成的题库了。所以,要在平时的开发中进行练习。比如,拿到一个功能需求的时候,我们先去思考一下如何设计,而不是上来就写代码。写代码时,我们时刻思考代码是否遵循了经典的设计思想、设计原则,比如是否足够可扩展、是否满足 SOLID 原则、可读性如何等等。写完代码之后,我们再思考一下,代码是否有进一步优化的空间。做 Code Review 的时候,看到别人的优秀的代码,我们就去思考一下,有哪些值得借鉴的地方。总之,在平时的开发中,我们要刻意地去做这种跟代码质量、代码设计相关的思考训练。时间长了,这种思考就能成为习惯、本能反应,慢慢地,你的代码能力也就不自觉地提高了。这一部分内容你可以回过头去再看下第 100 篇,我们前面有非常详细的讲解。
课堂讨论
实际上,不管从事什么行业,要积累的东西都可以分为“变”与“不变”两类。“不变”的是内功,“变”的是招式。我们要善于发现、持续积累那种“不变”的能力,而不是要去盲目追逐一直都在“变”的招式。除了编程能力之外,在 IT 技术领域,你觉得还有哪些不变的内功?
欢迎留言和我分享你的想法。如果有收获,也欢迎你把这篇文章分享给你的朋友。
30人觉得很赞给文章提建议;)
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
张创琦
Ctrl + Enter 发表
0/2000字
提交留言
精选留言(29)
Jie
沟通能力很重要,和领导和同事和客户可以聊得起来。和机器打交道久了,人不能变得像机器一样。
2020-07-01
__29
J.Smile
做人要务虚,做事要务实。程序员不懂得务虚在职场很吃亏,即便技术不差,但得出类拔萃,挺难。
2020-07-01
_4
_19
程序员jinjunzhu
技术上讲,数据结构,操作系统,网络,设计模式,数据库,这些都是不变的内功
2020-07-06
__11
守拙
“写一个函数将 IPv4 地址字符串(仅包含数字、点、空格)转化成 32 位整数。另外,数字和点之间的空格是合法的,其他情况均为非法地址,要求输出合法地址的 32 位整型结果。”
思路:
\1. 根据.和空格判定输入是否合法;
\2. 将输入以.分割为4个8bit的十进制int值;
\3. 遍历4个int值, 将其转为二进制值;
\4. 拼接4个二进制值, 得到32位整型结果.
2020-07-01
_4
_12
enjoylearning
产品意识也是一种内功,不能只是还原客户的需求,要了解客户需求背后的需求。
2020-07-01
__11
业余爱好者
量子计算普及之前,冯诺依曼架构的理论知识(如os,网络等)还是需要学习的,不过这些底层知识的学习最终还是要落到编程能力上,因为少有人去设计一个操作系统或编译器。
编程能力确实是一个衡量程序员水平的很好的指标,编程能力又可以拆分为算法能力,设计能力等。对程序员来说,一切工作都是围绕编程的,失去这一宗旨,一切都是扯淡,即使所谓沟通之类能力也是为了协作以完成编程目标而已。
2020-07-01
__7
程序员小跃
最近一直在和同学聊如何提升自己的能力,因为他刚转行到 Java 开发不久,我零零碎碎的整理了一些知识点给他;恨自己没早点学习专栏,没有把这篇文章做一个观后感的整理。
还早现在还来得及,我反复读了几遍,顺带分享给我同学,希望他能转行找到心仪的工作。
2020-10-24
__6
Geek7f18eb
private static String ipTo32(String str) {
String result = “”;
int length = str.length();
int i = 0;
for (; ; ) {
if (i > length - 1) {
if (result.length() < 32) {
result += “0”;
continue;
} else {
break;
}
}
char index = str.charAt(i);
Boolean last = Character.isDigit(index);
Boolean lastPoint = “.”.equals(String.valueOf(index));
if (last) {
result += String.valueOf(index);
}
int j = i + 1;
if (j < length && “ “.equals(String.valueOf(str.charAt(j)))) {
Boolean nextPoint = “.”.equals(String.valueOf(str.charAt(j + 1)));
Boolean next = Character.isDigit(str.charAt(j + 1));
if ((last && nextPoint) || (lastPoint && next)) {
} else {
result = “false”;
break;
}
}
i++;
}
return result;
}
2020-07-03
_4
我能走多远
底层得知识是内功,计算机理论基础;内核的内存管理及文件管理;数据结构与算法等等吧
2020-07-01
_3
_4
全炸攻城狮
合法字符串trim后,按照.分割成4个串,再分别转换成二进制,最后合并。这道题关键应该是对于字符串是否合法的判断吧,比较能体现出一个人逻辑思维是否缜密
2020-07-02
__3
辣么大
难啃的大部头算是内功,算程序员提高的门槛。例如操作系统,tcpip
2020-07-02
__1
守拙
计算机领域不变的内功除了数据结构和算法, 设计模式外, 应该加上计算机网络.
我们日常开发都离不开因特网, 应该对其5层结构, 每层职责, 重要协议都有了解.
未来是万物互联的时代, 大到使用http的主机/服务器, 小到使用蓝牙/ZigBee的小型智能设备, 都离不开计算机网络提供的服务.
现在非常火爆的5G, Wifi6, IPv6, 未来会有更多震撼人心, 造福人类的新技术等着我们去探索!
2020-07-01
__1
Jxin
1.置顶向下,点到为止(按需加方法和字段,控制方法和类的作用域,包级结构也是代码的重要组成,它可以描述作用域)
2.还得允许用idea编写程序。如果是java开发甚至可以要求基于spring来写。spring提供了很多编程范式和基建。合理使用实际上就是一种业务代码和技术代码很好解耦的落地。奈何大部分开发用sping就只用一个IOC,那套玩意可是spring 2.X就出了的。现在都5.X了…
3.架构、高可用、高并发、分布式这些确实很重要,论重要性我觉得比算法高。编码能力强不一定能写出好软件。写出好软件也不一定编码能力强。但编码能力强的团队,试错的成本会比较低,相对的,成功的可能会比较高。
2020-07-01
_2
_1
凫雁回塘
stay hungry, stay foolish, 在IT领域尤其重要
2020-07-01
__1
do it
编程能力更像是硬技能。
各种软技能也算是内功(沟通,协调组织等)
2020-07-01
__1
敲代码的老王
public long ipv4ConvertInt32(String ipv4) {
if (StrUtil.isBlank(ipv4)) {
throw new IllegalArgumentException(“ipv4 is empty”);
}
final String[] split = ipv4.split(“.“);
if (split.length != 4) {
throw new IllegalArgumentException(“ipv4 “ + ipv4 + “ is not illegal:is must split 4 by .”);
}
long result = 0;
int kf = split.length - 1;
for (String s : split) {
try {
int item = Integer.parseInt(s);
if (item > 255 || item < 1) {
throw new IllegalArgumentException(“ipv4 “ + ipv4 + “ is not illegal: each item must > 0 and < 256”);
}
result = (long) (result + item * Math.pow(256, kf));
kf -= 1;
} catch (NumberFormatException e) {
throw new IllegalArgumentException(“ipv4 “ + ipv4 + “is not illegal: is must a number”);
}
}
return result;
}
2021-06-29
jimmy
现场写个helloworld
2021-06-12
Geekpillar6699
除了编程能力之外,在 IT 技术领域,我觉得还有“产品思维”、“沟通能力”也是不变的内功。
2021-05-11
_
helloworld
总结
\1. 我感觉作为一个技术人, 技术能力是基础, 有了强大的技术能力后, 才能在其之上谈架构能力, 谈沟通能力, 谈产品能力, 谈领导能力! 技术能力是基石.
2021-05-06
yang
leetcode确实不错,我上周在上面写队列的题,才发现自己太差了,应该好好把基础打牢。甲方待久了就很伤内功,我现在唯一的擅长就是提需求啦。还好,经过几年的磨合,现在乙方终于把我的低代码开发引擎的需求实现了。但是我突然发现我们组织的用户很懒,给他们开了权限也没有看到他们自己实现几个业务应用,我觉得我又超前规划了,挺生气的。
2021-05-05
Geekea09cb
TEST(Ipv4Parser,Ipv4) {
map
{“127.0.0.1”,{0x7f000001, true}},
{“127..0.1”,{0, false}},
{“127.0.1”,{0, false}},
{“ 127 . 0 . 0. 1 “,{0x7f000001, true}},
{“a 127 . 0 . 0. 1 “,{0, false}},
{“0 127 . 0 . 0. 1 “,{0,false}},
{“255.255.255.255”,{0xffffffff, true}},
{“256.255.255.255”,{0,false}},
{“0.0.0.0”,{0, true}},
};
for(auto& kv : m) {
auto res = Parser(kv.first);
ASSERT_EQ(res,kv.second);
}
}
2021-04-29
_
Geekea09cb
class Ipv4Parser {
public:
static std::pairParser(const std::string& ip) { ParserOneSegment(const char str,char end,uint32_t& res) {
uint32_t val[5] = {0};
auto p = ParserOneSegment(ip.c_str(),’.’,val[0]);
if (!p) return {0,false};
p = ParserOneSegment(p + 1,’.’,val[1]);
if (!p) return {0,false};
p = ParserOneSegment(p + 1,’.’,val[2]);
if (!p) return {0,false};
p = ParserOneSegment(p + 1,0,val[3]);
if (!p) return {0,false};
val[4] = val[0];
val[4] = (val[4] << 8) + val[1];
val[4] = (val[4] << 8) + val[2];
val[4] = (val[4] << 8) + val[3];
return {val[4],true};
}
private:
static const char
const char p = strchr(str,end);
if(!p || p == str) {
return nullptr;
}
bool isFirst = true;
bool isFirstZero = false;
res = 0;
char cur = 0;
while (str != p) {
cur = str;
str++;
if (cur == ‘ ‘)
continue;
if (cur < ‘0’ || cur > ‘9’)
return nullptr;
if (isFirst && cur == ‘0’) {
isFirstZero = true;
}
isFirst = false;
res = res * 10 + (cur - ‘0’);
if (res > 255) {
return nullptr;
}
}
if (isFirstZero && res > 0) return nullptr;
return p;
}
};
2021-04-29
_
williamcai
良好的沟通能力,理解需求的能力
2021-04-27
木杉
网络 硬件 硬件接口 软硬结合的
2021-03-25
Geek151295
阅读思考,表达能力
2020-08-27
_
忆水寒
不错哦
2020-08-16
酸辣土豆丝
表达能力,沟通能力
2020-07-29
NYang
网络和计算机组成,另外从软实力上来讲,沟通和自我管理也挺重要的
2020-07-10
maybe
1、技术层面:底层原理,比如数据结构与算法,计算机原理、编译原理等
2、为人处世方面:团队协作,自我学习等等
2020-07-08