从ChatGPT与New Bing看程序员为什么要学习算法?

Posted 黑洞晓威

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从ChatGPT与New Bing看程序员为什么要学习算法?相关的知识,希望对你有一定的参考价值。

文章目录

为什么要学习数据结构和算法?

“算法难道不是脱离实际工作的知识?可能除了面试,这辈子也用不着吧?”

你是不是和我想的一样?

尽管计算机相关专业的同学在大学都学过这门课程,甚至很多培训机构也会培训这方面的知识,但是,很多程序员对数据结构和算法依旧一窍不通。还有一些人也只听说过数组、链表、快排这些最最基本的数据结构和算法,稍微复杂一点的就完全没概念。

当然,也有很多人说,自己实际工作中根本用不到数据结构和算法。所以,就算不懂这块知识,只要Java API、开发框架用得熟练,照样可以把代码写得“飞”起来。事实真的是这样吗?

我们先来听听ChatGPT与NEW Bing怎么回答“程序员为什么要学习数据结构和算法”。

ChatGPT与NEW Bing 的回答

NEW Bing

ChatGPT

想要通关大厂面试,就不能让数据结构和算法拖了后腿

很多大公司,比如BAT、Google、Facebook,面试的时候都喜欢考算法、让人现场写代码。有些人虽然技术不错,但每次去面试都会“跪”在算法上,很是可惜。那你有没有想过,为什么这些大公司都喜欢考算法呢?

校招的时候,参加面试的学生通常没有实际项目经验,公司只能考察他们的基础知识是否牢固。社招就更不用说了,越是厉害的公司,越是注重考察数据结构与算法这类基础知识。相比短期能力,他们更看中你的长期潜力。

你可能要说了,我不懂数据结构与算法,照样找到了好工作啊。那我是不是就不用学数据结构和算法呢?当然不是,你别忘了, 我们学任何知识都是为了“用”的,是为了解决实际工作问题的,学习数据结构和算法自然也不例外。

业务开发工程师,你真的愿意做一辈子CRUD boy吗?

如果你是一名业务开发工程师,你可能要说,我整天就是做数据库CRUD(增删改查),哪里用得到数据结构和算法啊?

是的,对于大部分业务开发来说,我们平时可能更多的是利用已经封装好的现成的接口、类库来堆砌、翻译业务逻辑,很少需要自己实现数据结构和算法。但是, 不需要自己实现,并不代表什么都不需要了解

如果不知道这些类库背后的原理,不懂得时间、空间复杂度分析,你如何能用好、用对它们?存储某个业务数据的时候,你如何知道应该用ArrayList,还是Linked List呢?调用了某个函数之后,你又该如何评估代码的性能和资源的消耗呢?

作为业务开发,我们会用到各种框架、中间件和底层系统,比如Spring、RPC框架、消息中间件、Redis等等。 在这些基础框架中,一般都揉和了很多基础数据结构和算法的设计思想。

比如,我们常用的Key-Value数据库Redis中,里面的有序集合是用什么数据结构来实现的呢?为什么要用跳表来实现呢?为什么不用二叉树呢?

如果你能弄明白这些底层原理,你就能更好地使用它们。即便出现问题,也很容易就能定位。因此, 掌握数据结构和算法,不管对于阅读框架源码,还是理解其背后的设计思想,都是非常有用的。

在平时的工作中,数据结构和算法的应用到处可见。我来举一个你非常熟悉的例子:如何实时地统计业务接口的99%响应时间?

你可能最先想到,每次查询时,从小到大排序所有的响应时间,如果总共有1200个数据,那第1188个数据就是99%的响应时间。很显然,每次用这个方法查询的话都要排序,效率是非常低的。但是,如果你知道“堆”这个数据结构,用两个堆可以非常高效地解决这个问题。

对编程还有追求?不想被行业淘汰?那就不要只会写凑合能用的代码!

