careercup-递归和动态规划 9.11

9.11 给定一个布尔表达式,由0、1、&、|和^等符号组成,以及一个想要的布尔结果result,实现一个函数,算出有几种括号的放法可使该表达式得出result值。

解法:

跟其他递归问题一样,此题的关键在于找出问题与子问题之间的关系。

假设函数int f(expression,result)会返回所有值为return的有效表达式的数量。我们想要算出f(1^0|0|1,true)(也即,给表达式1^0|0|1加括号使其求值为true的所有方式)。每个加括号的表达式最外层肯定有一对括号。因此,我们可以这么做:

也就是说,我们可以迭代整个表达式,将每个运算符当作第一个要加括号的运算符。

现在,又该如何计算这些内层的表达式呢,比如f((1^0)|(0|1),true)?很简单,要让这个表达式的值为true,左半部分和右半部分必有一位true。因此,这个表达式分解如下:

f((1^0) | (0|1),true)= f(1^0,true)* f(0|1,true) +

            f(1^0,false)* f(0|1,true)+

            f(1^0,true) * f(0|1,false)

对每个布尔表达式,都可以进行类似的分解:

对false结果,我们也可以执行非常类似的操作:

至此,要解决这个问题,只需反复套用这些递归关系即可。

C++实现代码:

#include<iostream>
#include<string>
using namespace std;

int f(string exp,bool result,int s,int e)
{
    if(s==e)
    {
        if(exp[s]==‘1‘&&result)
            return 1;
        else if(exp[s]==‘0‘&&!result)
            return 1;
        else
            return 0;
    }
    int c=0;
    int i;
    if(result)
    {
        for(i=s+1; i<=e; i+=2)
        {
            if(exp[i-1]==‘&‘)
            {
                c+=f(exp,true,s,i-1)*f(exp,true,i+1,e);
            }
            else if(exp[i]==‘|‘)
            {
                c+=f(exp,true,s,i-1)*f(exp,false,i+1,e);
                c+=f(exp,false,s,i-1)*f(exp,true,i+1,e);
                c+=f(exp,true,s,i-1)*f(exp,true,i+1,e);
            }
            else if(exp[i]==‘^‘)
            {
                c+=f(exp,true,s,i-1)*f(exp,false,i+1,e);
                c+=f(exp,false,s,i-1)*f(exp,true,i+1,e);
            }
        }
    }
    else
    {
        for(i=s+1;i<=e;i+=2)
        {
            if(exp[i-1]==‘&‘)
            {
                c+=f(exp,true,s,i-1)*f(exp,false,i+1,e);
                c+=f(exp,false,s,i-1)*f(exp,true,i+1,e);
                c+=f(exp,false,s,i-1)*f(exp,false,i+1,e);
            }
            else if(exp[i]==‘|‘)
            {
                c+=f(exp,false,s,i-1)*f(exp,false,i+1,e);
            }
            else if(exp[i]==‘^‘)
            {
                c+=f(exp,true,s,i-1)*f(exp,true,i+1,e);
                c+=f(exp,false,s,i-1)*f(exp,false,i+1,e);
            }
        }
    }
    return c;
}

int main()
{
    string str="1^0|0&1&0|1^1^0|1|1&0&1^0|0&1&0|1^1^0|1|1&0^1^0|0&1&0|1^1^0|1|1&0^1^0|0&1&0|1^1^0|1|1&0|1|0|0";
    cout<<f(str,true,0,4)<<endl;
}

虽然这么做可行,但不是很有效,对于同一个exp的值,他会重复算f(exp)很多次。

要解决这个问题,我们可以运用动态规划,缓存不同表达式的结果。注意,我们需要根据expression和result进行缓存。

dp C++实现代码:

#include<iostream>
#include<string>
#include<map>
using namespace std;

