`
j2eetop
  • 浏览: 64383 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

《开源框架那点事儿14》:教计算机程序解数学题

 
阅读更多


周末,看关于专家系统方面的书,其中有关于规则方面的内容,忽然就想,能不能模仿人的学习方式来提升计算机程序的计算能力呢? 

 试想,一个小孩子,他一开始什么也不会,首先,你要告诉他什么是数字,然后告诉他什么是加、减;然后告诉他什么是乘、除,还要告诉他有乘、除要先计算乘除,然后又引入了括号说,有括号永远要先计算括号。如此,随着告诉他的技能越多,他的解题能力也就越强。 
于是就想着试验一下。 
第一步,教计算机学习什么是数字。 
下面的正则表达式,就是告诉“孩子”,数字就是前面可能有“-”号,当然也可能没有,接下来连续的数字0-9,组成的数字,后面可能还会有小数点开始加一堆0-9的数字,当然没有也没有关系。如此,它就算懂得认数字了。 

 

public final class MathNumber {
    private MathNumber() {
    }
 
    public static String numberPattern = "[-]?[0-9]+([.][0-9]*)?";
    public static Pattern pattern = Pattern.compile(numberPattern);
 
    public static Matcher match(String string) {
        Matcher match = pattern.matcher(string);
        if (match.find()) {
            return match;
        }
        throw new RuntimeException(string + " is not a number.");
    }
}

 

第二步就是告诉“孩子”,计算数学题的过程。 
如果两边有空格就忽略它,然后呢,看看是不是已经是一个数字了,如果已经是一个数字,那说明就算出结果了。如果不是,就从最高优先级找起,如果找就就计算。如果找不到,说明这个式子有问题,不是一个合法的数学式子。

public static String eval(String string) {
        string = string.trim();
        while (!isMathNumber(string)) {// 同一优先级的哪个先找到算哪个
            System.out.println("求解算式:" + string);
            boolean found = false;
            for (MathInterface math : mathList) {
                Matcher matcher = math.match(string);
                if (matcher.find()) {
 
                    String exp = matcher.group();
                    String sig = "";
                    if (exp.charAt(0) == '-' && matcher.start() != 0) {// 如果不是第一个数字,-号只能当运算符
                        sig = "+";
                    }
                    System.out.println("发现算式:" + exp);
                    String evalResult = math.eval(exp);
                    string = string.substring(0, matcher.start()) + sig
                            + evalResult + string.substring(matcher.end());
                    System.out.println(exp + "计算结果为:" + evalResult + ",代回原式");
                    found = true;
                    break;
                }
            }
            if (!found) {
                throw new RuntimeException(string + " 不是合法的数学表达式");
            }
        }
        return string;
    }


从现在开始,这孩子已经会解题思路了,不过他还是啥也不懂,他还不知道啥是加,减、乘、除啥的,没有办法,孩子笨,只要多教他了。 
下面就教他如何计算,加、减、乘、除、余、括号、指数。

addMathExpression(new Add());
 addMathExpression(new Subtract());
 addMathExpression(new Multiply());
 addMathExpression(new Devide());
 addMathExpression(new Minus());
 addMathExpression(new Factorial());
 addMathExpression(new Remainder());
 addMathExpression(new Bracket());
 addMathExpression(new Power());
 Collections.sort(mathList, new MathComparator());


由于大同小异,就里就只贴出来加法和括号的实现方式。 
加法实现,它的优先级是1,它是由两个数字中间加一个“+”号构成,数字和加号前面的空格没用,不用管它。计算的时候呢,就是用加的方式把两个数字加起来,这一点计算机比人强,呵呵,告诉他怎么加永远不会错的。而且理解起加减乘除先天有优势。 

public class Add implements MathInterface {
    static String plusPattern = BLANK + MathNumber.numberPattern + BLANK
            + "[+]{1}" + BLANK + MathNumber.numberPattern + BLANK;
    static Pattern pattern = Pattern.compile(plusPattern);
    static Pattern plus = Pattern.compile(BLANK + "\\+");
 
    @Override
    public Matcher match(String string) {
        return pattern.matcher(string);
    }
 
    @Override
    public int priority() {
        return 1;
    }
 
    @Override
    public String eval(String expression) {
        Matcher a = MathNumber.pattern.matcher(expression);
        if (a.find()) {
            expression = expression.substring(a.end());
        }
        Matcher p = plus.matcher(expression);
        if (p.find()) {
            expression = expression.substring(p.end());
        }
        Matcher b = MathNumber.pattern.matcher(expression);
        if (b.find()) {
 
        }
        return new BigDecimal(a.group()).add(new BigDecimal(b.group()))
                .toString();
    }
 
}


接下来是括号,括号的优先级是最大啦,只要有它就应该先计算。当然,要先计算最内层的括号中的内容。括号中的内容,计算的时候,可以先拉出来,不用管外面的内容,计算好了,放回去就可以了。 

public class Bracket implements MathInterface {
 
    static String bracketPattern = BLANK + "[(]{1}[^(]*?[)]" + BLANK;
    static Pattern pattern = Pattern.compile(bracketPattern);
 
    @Override
    public Matcher match(String string) {
        return pattern.matcher(string);
    }
 
    @Override
    public int priority() {
        return Integer.MAX_VALUE;
    }
 
    @Override
    public String eval(String expression) {
        expression = expression.trim();
        return MathEvaluation.eval(expression.substring(1,
                expression.length() - 1));
    }
 
}


到目前为止,我们的程序“宝宝”已经学会数学计算了,出个题让伊试试。 

public static void main(String[] args) {
String string = "1+2^(4/2)+5%2";
System.out.println("结果是 :" + MathEvaluation.eval(string));
}


程序宝宝的做题过程如下:

求解算式:1+2^(4/2)+5%2
发现算式:(4/2)
求解算式:4/2
发现算式:4/2
4/2计算结果为:2.00,代回原式
(4/2)计算结果为:2.00,代回原式
求解算式:1+2^2.00+5%2
发现算式:2^2.00
2^2.00计算结果为:4,代回原式
求解算式:1+4+5%2
发现算式:5%2
5%2计算结果为:1,代回原式
求解算式:1+4+1
发现算式:1+4
1+4计算结果为:5,代回原式
求解算式:5+1
发现算式:5+1
5+1计算结果为:6,代回原式
结果是 :6


呵呵,程序宝宝的做题过程和人的做题过程非常一致,而且程序实现也非常简单易懂。神马编译原理,神马中缀表达式都用不上。(执行效率与其它算法比较不一定高,仅用于验证通过规则让程序的处理能力增强,由于没有进行深入测试,正则表达式和程序逻辑是否写得严密没有经过深入验证) 

其实程序虽然很简单,但是,实际上已经是一个简单的规则引擎的雏形。 
首先,他加载了许多的业务处理规则,加,减,乘,除,插号,指数,余数等等。 
第二,他的业务规则是可以不断进行扩展的。 
第三,只要给出事实,最后,他通过规则的不断应用,最后会导出结果,要么是正确的结果,要么说给出的事实是错误的。 

需要源码的童鞋请到GIT上直接获取代码。

git地址:http://git.oschina.net/tinyframework/mathexp.git

 


 

 点击右上角的watch、star、fork,变成unwatch、unstar、unfork(如果Tiny项目没有fork到用户空间的话,后面clone命令会出错),以支持Tiny框架发展!

 

 

欢迎访问开源技术社区:http://bbs.tinygroup.org。本例涉及的代码和框架资料,将会在社区分享。《自己动手写框架》成员群:228977971,让我们一起动手,了解开源框架的奥秘! 

 

分享到:
评论

相关推荐

    .net开源框架

    .NET开源框架是软件开发领域中的一个重要组成部分,它为开发者提供了构建高效、稳定和可扩展的应用程序的基础设施。本文将深入探讨.NET开源框架的核心概念、功能特性以及如何利用它来加速开发进程。 首先,.NET框架...

    WPF MVVM 开源框架,包含数据验证,很不错的

    **WPF MVVM 开源框架概述** WPF(Windows Presentation Foundation)是微软.NET Framework的一部分,用于构建具有丰富用户界面的应用程序。MVVM(Model-View-ViewModel)是一种设计模式,常用于WPF开发中,它将业务...

    基于Devexpress的winform开源框架,带源码

    ### 基于Devexpress的WinForm开源框架知识点解析 #### 一、DevExpress与WinForms简介 DevExpress是一家专注于为.NET平台提供高质量控件组件的软件公司。DevExpress的产品因其丰富的功能、出色的性能以及良好的用户...

    STM32的开源Bootloader框架 APP 程序制作

    STM32的开源Bootloader框架 APP 程序制作 文档对应的源码程序 https://blog.csdn.net/weixin_39946218/article/details/124514645

    SSH面试题总结 开源框架

    SSH面试题总结涵盖了三个主要的开源框架:Struts、Spring和Hibernate。这些框架在Java Web开发中起着至关重要的作用,帮助开发者更高效地管理和组织应用程序的各个层面。 首先,让我们来看看Hibernate,这是一个...

    一种开源业务系统框架winform源码

    【标题】"一种开源业务系统框架winform源码"揭示了这是一个基于Windows Forms(Winform)的开源软件开发框架,专用于构建业务系统。Winform是.NET Framework中的一个UI库,用于创建桌面应用程序,它提供了丰富的用户...

    saas 开源开发框架

    tosaas 开发框架是一款通用的 saas 开发框架,能快速的将传统单企业应用转化成saas多租户模式,支持大数据存储高并发访问,支持无限水平扩展存储服务器、应用服务器数量,经受过百万级用户的实际应用考验。

    csp开源框架 csp 11 (带pkcs)

    CSP(Cryptographic Service Provider)开源框架是用于提供加密服务的软件组件,它与操作系统紧密集成,为应用程序提供加密、签名、哈希等密码学功能。CSP 11 版本是该框架的一个更新迭代,它引入了PKCS #11标准的...

    android开源框架大全(demo)

    这是一份开源框架的demo,内含可执行程序+源代码,含有22款开源框架的demo演示,内含注释。博客:https://blog.csdn.net/u011038298/article/details/82258835

    掌上题库V1.2.2全开源版本小程序带后端

    本模块支持 多选题、单选题、填空题(支持多空)、判断题、语音题(即一段语音下面多个选择题)等题型 2、可导入内部学员信息,创建内部学员专属题库 3、后台可批量生成指定题库激活码,学员输入激活码可激活...

    ERP开源框架

    一个很好的ERP开源框架,大家有兴趣的可以下载去研究学习;

    java开发常用开源框架架构图

    "Java开发常用开源框架架构图" 本资源摘要信息对 Java 开发中常用的开源框架架构图进行了介绍,涵盖了Struts、Hibernate、Spring、iBatis 等多种框架的架构图和简介文章分类。这些框架都是 Java 开发中最常用的框架...

    史上最酷最炫的PB开源框架

    【标题】:“史上最酷最炫的PB开源框架” 在IT领域,开源框架一直是开发者们热衷探讨的话题。PB,全称PowerBuilder,是由Sybase公司开发的一种可视化的、面向对象的编程工具,主要用于构建数据库应用系统。而"史上...

    Silverlight权限系统开发开源框架

    在IT行业中,Silverlight是一种由微软开发的富互联网应用程序(RIA)平台,它结合了XAML、.NET Framework和浏览器插件技术,用于创建交互性强、视觉效果丰富的网络内容。本开源框架专注于权限系统的开发,为开发者...

    [开源]汇编程序库

    1:判断闰年程序 2:产生随机数并计算程序 3:计算平台长度程序 4:加减乘除运算 5:数字方阵程序 6:数据加密程序 7:代码转换程序 8:键盘录入数据的转换与显示 9:音乐演奏 10:显示色彩矩形块 11、12:字符图形...

    SimpleCV:机器视觉的开源框架

    使用SimpleCV(计算机视觉的开源框架)使计算机具有视觉效果 SimpleCV是使用OpenCV和Python编程语言的开放源代码机器视觉的框架。 它为相机,图像处理,特征提取和格式转换提供了简洁易读的界面。 我们的使命是为...

    强烈推荐:绝对是最好的一个小程序开源框架

    特别说明:这是我目前在接触过和使用过的,自认为是最好用的一个小程序开源框架库。 小程序到底有多火,看看目前推出的开源框架以及组件库就知道了。由于小程序开发的火爆,大家都在致力于探索如何更好的,更加高效...

    掌上题库V1.2.2全开源版本后端+小程序源码

    本模块支持 多选题、单选题、填空题(支持多空)、判断题、语音题(即一段语音下面多个选择题)等题型 2、可导入内部学员信息,创建内部学员专属题库 3、后台可批量生成指定题库激活码,学员输入激活码可激活指定...

    开源框架ActionBarSherlock 和 ViewPager 仿网易新闻客户端

    开源框架ActionBarSherlock 和 ViewPager 仿网易新闻客户端,项目详情:http://blog.csdn.net/xiaanming/article/details/9971721

    经典的基于ASP.NET MVC框架的开源博客系统

    一个国人写的开源blog系统,使用ASP.NET MVC框架开发. 具有以下优点: 1,代码精练,优雅,。 2,使用了大量的设计模式 3,采用N层架构设计,分层非常合理。 4,支持强大的扶肤功能 5,大量使用了.net Framework 3.5...

Global site tag (gtag.js) - Google Analytics