Codeforces Hello 2018 E题Logical Expression dp+最短路 好题
j题目链接: http://codeforces.com/contest/913/problem/E
题意:
给你x,y,z三个变量,与& 或| 非! 括号() 四种运算符,规定括号和非优先级最高,其次与&,最后或|。
对于输入的真值表,输出最小字典序的逻辑表达式
挺难想的。关键点在于怎么定义状态并利用已有状态更新出新状态,并处理优先级问题。
状态转移:
对于两个表达式A1,A2及其真值表T1,T2,若没有优先级的问题,A1&A2的真值表为T1&T2,A1|A2的真值表为T1|T2,!A1的真值表为~T1。
为了解决优先级的问题,按优先级不同进行分类,以dp[i][j]存储真值表为i,优先级为j的字典序最小逻辑表达式。,操作只在相同优先级表达式之间进行,维护一个优先级之间的升降通道。
模仿dijkstra的思想,维护一个序列,每次选出字典序最小的表达式作为A1并pop掉,在dp[i][j]中枚举A2进行更新操作。
真值表共有2^(2^3)共256种,优先级共3种可能。状态数n=3*256。
每种状态至多作为A1被考虑一次,枚举A2O(n),综合得算法复杂度O(n^2)
AC代码:
#include<bits/stdc++.h>
using namespace std;
const int FULL=0xFF;
bool cmp(string a,string b)
{
if(a.length()==b.length()) return a<b;
return a.length()<b.length();
}
struct Expression
{
int table;
int level;
string exp;
Expression(int a,int b,string c) {table=a;level=b;exp=c;}
bool operator < (const Expression& b)const
{
if(b.exp==exp)
{
if(level!=b.level) return level<b.level;
return table<b.table;
}
return cmp(exp,b.exp);
}
};
set<Expression> q;
string dp[<<][];
void add(int level,int state,string s)
{
if(dp[state][level].empty()||cmp(s,dp[state][level]))
{
dp[state][level]=s;
Expression e(state,level,s);
q.insert(e);
}
}
void pre()
{
int X_table=,Y_table=,Z_table=;
for(int i=;i<;++i)
{
if(i&(<<)) X_table+=(<<i);
if(i&(<<)) Y_table+=(<<i);
if(i&(<<)) Z_table+=(<<i);
}
add(,X_table,"x");
add(,Y_table,"y");
add(,Z_table,"z");
while(!q.empty())
{
set<Expression>:: iterator it;
it=q.begin();
Expression u=*it;
q.erase(it);
// auto u=*q.begin();
// q.erase(q.begin());
if(u.level==)
{
add(,u.table^FULL,"!"+u.exp);
add(,u.table,u.exp);
}
if(u.level==)
{
for(int i=;i<=FULL;++i)
{
if(!dp[i][].empty())
{
add(,i&u.table,dp[i][]+"&"+u.exp);
add(,i&u.table,u.exp+"&"+dp[i][]);
}
}
add(,u.table,"("+u.exp+")");
add(,u.table,u.exp);
}
if(u.level==)
{
for(int i=;i<=FULL;++i)
{
if(!dp[i][].empty())
{
add(,i|u.table,dp[i][]+"|"+u.exp);
add(,i|u.table,u.exp+"|"+dp[i][]);
}
}
add(,u.table,"("+u.exp+")");
}
}
}
void work()
{
int n;
scanf("%d",&n);
string s;
for(int i=;i<=n;++i)
{
cin>>s;
// cout<<"s="<<s<<endl;
int state=;
for(int j=;j<;++j)
{
if(s[j]=='') state|=(<<j);
}
//printf("state=%d\n",state);
cout<<dp[state][]<<endl;
}
}
int main()
{
pre();
work();
return ;
}
Codeforces Hello 2018 E题Logical Expression dp+最短路 好题的更多相关文章
- 思维题练习专场-DP篇(附题表)
转载请注明原文地址http://www.cnblogs.com/LadyLex/p/8536399.html 听说今年省选很可怕?刷题刷题刷题 省选已经结束了但是我们要继续刷题刷题刷题 目标是“有思维 ...
- codeforces 931E Logical Expression dp
time limit per test 3 seconds memory limit per test 256 megabytes input standard input output standa ...
- 【CodeForces】913 E. Logical Expression
[题目]E. Logical Expression [题意]令x=11110000(2),y=11001100(2),z=10101010(2),n次询问,每次要求用[与][或][非][括号]构成含至 ...
- 2018 HDU多校第四场赛后补题
2018 HDU多校第四场赛后补题 自己学校出的毒瘤场..吃枣药丸 hdu中的题号是6332 - 6343. K. Expression in Memories 题意: 判断一个简化版的算术表达式是否 ...
- 2018 HDU多校第三场赛后补题
2018 HDU多校第三场赛后补题 从易到难来写吧,其中题意有些直接摘了Claris的,数据范围是就不标了. 如果需要可以去hdu题库里找.题号是6319 - 6331. L. Visual Cube ...
- Codeforces 438E The Child and Binary Tree [DP,生成函数,NTT]
洛谷 Codeforces 思路 看到计数和\(998244353\),可以感觉到这是一个DP+生成函数+NTT的题. 设\(s_i\)表示\(i\)是否在集合中,\(A\)为\(s\)的生成函数,即 ...
- 乘风破浪:LeetCode真题_010_Regular Expression Matching
乘风破浪:LeetCode真题_010_Regular Expression Matching 一.前言 关于正则表达式我们使用得非常多,但是如果让我们自己写一个,却是有非常大的困难的,我们可能想到状 ...
- Codeforces 148D Bag of mice:概率dp 记忆化搜索
题目链接:http://codeforces.com/problemset/problem/148/D 题意: 一个袋子中有w只白老鼠,b只黑老鼠. 公主和龙轮流从袋子里随机抓一只老鼠出来,不放回,公 ...
- [CodeForces - 1272D] Remove One Element 【线性dp】
[CodeForces - 1272D] Remove One Element [线性dp] 标签:题解 codeforces题解 dp 线性dp 题目描述 Time limit 2000 ms Me ...
随机推荐
- 【转载】MessageFormat.format方法
MessageFormat.format方法实例 public static void main(String[] args) { String a= "aaa"; String ...
- Spvmn测试环境搭建及其安全性讨论
一.说明 这几天都在做设备的协议分析,然后看到有个叫Spvmn的不懂要怎么操作才能触发其操作过程,问了测试部的同事说也没有测试文档,自己研究了一下这里做个记录. 按我现在理解,各厂商有自己的私有协议. ...
- luogu1706全排列
#include<bits/stdc++.h> using namespace std; +]; +]; int search(int k); int print(); int n,num ...
- AStar算法()
把网上的AStar算法的论述自己实现了一遍,一开始只是最基础的实现.当然,现在AStar算法已经演变出了各种优化的版本,这篇也会基于各种优化不断的更新. 如果对算法不熟悉可以看下Stanford的这篇 ...
- Python语言:Day11练习题
24.实现一个整数加法计算器:如:content=input('请输入内容:')#如:5+9或5+ 9或5 + 9 content = input("请输入需要计算的:") #如: ...
- Vultr新用户充值优惠 – 最多充值100美元送100美元
Vultr商家向来是搅局的,当初海外VPS商家被Linode一家独大的时候,由于VULTR商家进入市场进行相似产品的营销,使得目前我们看到海外主机商各种低价.当然这些说法也有些武断,但是肯定是有一定的 ...
- Python中的传参是传值还是传址?
传值:在C++中,传值就是把一个参数的值给这个函数,其中的更改不会影响原来的值. 传址:即传引用,直接把这个参数的内存地址传递进去,直接去这个内存地址上进行修改. 但是这些在Python中都没有,Py ...
- float和position谁好?
float从字面上的意思就是浮动,float能让元素从文档流中抽出,它并不占文档流的空间,典型的就是图文混排中文字环绕图片的效果了.不过需要注意的是清除浮动是我们可能需要注意的地方.而position ...
- Problem 3: Largest prime factor
The prime factors of 13195 are 5, 7, 13 and 29. What is the largest prime factor of the number 60085 ...
- js★★★【面向对象的使用方法】*****************★★★★ 相当重要
标准用法: function Sprite(){ //函数内容部设置属性 this.name='shimily'; } //原型上设置方法 Sprite.prototype.show=function ...