不学无术的下场——OO第一单元总结

时间:2021-11-16 09:25:53

第一单元OO作业总结

第一次作业

​ 第一次作业的要求是对仅有常数和幂函数的式子进行求导。

​ 由于是第一次接触JAVA,对很多东西都还不熟悉,尤其是正则表达式做得不好。当时我的思路如下:

  • 建立Poly类,存储若干个幂函数(但是没有把幂函数建立成一个对象,对后面产生了麻烦)
  • 先对字符串进行预处理(好坏参半)
  • 使用符号作为关键字对原字符串进行分割(好坏参半,原因见后文分析)
  • 对特殊符号采用了保护符(有效,但是不够巧妙)
  • 进行简单判断之后,把大部分异常交给BigInteger处理(好)
  • 在Poly类中,使用Hashmap来进行合并同类项

程序思路分析&更优的思路分析

​ 这一次作业中,我有许多做得不够好的地方。可以说很多问题是不会正则表达式所导致的

对项的分割:

​ (较优的解决方案):

​ 写一个通项的正则表达式,(注意第一项和后续项的区别)。然后疯狂match,如果发现有连续匹配到的两项无法连接,或者出现了字符串前后未匹配,则抛出异常。

​ (当时我用的比较傻的方案):

​ 使用+-号作为字符串分割的关键字,调用String自带函数完成分割。但是同时产生了问题,第一个问题是指数中的符号如何处理。我的解决方案是在预处理时,将这样的符号替换成特殊字符A和S,也就是之前说的符号保护

​ 对于连续的加号减号问题,也是用字符串替换来进行化减。但是直接把形如“+ +”给换成“+”会出现问题,因为“+ +【空格】1“不合法,而”+【空格】1“合法,所以直接替换之后可能出现问题。所以再次需要使用到符号保护问题。

对格式错误的处理

​ 完成了字符串的分割之后,基于实现方法的不同,会有两种不同的情况:

​ (使用正则去判断):

​ 到这一步基本不会存在非法情况了,因为能匹配进来的字符串都是合法公民(笑)

​ (使用符号分割):

​ 到这一步之后,我选择的是做一些基础判断,比如是否为常数。是否存在系数以及指数等,如果异常则抛出相应异常。

​ 接着,把字符串再进行分割,分隔成系数,指数部分,然后把得到的结果送给BigInteger处理,其他的格式异常全部由BigInteger构造函数内部进行判断。感觉这样的做法还是挺不错的。只要保证自己分隔方法正确,任何的格式错误均会被BigInteger给捕捉。

项的存储以及求导

​ 为了方便同类项的合并,我采取了Hashmap来进行数据的存取。采用的是<BigInteger,BigInteger>的形式。然后求导的结果是返回一个String,直接输出即可。

​ 但是这样的实现方式并不好。

更优的解决方案:

​ 将项建立为一个单独的类,实现ToString方法,实现求导方法。项的求导返回的还是一个项。多项式求导返回一个新的多项式。多项式实现ToString方法。

​ 这样子,可以在读入数据的时候,调用ToString判断数据是否正确地读入。求导之后得到的结果不会影响多项式本身。再用一次ToString即可观察求导的结果。

被发现的bug&&找bug的思路

​ 我的问题发生在做符号保护这个地方。既然符号保护,就必须要使用一些特殊的字符。而我忘记判断了原始字符串里面有无出现这些特殊字符。所以导致了被人用我的保护字符进行hack的情况。

​ 还有就是对于正则表达式不熟悉,造成[-+]写成了[-|+],然后被hack。。。。

找bug的思路:

​ 第一点是对于特殊空白字符的hack。很多同学使用了\\s来进行判断空白字符。于是。。。。正义的\v出场,hack了6个人

​ 第二个点,是出现在多个符号的情况下。形如- - x是合法的,但是有的同学就没判断得出来,认为这个格式错误。hack了2个人

​ 第三个点是,单独一个符号的问题。我的测试数据是“+”,有一个人啥都没输出,有一个人输出了0,共hack2人

​ 第四个点是,当串内为空的时候,是否补0,比如“x-x”。有的同学在这个情况下就输出空串了。。。。hack2人

利用工具对自己代码进行分析

不学无术的下场——OO第一单元总结

不学无术的下场——OO第一单元总结

可以看出,主要的问题存在于不会用正则表达式,造成了字符串预处理、获取系数、属于指数这三个部分特别复杂。我的bug也是在这地方产生的。所以,学好正则,减少函数复杂度很有必要。

第二次作业

程序思路分析&更优的思路分析

这一次的作业,要求我们求导的式子变得更加复杂了一点。主要是新增了sin、cos,以及由于乘法所带来的运算符素质三连问题。

对于字符串的分隔

​ 采用双重正则表达式判断:

​ 由于第一次不会正则,吃了大亏。所以这一次作业恶补了一下正则表达式。我采用正则表达式将原字符串分隔成若干个乘法项。然后在构造函数内再对乘法项进行分隔,处理。

​ 这一部分判断格式错误的逻辑是:出现前后未匹配的串,或者是前后匹配项未相连,则扔出格式错误异常

注意理解三连符号出现的原因,还有首项和后续项的区别。

对于程序结构设计

​ (我之前采用的方法:)

  • 最上层的是Poly,也就是加法式。使用ArrayList来存储若干个乘法式。

  • 每个乘法式内有三个Factor,分别是X,Sin,Cos型,还有一个BigInteger型的系数。(原本想的结构是支持扩展的,但为了优化,这里的代码特化了。)

  • 每个Factor里面有一个Type,系数,指数。