何为编程能力强?是代码的可读性好、健壮?还是扩展性好?我觉得没法列,也列不全。但是,在我看来, 性能好坏起码是其中一个非常重要的评判标准。但是,如果你连代码的时间复杂度、空间复杂度都不知道怎么分析,怎么写出高性能的代码呢?

你可能会说,我在小公司工作,用户量很少,需要处理的数据量也很少,开发中不需要考虑那么多性能的问题,完成功能就可以,用什么数据结构和算法,差别根本不大。但是你真的想“十年如一日”地做一样的工作吗?

经常有人说,程序员35岁之后很容易陷入瓶颈,被行业淘汰,我觉得原因其实就在此。有的人写代码的时候,从来都不考虑非功能性的需求,只是完成功能,凑合能用就好;做事情的时候,也从来没有长远规划,只把眼前事情做好就满足了。

可是如果你在一家成熟的公司,面对的是千万级甚至亿级的用户,开发的是TB、PB级别数据的处理系统。性能几乎是开发过程中时刻都要考虑的问题。一个简单的ArrayList、Linked List的选择问题,就可能会产生成千上万倍的性能差别。这个时候,数据结构和算法的意义就完全凸显出来了。

其实,我觉得,数据结构和算法这个东西,如果你不去学,可能真的这辈子都用不到,也感受不到它的好。但是一旦掌握,之前你可能需要费很大劲儿来优化的代码,需要花很多心思来设计的架构,用了数据结构和算法之后,很容易就可以解决了。

学习计划

还记得你刚学算法时手写出冒泡排序时的喜悦,以及被其精妙的想法折服时的心情吗?

public class BubbleSort 
    public static void main(String[] args) 
        int[] array = 5, 3, 8, 4, 2;
        bubbleSort(array);
        for (int i : array) 
            System.out.print(i + " ");
        
    
    
    public static void bubbleSort(int[] array) 
        int n = array.length;
        for (int i = 0; i < n - 1; i++) 
            for (int j = 0; j < n - i - 1; j++) 
                if (array[j] > array[j + 1]) 
                    int temp = array[j];
                    array[j] = array[j + 1];
                    array[j + 1] = temp;
                
            
        
    

让我们重拾这片心情,继续朝着算法这条路出发吧!

出于对算法的探索以及为了备战四月的蓝桥杯,我报名了这次新星计划中的算法赛道,在接下来的两周里我会持续分享我学习算法的过程与总结。学习任何东西都需要有一个计划,而且这个计划要根据自己的实际情况来制定。比如:

  • 我们应该先从哪些基础知识开始入门?
  • 我们应该按照什么样的顺序和难度来逐渐深入?
  • 我们应该花多少时间来练习编码和调试?
  • 我们应该如何评估自己的进度和效果?

就让我带着大家一起领略算法的神秘与五彩缤纷的世家吧!

谁能真正替代你?AI辅助编码工具深度对比(chatGPT/Copilot/Cursor/New Bing)

写在开头

这几个月AI相关新闻的火爆程度大家都已经看见了,作为一个被裹挟在AI时代浪潮中的程序员,在这几个月里我也是异常兴奋和焦虑。甚至都兴奋的不想拖更了。不仅仅兴奋于AI对于我们生产力的全面提升,也焦虑于Copilot等AI辅助编码工具,会将程序员这个工种和我们所熟悉的传统软件开发流程彻底颠覆,用计算机的极高效率碾压人类的低效率。

当然这些也都是后话了,我们目前能做的,就是保持敏锐的嗅觉,尝试去迎接AI时代的来临。做“未来世界的幸存者”。

本文是我深度体验了Github Copilot, ChatGPT等产品后,对于这些AI辅助开发工具的一次横向评测。写本文的初衷是帮助大家快速筛选出一款合适你的AI辅助工具。相信我,请不要再怀疑这些工具是否能给你带来效率提升。当你尝试使用后,很快就会习惯它们,甚至是离不开它们。