int f(string exp,bool result,int s,int e,map<string,int> &mp)
{
    string key=""+result+s+e;
    if(mp.find(key)!=mp.end())
        return mp[key];
    if(s==e)
    {
        if(exp[s]==‘1‘&&result)
            return 1;
        else if(exp[s]==‘0‘&&!result)
            return 1;
        else
            return 0;
    }
    int c=0;
    int i;
    if(result)
    {
        for(i=s+1; i<=e; i+=2)
        {
            if(exp[i-1]==‘&‘)
            {
                c+=f(exp,true,s,i-1,mp)*f(exp,true,i+1,e,mp);
            }
            else if(exp[i]==‘|‘)
            {
                c+=f(exp,true,s,i-1,mp)*f(exp,false,i+1,e,mp);
                c+=f(exp,false,s,i-1,mp)*f(exp,true,i+1,e,mp);
                c+=f(exp,true,s,i-1,mp)*f(exp,true,i+1,e,mp);
            }
            else if(exp[i]==‘^‘)
            {
                c+=f(exp,true,s,i-1,mp)*f(exp,false,i+1,e,mp);
                c+=f(exp,false,s,i-1,mp)*f(exp,true,i+1,e,mp);
            }
        }
    }
    else
    {
        for(i=s+1;i<=e;i+=2)
        {
            if(exp[i-1]==‘&‘)
            {
                c+=f(exp,true,s,i-1,mp)*f(exp,false,i+1,e,mp);
                c+=f(exp,false,s,i-1,mp)*f(exp,true,i+1,e,mp);
                c+=f(exp,false,s,i-1,mp)*f(exp,false,i+1,e,mp);
            }
            else if(exp[i]==‘|‘)
            {
                c+=f(exp,false,s,i-1,mp)*f(exp,false,i+1,e,mp);
            }
            else if(exp[i]==‘^‘)
            {
                c+=f(exp,true,s,i-1,mp)*f(exp,true,i+1,e,mp);
                c+=f(exp,false,s,i-1,mp)*f(exp,false,i+1,e,mp);
            }
        }
    }
    mp[key]=c;
    return c;
}

int fDP(string exp,bool result,int s,int e)
{
    map<string,int> dp;
    return f(exp,result,s,e,dp);
}
int main()
{
    string str="1^0|0&1&0|1^1^0|1|1&0&1^0|0&1&0|1^1^0|1|1&0^1^0|0&1&0|1^1^0|1|1&0^1^0|0&1&0|1^1^0|1|1&0|1|0|0";
    cout<<fDP(str,true,0,4)<<endl;
}
时间: 2024-10-07 13:01:37

careercup-递归和动态规划 9.11的相关文章

斐波那契数列的实现(简单递归和动态规划)

斐波那契数列的实现(简单递归和动态规划) 一.简单递归的实现 1 #include "stdafx.h" 2 #include <string> 3 using namespace std; 4 int f(int n) 5 { 6 if (n == 0) 7 { 8 return 0; 9 } 10 if (n == 1) 11 { 12 return 1; 13 } 14 return f(n - 1) + f(n - 2); 15 } 16 int _tmain(in

算法初级面试题08——递归和动态规划的精髓、阶乘、汉诺塔、子序列和全排列、母牛问题、逆序栈、最小的路径和、数组累加成指定整数、背包问题

第八课主要介绍递归和动态规划 介绍递归和动态规划 暴力递归: 1,把问题转化为规模缩小了的同类问题的子问题 2,有明确的不需要继续进行递归的条件(base case) 3,有当得到了子问题的结果之后的决策过程 4,不记录每一个子问题的解 动态规划 1,从暴力递归中来 2,将每一个子问题的解记录下来,避免重复计算 3,把暴力递归的过程,抽象成了状态表达 4,并且存在化简状态表达,使其更加简洁的可能 图灵引入的是:我不知道怎么算,但是我知道怎么试.知道怎么暴力破解出来. 要学会,练习懂得怎么尝试.

左神算法第八节课:介绍递归和动态规划(汉诺塔问题;打印字符串的全部子序列含空;打印字符串的全排列,无重复排列;母牛数量;递归栈;数组的最小路径和;数组累加和问题,一定条件下最大值问题(01背包))