求导的时候:

  • Factor的求导函数返回Factor的Arraylist,因为型如sin(x)^2^这样的Factor,求导之后的结果会是2*sin(x)*cos(x)

,也就是可能有若干项。所以需要返回的是Arraylist

  • Term的求导是基于乘法求导法则的,即一项不求导,乘上其他项全部求导。返回的是一个Term的Arraylist
  • Poly的求导是对所有的Term分别求导,然后加入到Arraylist中。
  • 对三角函数的优化集成在AddTerm这个函数里面了。

一些感觉还不错的细节:

  • 任何级别的求导,返回的均是一个相应的对象,而不是一个String。这样可以方便求导后的化简。
  • 在添加Term或者其他东西的时候。我是用了自己写的一个函数,AddTerm。刚开始写的时候,这个函数里面就是一个对Arraylist的添加。但是在后面优化的时候,比如合并,化简。只需要对这个函数修改即可完成。

被发现的bug&&找bug的思路

被发现的bug,其实是我优化的时候犯的糊涂。我在一个函数中试图把sin^2^与cos^2^合并,但是随后紧接着又试图通过拆分另一项来与当前项进行合并。于是出现了优化--复原--优化--复原的无穷递归。

找bug的时候,我们那一组的成员都比较的坚不可摧。我是采用写一个比较复杂的式子,然后随机加入空格,或者符号来进行修改。

然后在本地,先将下载下来的代码编译。分别放入文件夹中,再编写cmd脚本来进行本地的测试。

然后就基本上是加特林突突突的过程,本地运行发现有某个代码比较异常,那么就揪出来看看。

利用工具对自己的代码进行分析

不学无术的下场——OO第一单元总结

不学无术的下场——OO第一单元总结

可以看出,复杂度比较高的几个点存在于对于表达式化简的部分。也就是说,其他部分都相对来说比较简单。说明我的架构还是可以接受的。

第三次作业

这一次,比较麻烦。因为出现了嵌套

最初构想的思考过程&&架构&&数据处理

架构部分

​ 对于加法项和乘法项,其实改造的时候都好说。因为都是Arraylist,往后添加就好了。问题就出现在sin,cos复合的情况。求导过程中肯定要出现链式法则,而链式法则大概是f(g(x))这样的形式,所以也就是说,一个表达式要嵌套一个表达式。

​ 既然出现了嵌套相同类型,就想到了递归算法。而什么时候停止递归下去呢?遇到基本项,比如x,sin(x),cos(x)这样的情况就停止。

​ 然后我就联想到了,可以把整个表达式建成一棵子节点不定的表达式树。他的节点有这几种类型:

  • 叶节点:
    • sin(x)
    • cos(x)
    • x
    • 常数
  • 只有一个儿子的节点
    • sin(f(x))型
    • cos(f(x))型
  • 有多个儿子的节点
    • 加法类型
    • 乘法类型

数据处理部分

​ 在正则表达式中,有一种东西叫做平衡组。使用平衡组能解决对于成对括号的匹配问题,从而把sin(f(x))之类的项给提出,以交给下一级处理。

实际遇到的问题

架构方面

前文给出的架构,实际上就是一个面向对象中的继承关系。我希望达到的目标是,向程序投喂一个String,然后程序自动判断对应的类型,建立相应类型的对象。但以我的知识无法解决(据说这个是要用工厂方法。这个我之后去学学看。。。。)

于是只能把所有类都归纳成一个Term,然后在其中定义一个Type属性,根据这个属性来判断这个对象的行为。

这也就造成了写出来的代码特别丑。

当然,也有因为时间的原因啦。我只有一天的时间去写完这作业,不敢轻举妄动。

数据处理部分

JAVA的正则表达式不支持平衡组!!!!!

既然正则不好使了,原本想过利用状态机加上递归的方法去判断并分割式子的(好像这叫做递归下降?),但是助教说不建议用状态机。并且之前自己没写过这样的东西,所以就不敢乱来。

我实际使用的是小正则判断加上for循环平衡搜索

  • 先用正则表达式判断是否为x,sin,cos(这一步我最后出了bug)
  • 如果不是,则试图采用加减法分割原表达式,分隔的时候,需要保证分隔的点之前括号是匹配的,以保证不会分割掉括号内的内容
  • 再试图按照乘法类型分割原表达式,同样要求括号匹配。
  • 如果上述尝试都无法成功,抛出无法识别类型异常

之所以我先分割加减法再分割乘除法,是因为乘法优先级更高。

被发现的bug&&找bug的思路

我的bug有两点:

  • 忘了还有一种类型是+(f(x)),少判断一种类型,直接爆炸
  • 如果出现了*++5这种情况。我的分割函数没有识别出来这个是一个格式错误

于是炸得香。强测加互测共炸了20个点。结果打了两次补丁就修完了。。。。

找bug的思路:

多层括号嵌套加上乘法

​ 如果没有对括号嵌套进行优化,如剥去外层的括号。那么多层括号很可能就直接让程序tle了。特别是加上乘法之后,深层树进行乘法运算,那酸爽。。。

一发入魂,hack了6个人

格式分隔错误

​ 这一次无法使用正则表达式了,所以也就是说,在分割字符串的时候肯定会有很多容易错的地方。

​ 我写了一个比较复杂的式子,然后在里面把能加符号的地方全部加上符号,再配合以空格。hack了2人

利用工具对自己的代码进行分析

不学无术的下场——OO第一单元总结

不学无术的下场——OO第一单元总结

可以看出,这一次的代码是写得真的丑。主要是我的知识无法支撑我的架构,于是只能以比较粗暴的方式去写了。所以JAVA基础知识还是要学好,以我为诫。。。