本文评测的工具有:

  • Github Copilot
  • ChatGPT(GPT-3.5)
  • New Bing
  • Cursor.so

这些工具可以结合起来使用,提升你的开发效率。所以这些工具之间并不是互斥关系。文章的最后会给出总结以及我的一些想法。

Github Copilot

Github Copilot是由Github和OpenAI合作推出的一个人工智能代码辅助工具,采用了OpenAI的GPT技术,能够为开发人员提供实时的代码提示和生成功能,类似于一个AI助手,帮助开发人员更快速、更方便地编写代码。

当前的Github Copilot基于GPT-3模型,它可以分析上下文并根据已有的代码和注释来推断出应该写什么代码。通过使用Github Copilot,开发人员可以减少手动输入代码的时间,提高代码的质量和效率。它支持多种编程语言,如Python、JavaScript、TypeScript、Ruby等,并可以与主流的集成开发环境(IDE)和文本编辑器配合使用。

使用体验

我已经深度使用了Copilot接近一个月,但每当我和同事朋友们聊到Copilot的使用体验,以及它在哪方面能够提高我的效率时,我仍很难用语言去描述,我只能粗略的总结为下面几个结论:

  • 编写你熟悉的语言时,他仅能帮助你减少一些重复模板代码的编写。编写你不熟悉的语言时,他能够准确推断你的意图,直接生成代码,免除了查询如何使用API的耗时工作
  • 它顺着你的心流生成片段代码,但很难从0开始为你创造整段代码,即使它是可以生成整段代码的,但是也常常是需要你手工修改的。

接下来看一下我常用的几种使用方式。

1. 根据上下文生成代码

它可以根据函数名,类名,注释,来推断你想写的代码,帮你填充。

2. 根据代码生成注释

反过来,它可以尝试理解你的代码,为你生成注释,你只需要给他一个 // 前缀

3. 帮你起变量名

它可以帮你器变量名,这可是很多英语不好的程序员的大福音。毕竟,编码的30%时间,是在想变量名。

4. 和你聊天,当然,是聊代码!

你没有听错,copilot可以在你的代码里聊天,但是显然不能和他唠家常。它并不是chatGPT,无法给你常识回答(应该是被故意限制了交流范围),只会和你扯皮。

正确的使用方式是让它和你讨论你写代码,他会总结你的上下文代码,并且给你一个它认为“合理”的解释。

以上就是我常用的几种方式,我查阅了很多资料,基本上面涵盖了大部分操作方式。当然,可能还有我没挖掘到的使用方式。

编码能力

说完使用体验,我想要引出我本次横向评测的一个评测标准,就是通过相近的试题,看下这几个工具的编码能力有何区别,给大家直观地对比。我们就先从Copilot开始。

独立编写:单例模式

我给它们设定的题目是独立写一个单例模式,这个题目是国内Java开发者“常考题”,里面除了基本的代码,还有很多细节需要注意,也可以很好的用来向AI们提问,看看它们是否真正地理解它们写的内容。

让我们欢迎第一位选手,Github Copilot。

上面的编码动画也是我认为最符合我日常使用copilot的案例,从创建类文件后,copilot帮我自动生成了私有变量,私有构造方法,以及获取单例的公有方法。在写公有方法的期间,它最开始的代码没有考虑双重检查锁,我给了它一些提示,它补全了剩余的代码。

需要注意的是,由于单例模式在网上有太多的学习资料,Copilot肯定也吸取了大量优秀代码,才能写的如此高效。并不代表所有复杂的代码题都能够帮你自动补全,并保证正确性。所以大家谨慎看待它的独立编码能力,也不要过分乐观。

补全项目现有代码

第二个测试,我们让它补全项目的现有代码,我拿一个我自己写的代码举例,其中有一个DiffDTO实体类,有一个addDiffDTO方法一行都没写,需要补全。

