模拟 机器翻译

模拟 机器翻译

大概题意:有M个单元均可存放一单词。但出现单元里面没有的新单词时,查询一次,如果当前内存中已存入的单词数不超过M-1,自动存储在后面;若内存中已存入M个单词,清空最早储存的单词,来存放新单词。
长度为N个单词。需要查找多少次词典?
思路很简单,先暴力搜索,没有就考虑加入单元,没满直接加后面,满了从已经删除的地方+1开始删除,这个地方肯定是单元里面现有的最先进来的,替代它就好。

#include <stdio.h>
#include <iostream>
#include <cstdlib>
#include <cmath>
#include <cctype>
#include <string>
#include <cstring>
#include <algorithm>
#include <stack>
#include <queue>
#include <set>
#include <map>
#include <ctime>
#include <vector>
#include <fstream>
#include <list>
#include <iomanip>
#include <numeric>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
const int inf = 0x3f3f3f3f;
struct  Dir
{
	int d[105];
	int c,len=0,num=0;
	bool find(int n){
		for(int i=0;i<len;i++){
			if(d[i]==n){
				return true;
			}
		}
		return false;
	}
	void Add(int n){
		if(len>=c){
			d[num%c]=n;
			num++;
		}else{
			d[len++]=n;
		}
	}
};
int main() 
{
	Dir t;
	int m,n,tmp,cnt=0;
	cin>>t.c>>n;
	while(n--){
		cin>>tmp;
		if(t.find(tmp)){
			continue;
		}else{
			t.Add(tmp);
			cnt++;
		}
	}	
	cout<<cnt<<endl;
    return 0;
}

下面开始膜大佬

引用网址

先开两个数组,一个数组a存标记,在读入单词x时,若当前数组a中的第x位的标记为“1”,则表示该单词在内存中,若标记为“0”,则表示该单词不在内存中,这样做就可以一步判断读入的单词在当前是否在内存中,而不必从头到尾找。这样做还有一个好处,在存入单词时只需要把数组a中的第x位的标记“0”改为“1”,而在删去内存中的单词x时,只需要把数组a的第x位的标记“1”改为“0”,这样就可以做到一步到位,大大降低时间复杂度,提高程序效率。这是本题的关键之一,需要各位好好体会体会,理解理解。

再说另一个数组b。数组b存储的是内存中的单词,而且要按读入顺序存入,如b[1]中存储的单词x是在时间1存入的。要注意的是,若当前读入的单词x已经在内存中(即a[x]==1)那就不用存入数组b中,遇到内存中没有的新单词才存入。

接下来要讲指针。指针有两个,一个是l,指向当前内存中的单词中最先一个存入的,如b[l]是当前内存中第一个存入的。另一个是r,指向当前内存中的单词中最后一个存入的,如b[r]是当前内存中最后一个存入的。所以,数组b的第l位到第r位存储的就是当前内存中的单词。

遇到新单词时(即a[x]==0),情况有两种:

1.内存没被用完(即r<=m)。此时指针r向右移一位,在b[r]中存入新单词,并在数组a中把单词x的标记改为1;

2.内存已满(即r>m)。此时先删去当前内存中最先存入的单词(b[l])。删除操作不用太复杂,只需要先把指针l向右移一位,然后再修改a数组的第b[l]位的标记就可以了,可联系上文加深理解。不要忘了最后在b[r]中存入新单词x。

这题大概就是这样做了,如果你感觉可以自己AC掉,就先不要看以下程序,自己试着打打程序。如果WA了就再认真看看上文,如果AC了就把你的程序和以下程序对比对比,看看有什么可以改进的地方。

代码如下:

  #include <iostream>
    #include <stdio.h>
    #include <algorithm>
    using namespace std;
    int n,m,x,ans,l,r,a[1005],b[1005];
    int main()
    {
        cin>>m>>n;
        l=0;r=0;//初始化两个指针
        for (int i=1;i<=n;i++)
         {
             scanf("%d",&x);//边读入边做
             if (a[x]==0) 
             {
                 ans++;
                r++;b[r]=x;a[x]=1;//因为每次遇到新单词都要做这些操作,不如搬到判断语句外做,这样程序更简洁
                if (r>m) {l++;a[b[l]]=0;}
             }
         }
        cout<<ans;
        return 0;//千万不能忘记打这句,不然在比赛中会出错
}
posted @ 2018-10-14 19:43  Janspiry  阅读(146)  评论(0编辑  收藏  举报