暴力递归: 1,把问题转化为规模缩小了的同类问题的子问题 2,有明确的不需要继续进行递归的条件(base case) 3,有当得到了子问题的结果之后的决策过程 4,不记录每一个子问题的解 动态规划 1,从暴力递归中来 2,将每一个子问题的解记录下来,避免重复计算 3,把暴力递归的过程,抽象成了状态表达 4,并且存在化简状态表达,使其更加简洁的可能 一:递归 1. 汉诺塔问题 汉诺塔问题(不能大压小,只能小压大),打印n层汉诺塔从最左边移动到最右边的全部过程. 左中右另称为 from.to.hel

【动态规划专题】1:斐波拉契数列问题的递归和动态规划

<程序员代码面试指南--IT名企算法与数据结构题目最优解> 左程云 著 斐波拉契数列问题的递归和动态规划 [题目]:给定整数N,返回斐波拉契数列的第N项.补充问题1:给定整数N,代表台阶数,一次可以跨2个或者1个台阶,返回有多少种走法.补充问题2:假设农场中成熟的母牛每年只会生产1头小母牛,并且永远不会死.第一年农场只有1只成熟的母牛,从第2年开始,母牛开始生产小母牛.每只小母牛3年后成熟又可以生产小母牛.给定整数N,求出N年后牛的数量. [举例]斐波拉契数列f(0)=0, f(1)=1,f(

70. Climbing Stairs【leetcode】递归,动态规划,java,算法

You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb 1 or 2 steps. In how many distinct ways can you climb to the top? Note: Given n will be a positive integer. 题目分析:每次只能走1或2步,问n步的话有多少中走法???? 可以用动态规划和递归解

利用递归和动态规划来求解组合数

组合数定义:从m个不同元素中,任取n(n≤m)个元素并成一组,叫做从m个不同元素中取出n个元素的一个组合:从m个不同元素中取出n(n≤m)个元素的所有组合的个数,叫做从m个不同元素中取出n个元素的组合数. 下面是一种比较通俗的计算公式: 其递归公式为: c(n,m)=c(n-1,m-1)+c(n-1,m) 下面是c++实现该递归算法: #include <iostream> #include <stdlib.h> #define EXIT -1 using namespace st

爬楼问题—递归、动态规划

问题:假设你正在爬楼梯,需要n步你才能到达顶部.但每次你只能爬一步或者两步,你能有多少种不同的方法爬到楼顶部? code: 1 //动态规划解决爬楼问题 2 int dp_climbStairs(int n) { 3 if (n == 1) { 4 return 1; 5 } else if (n == 2) { 6 return 2; 7 } else { 8 int * M = new int[n]; 9 for (int i = 0; i < n; i++) { 10 M[i] = 0;

背包问题---递归及动态规划

一.原题 如果有一组物品,各个物品的质量已知,现有一个背包,背包可以容纳的质量总和S已知,问是否能从这N个物品中取出若干个恰好装入这个背包中. 二.递归算法 本质思想:设法尝试全部组合,当部分组合已经无法满足条件时,马上停止当前组合的尝试:若出现第一个满足条件的组合,马上停止尝试.使用递归回溯法实现.(感觉这东西不是我这样的菜鸟可以说明确的,还得自己慢慢体会,最好的方法就是耐住性子跟踪调试). 上"酸菜" #include <stdio.h> #include <st

java——递归(动态规划,回溯)

最近刷面试题经常刷到递归方面的算法,一直以为都是递归,后来发现竟然都有具体的叫法,所以写了这篇博客来牢记以下 1. 五大常用算法 (1) 分治算法 把一个复杂的问题分成两个或多个相同或者相似的子问题,然后不断地细分,直到最后的子问题可以很简单地求解出来,原问题的解就是自问题的合并.比如常见的快速排序算法和归并算法 分治法的核心思想就是把大的难解的问题不断分割,分而治之. (2) 动态规划 类似于分治法,将带求解的问题分化成几个小问题,每个小问题的解会影响原问题的解. 先求每个子问题的局部解,然后