可以看到它自动补全了代码,并且会用到上面已有的方法和变量。这也是它的强项,根据上下文信息推断代码该怎么补全,写出来的代码质量很高,不能说每次都可直接使用,但80%的情况下你是无需做大幅度修改的。

不足之处

Github Copilot有不足之处吗?我想了想,有以下几点:

  • 所有代码上下文通过https传到云端(微软的服务器),尽管我通读了微软的隐私保护政策,它承诺不会有任何代码被泄露。但放在国内网络安全大环境下,这样的处理数据方式,在国内公司内使用很可能会受到影响,需要用户注意。
  • 对于复杂的代码逻辑,比如连你都一下子看不懂的屎山代码,它的理解能力同样欠缺。它擅长的是网络上有很多公开资料的代码段落和实现方式。

ChatGPT

让我们来看下一位选手,最近火出圈的chatGPT。

chatGPT是一款基于GPT模型的聊天机器人,它可以像人类一样进行自然语言对话。它不仅可以回答用户的各种问题,还可以学习和理解用户的需求,提供更恰当的回答和建议。

得益于其基于GPT模型的强大语言处理能力。GPT模型可以理解代码中的自然语言描述部分。其次,chatGPT在训练阶段使用了大量的代码相关文本语料,包括代码库、API文档、程序员交流论坛等,因此具有一定的代码理解和识别能力。此外,chatGPT还可以通过不断的学习和自我优化,逐渐提高其对于代码相关问题的处理能力和准确度。

我文中使用的chatGPT是GPT-3.5版本。我们直接进入正题,看看它的编码能力。由于是对话方式,所以我们需要向它描述我们的需求。

编码能力

独立编写:单例模式

我:

请用Java写一个单例模式

ChatGPT:

public class Singleton 
​
 private static Singleton instance;
​
 private Singleton() 
 // 私有构造函数
 
​
 public static synchronized Singleton getInstance() 
     // 双重检查锁定
     if (instance == null) 
        synchronized (Singleton.class) 
           if (instance == null) 
               instance = new Singleton();
           
       
     return instance;
      
   

​
// 使用方法
Singleton s = Singleton.getInstance();

写的不错,很有面试题刷多了的开发人员的风格,写了经常会被问到的双重校验锁,但是没有用volatile,让我们继续发问。

我:

为何要用双重检查锁定?

ChatGPT:

双重检查锁定(double-checked locking)是一种常见的单例模式的实现方式。它的目的是在不使用同步锁的情况下实现单例模式的线程安全,避免了每次获取对象实例时都进行同步锁的开销。但是,这种实现方式也有可能存在线程安全问题。

具体来说,双重检查锁定是在 getInstance 方法中加入了双重检查,即先判断实例是否已经存在,如果不存在再进行同步锁的操作。这样可以减少同步锁的开销,提高了效率。

但是,在多线程环境下,双重检查锁定也有可能出现线程安全问题,原因是在某些情况下,多个线程可能同时通过第一重检查,然后同时进行同步锁的操作,导致出现多个实例。

为了避免这种情况,需要将 getInstance 方法设置为 synchronized 关键字,保证在同一时刻只有一个线程可以访问该方法,从而避免多个实例的出现。但是,这样会带来一定的性能开销,因为每次访问 getInstance 方法都需要进行同步锁操作。

讲得不错,但很难不让人觉得是看多了面试文章。我们顺着volatile的问题继续问问他为啥不写volatile。

我:

上面的代码依然有漏洞,可以优化下吗

ChatGPT:

是的,即使使用了双重检查锁定,上面的代码仍然有漏洞,也就是可能出现线程安全问题。为了解决这个问题,可以使用 Java5 中引入的 volatile 关键字,通过禁止指令重排来保证线程安全。

优化后的代码如下:

public class Singleton 
private static volatile Singleton instance;
​
private Singleton() 
// 私有构造方法

​
public static Singleton getInstance() 
    if (instance == null) 
        synchronized (Singleton.class) 
            if (instance == null) 
            instance = new Singleton();
            
        
    return instance;
    


​

上面的代码中,我们将 instance 字段添加了 volatile 关键字,这样可以保证在多线程环境中,instance 变量的读取和赋值操作是原子性的,即保证了线程的可见性和有序性,从而避免了出现多个实例的情况。

注意,这里的 volatile 关键字只是作用于 instance 字段本身,而不是整个 getInstance 方法。这样做可以减小同步锁的范围,提高了效率。

它自己想到并解决了volatile的细节问题,我们再结合他写的代码,问问一些面试官会问的问题。

我:

上面的代码中,同步的是一个Singleton.class,可以用int变量代替吗

ChatGPT:

不可以,使用 int 变量代替 Singleton.class 进行同步是不可行的。

因为在 Java 中,每个类都有一个唯一的 Class 对象,多个线程使用同一个类的 Class 对象进行同步,可以保证线程安全。而使用 int 变量进行同步,由于 int 是基本数据类型,它在内存中的每一份拷贝都可能不同,因此不能保证多个线程都使用同一个 int 变量进行同步,从而无法保证线程安全。

此外,使用 Class 对象进行同步还有一个好处,即可以保证实例变量的可见性,因为同步代码块中的读写操作都是在同一条线程中执行的。

因此,使用 Singleton.class 对象进行同步是保证线程安全的最佳方式,不建议使用其他方式进行同步。

回答的不错,理解了我提示语中的 Singleton.class 是指的它自己代码中的类,最后问一个volatile的细节。

我:

上面的代码中,如果不使用volatile,有可能会出现重排序问题,能具体展开讲讲吗

ChatGPT:

是的,如果没有使用volatile关键字,就有可能出现重排序问题。

在Java中,编译器和处理器都会对指令进行重排序,这是为了优化程序性能。例如,在一个if语句中,为了避免每次都进行判断,编译器可能会将判断条件提前到if语句外面。在单线程环境下,这种重排序是没有问题的,因为程序执行的结果是一样的。但是,在多线程环境下,重排序可能会导致线程安全问题。

考虑下面的代码:

public class Singleton 
private static Singleton instance;
​
private Singleton() 
​
public static Singleton getInstance() 
    if (instance == null) 
        synchronized (Singleton.class) 
            if (instance == null) 
            instance = new Singleton();
            
        
    return instance;
    


在这个代码中,如果没有使用volatile关键字,就有可能出现下面的重排序:

  1. 线程A调用getInstance()方法,发现instance为null,进入同步块。
  2. 线程B调用getInstance()方法,由于instance还没有被初始化,也发现instance为null,也进入同步块。
  3. 线程A执行instance = new Singleton(),在执行该语句时,可能会被重排序为下面这样:
memory = allocate(); //1:分配对象的内存空间
ctorInstance(memory); //2:初始化对象
instance = memory; //3:将对象赋值给instance变量

这个重排序是允许的,因为在单线程环境下,这个顺序和原来的顺序没有区别。但是,在多线程环境下,这个重排序可能会导致线程安全问题,因为线程B此时检查instance不为null,就会直接返回instance,这个instance还没有被初始化,就会导致错误。

为了避免这种情况,可以使用volatile关键字来确保instance变量的可见性,从而避免重排序问题。

上面的对话,是一个典型的面试题,网上有很多文章也会顺着这个思路写下来,所以chatGPT回答的得心应手,并不代表在所有代码优化方面都能回答的这么细节和流畅。但不得不说,如果你第一次用chatGPT,你还是会被它的能力和思路所震惊。

接下来我们让chatGPT解决一个kotlin中并发接口的设计,看看它对于不那么常见的复杂问题的解决能力

独立编写:Kotlin+Reactor设计并发接口

这是我在实际开发过程中遇到的一个比较难写的代码例子,在kotlin中使用reactor框架来写一个批量查询接口,由于性能要求,该接口中的每个查询必须并发执行(线程池执行)。我把这个遇到的实际问题作为测试的case,让我们来问问chatGPT。

我:

Kotlin中,如何用reactor框架写一个并发处理的方法,要求入参是List,方法中将字符串后面拼接同样的字符串,输出List

ChatGPT:

您可以使用Reactor框架中的Flux来实现这个要求。下面是一个示例代码:

import reactor.core.publisher.Flux
import reactor.core.scheduler.Schedulers
​
fun processStrings(strings: List<String>): List<String> 
      val flux = Flux.fromIterable(strings)
      .flatMap  str -> Mono.fromCallable  str+str 
      .subscribeOn(Schedulers.parallel())
  
  return flux.collectList().block() ?: emptyList()

​

在这个例子中,我们使用Flux.fromIterable方法创建一个Flux,并使用flatMap方法处理每个字符串。在flatMap中,我们使用Mono.fromCallable方法来创建一个异步执行的函数,该函数将每个字符串转为大写。同时我们使用Schedulers.parallel()指定执行该函数的调度器,以实现并发执行的效果。

最后,我们使用collectList()方法等待所有异步操作完成并将结果作为List<String>返回。

需要注意的是,我们使用了并发执行,因此结果顺序可能与输入顺序不同。如果您需要保持结果顺序与输入顺序相同,可以使用concatMap方法代替flatMap

在它给出的代码上,我加入了一个随机休眠的函数来验证是否代码真的执行了异步,并且给了它入参 [0-9],然后执行,事实证明chatGPT给出的代码直接可以跑通,并且真正是并行执行的。给出结果:

Current ThreadName: parallel-1  sleep time: 188
Current ThreadName: parallel-2  sleep time: 483
Current ThreadName: parallel-3  sleep time: 785
Current ThreadName: parallel-4  sleep time: 524
Current ThreadName: parallel-5  sleep time: 175
Current ThreadName: parallel-6  sleep time: 894
Current ThreadName: parallel-7  sleep time: 440
Current ThreadName: parallel-8  sleep time: 158
Current ThreadName: parallel-9  sleep time: 227
Current ThreadName: parallel-10  sleep time: 300
Result:[99, 33, 66, 1010, 22, 11, 44, 77, 88, 55]

但是它的代码中使用了flatMap并没有保证返回结果List中的顺序和入参顺序相同,对于我的需求来说这是不可接受的,chatGPT在刚才的回答中考虑到了这个问题,告诉了我解决方案,使用concatMap代替,于是我修改为concatMap,结果如下:

Current ThreadName: parallel-1  sleep time: 933
Current ThreadName: parallel-2  sleep time: 715
Current ThreadName: parallel-3  sleep time: 278
Current ThreadName: parallel-4  sleep time: 182
Current ThreadName: parallel-5  sleep time: 14
Current ThreadName: parallel-6  sleep time: 313
Current ThreadName: parallel-7  sleep time: 228
Current ThreadName: parallel-8  sleep time: 650
Current ThreadName: parallel-9  sleep time: 554
Current ThreadName: parallel-10  sleep time: 334
Result:[11, 22, 33, 44, 55, 66, 77, 88, 99, 1010]

看上去一切ok,但其实问题很大。使用了concatMap后,实际上使用concatMap的这段代码并没有真正的并发执行,而是类似串行的方式,得到了结果。(本文不具体展开解释为何使用了concatMap后会有如此效果)

既然答案和我想要的有偏差,那就继续追问chatGPT。

我:

这段代码如何在保证返回值顺序和入参顺序相同的情况下,真正的并发执行?

ChatGPT:

可以使用flatMapSequential操作符替换concatMap操作符来实现真正的并发执行,并且保证返回值顺序和入参顺序相同。

修改后的代码如下:

fun processStrings(strings: List<String>): List<String> 
     val flux = Flux.fromIterable(strings)
         .flatMapSequential  str ->
             Mono.fromCallable 
                 str+str
             .subscribeOn(Schedulers.parallel())
         
     return flux.collectList().block() ?: emptyList()

Current ThreadName: parallel-1  sleep time: 755
Current ThreadName: parallel-2  sleep time: 342
Current ThreadName: parallel-3  sleep time: 340
Current ThreadName: parallel-4  sleep time: 562
Current ThreadName: parallel-5  sleep time: 405
Current ThreadName: parallel-6  sleep time: 399
Current ThreadName: parallel-7  sleep time: 250
Current ThreadName: parallel-8  sleep time: 351
Current ThreadName: parallel-9  sleep time: 169
Current ThreadName: parallel-10  sleep time: 118
Result:[33, 22, 11, 44, 55, 66, 77, 88, 99, 1010]

chatGPT回答基本正确了。

不过,有时候提问同样的问题,chatGPT会给出很多种错误答案,有的是编译错误,有的是串行执行不符合我的要求,以至于我不得不重新提问了好几次。这个过程中,我没有好好做截图。但我发现它其实就是在给出一些网络上常见的博客写的解法,很多时候都是错误的,或者说其实并不对应我对它的诉求。

不足之处

chatGPT针对代码方面的回答做了很多特殊的调优,所以它能够作为一个帮助程序员编码的工具。很多时候,我都更愿意问chatGPT而不是自己去谷歌答案。这已经证明了我足够信任它。如果要说有什么不足之处,我想到以下几点:

  • 面对不常见的复杂代码设计题,没有做到优秀的水平,但相信未来不断地训练后,会变得更加完善。
  • 它不能全程辅助你编码,这一点比不上Copilot。
  • 它不能阅读你整个项目的代码,无法和Copilot一样有强大的上下文能力。当然你可以手动给他很多上下文代码,但是相比Copilot肯定还是差距很大。毕竟Copilot可能阅读了你整个项目后给出一些建议。

New Bing

引用New Bing官网的介绍,New Bing 就像您在搜索⽹络时身边有⼀个研究助理、个⼈规划师和创意伙伴。您可以问你的实际问题,当你提出复杂的问题时,Bing 会给你详细的答复。 得到⼀个实际的答案。 Bing 会查看⽹络上的搜索结果,为您提供⼀个总结性的答案。 要有创意。当你需要灵感时,必应可以帮你写诗、写故事,甚⾄为你创造⼀个全新的形象。

总的来说,你可以理解为New Bing是一个chatGPT + Bing搜索引擎内网络信息 的加强版对话机器人。

使用体验

New Bing目前已经和谐了国内的IP,用国内IP访问任何new Bing相关的网页会强制跳转的Bing搜索首页。所以需要打开科学工具后使用,本文不具体展开。网上有很多攻略可查。

编码能力

我们仍然使用刚才的测试例子(单测+Kotlin并发接口)来测试NewBing的代码编写能力。Bing有三种对话模式可选,需要选到“更多 精确”,这种模式下,它会认认真真地给我们写代码。

独立编写:单例模式

我们直接提问,请看截图。

给出的代码比较基础,让我们追问下,让他修改。

它修改的很快,现在代码基本已经成型了。让我们和之前测试chatGPT提问一模一样的问题。

可以看到,页面下方,他还会给一些符合你问题上下文的推荐提示。

独立编写:Kotlin+Reactor设计并发接口

让我们来欣赏一下NewBing面对复杂问题的编码能力。

将它写的代码放入Demo,编译不通过。

继续质问它

最终它完成了代码,但是写法在我看来有些奇怪。整体来看,和chatGPT的使用体验是相似的。

不足之处

NewBing和chatGPT相比,给我感觉不分伯仲,面对不常见的复杂代码设计题,依然有一些力不从心。并且,不知道是不是使用了搜索引擎内的原因,比chatGPT更容易出现错误的结果。

相比Github Copilot,则和chatGPT一样,由于没有足够的代码上下文,对你的帮助远没有Copilot那么好用。

Cursor.so

Cursor.so 是 OpenAI 最近推出的一款IDE,它可以帮助你提供想法并编写代码。最值得一提的是,它是一款免费软件,OpenAI 还承诺将持续更新和改进,为用户带来更多新的功能和体验。

为什么它会受到关注,主要是因为它能够免费使用内置了类似Github Copilot的插件,毕竟Copilot是付费软件,很多小伙伴还没法免费体验到。

编码能力

独立编写:单例模式

cursor主要有两个功能,一个可以自动生成代码edit,一个是根据当前代码进行聊天chat。各自有独立的快捷键来启动。

我们让他写一个Java中的单例模式类,快捷键command+K,输入中文 ”写一个线程安全的单例模式“,它给出了如下答案。

public class Singleton 
    private static volatile Singleton instance;
    private Singleton() 
    public static Singleton getInstance() 
        if (instance == null) 
            synchronized (Singleton.class) 
                if (instance == null) 
                    instance = new Singleton();
                
            
        
        return instance;
    

写的不错,带上了双重校验和volatile,很像是熟读面试代码的面试者。既然本段代码没什么细节问题,那就试用下聊天功能command+L

解释的和chatGPT以及NewBing都类似,并且还支持聊天历史记录展示。

补全项目现有代码

让他补全之前Copilot补全过的同样测试代码,写入threadLocal变量的代码,可以完成。

当我准备更进一步问问他对于这个代码怎么看的时候,它服务开始了长时间的不稳定。

不足之处

服务不稳定

不支持插件

没有插件市场,甚至没有集成版本控制,比如Git,在UI中没法查看改动的代码,所以几乎已经告别了开发大型项目了。只要稍微做一些代码改动,你就会忘记了代码的改动是哪里。

基础功能缺失

比如打开一个项目文件夹后居然不支持关闭,文件类型没有高亮区分,整体用下来感觉Cursor.so还处于一个很早期的阶段。

总结

最后,总结下这几个产品作为一个AI辅助编码工具的优缺点。

Github Copilot:

优点:

  • 可以根据提示自动生成代码,提高开发效率。
  • 可以学习你项目中的代码风格,获取足够多的上下文,并根据其生成代码。
  • 支持多种编程语言,适用范围广。

缺点:

  • 可能会存在隐私问题

chatGPT和New Bing:

优点:

  • 随时随地可用,不依赖代码项目,是你查询谷歌时的完美替代品。

缺点:

  • 它不能全程辅助你编码,这一点比不上Copilot,并且无法和Copilot一样有强大的上下文能力。
  • 对于复杂的代码逻辑,理解能力未必能让你满意。

Cursor.so

优点:

  • 免费的同时能够体验AI辅助编程,就是最大的优点

缺点:

  • 基础功能缺失,不能称之为一个可靠的IDE
  • 服务不稳定

一句话总结,如果你希望将这几个产品用于辅助你编程,提高编码效率,使用Github Copilot结合chatGPT是一个可行的方式。你可以在编写代码时使用Github Copilot,遇到问题时再求助chatGPT。如果暂时不想为Github Copilot付费,可以只使用chatGPT。不过,由于Cursor.so的使用体验不够好,且不易替代Github Copilot,建议还是等待Cursor.so之后的版本。

以上是关于从ChatGPT与New Bing看程序员为什么要学习算法?的主要内容,如果未能解决你的问题,请参考以下文章

新必应(New Bing)申请与使用教程

讯飞星火大模型 与New Bing实测对比

谁能真正替代你?AI辅助编码工具深度对比(chatGPT/Copilot/Cursor/New Bing)

New Bing怼人说谎PUA,ChatGPT已经开始胡言乱语了

我谈一下New Bing

钓鱼实测写bug给new bing和chatGPT查。问他们林黛玉倒拔垂杨柳