网易互娱笔试2021批题解

总结:不给数据范围超级差评…

T4

题目大意:a和b轮流出牌,打n轮。每一轮中你点数大分数+3平局分数+1,。给出了你手上有的手牌,以及b的出牌顺序。你要确定自己的出牌顺序,使得自己的总得分最大。牌面大小范围 [ 3 , 13 ] [3,13] [3,13]
数据范围未知。

有个十分显然的费用流模型…

将自己的手牌一列,对方的手牌一列,变成一个二分图模型…然后连 n 2 n^2 n2条边直接跑费用流即可。

有个显然的优化:还是二分图,每边放11个点,代表权值,然后连边方式类似,就能跑 n = 1 e 6 n=1e6 n=1e6的数据范围了。

由于没有Java费用流板子…以下是从网上扒的板子

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.Queue;
import java.util.Scanner;
import java.util.StringTokenizer;

public class zbr01 {
	static class InputReader
	{
	    BufferedReader buf;
	    StringTokenizer tok;
	    InputReader()
	    {
	        buf = new BufferedReader(new InputStreamReader(System.in));
	    }
	    boolean hasNext()
	    {
	        while(tok == null || !tok.hasMoreElements()) 
	        {
	            try
	            {
	                tok = new StringTokenizer(buf.readLine());
	            } 
	            catch(Exception e) 
	            {
	                return false;
	            }
	        }
	        return true;
	    }
	    String next()
	    {
	        if(hasNext()) return tok.nextToken();
	        return null;
	    }
	    int nextInt()
	    {
	        return Integer.parseInt(next());
	    }
	    long nextLong()
	    {
	        return Long.parseLong(next());
	    }
	    double nextDouble()
	    {
	        return Double.parseDouble(next());
	    }
	    BigInteger nextBigInteger()
	    {
	        return new BigInteger(next());
	    }
	    BigDecimal nextBigDecimal()
	    {
	        return new BigDecimal(next());
	    }
	}
	static 	Queue<Integer> q;
	static int n,m,con,start,end;
	static int cost,minflow;
	static int[]head ,dis, vis,belong_e ,  pre;
	static class e{
		int u,v,next,w, cost;
	}static e[]es;
	static void add(int u,int v,int w, int cost) {
		if(es[con]==null) es[con]=new e();
		es[con].cost=cost;es[con].v=v;es[con].w=w;es[con].next=head[u];head[u]=con++;
	}
	//这个本质也是dfs
	//就是为了找一条从源点到汇点的最短路
	static boolean spfa() {
		for(int i=0;i<=2*n+3;i++) {
			dis[i]=0x3f3f3f3f;
			vis[i]=0;
		}
		dis[start]=0;vis[start]=1;
		q.clear();q.add(start);
		while(!q.isEmpty()) {
			int u=q.poll();
			vis[u]=0;
			for(int i=head[u];i!=-1;i=es[i].next) {									
				  int v=es[i].v;
				  if( es[i].w>0 &&  dis[v]> dis[u]+es[i].cost ) {
					  pre[v]=u;belong_e[v]=i;
					  dis[v]=dis[u]+es[i].cost;
					  if(vis[v]==0) {	
						  q.add(v);
						  vis[v]=1;
					  }
				  }
			}
		}
		return dis[end]<0x3f3f3f3f;
	}
	
	public static int a[]={8,7,5};
	public static int b[]={7,4,9};
	
	public static int wk(int a[],int b[])
	{
		InputReader  sc=new InputReader();
		head=new int[5005];
		belong_e=new int[150000];
		pre=new int[5005];
		dis=new int[5005];
		vis=new int[5005];
		es=new e[150005];
		q=new LinkedList<Integer>();
		
		
		n=a.length;
		start=2*n+1; end=2*n+2;
		
		
		Arrays.fill(head, -1);con=0;
		
		for(int i=0;i<n;i++) for(int j=0;j<n;j++)
		{
			if(a[i]>a[j])
			{
				add(i,n+j,1,-3);
				add(n+j,i,0,3);
			}
			
			else if(a[i]==a[j])
			{
				add(i,n+j,1,-1);
				add(n+j,i,0,1);
			}
			
			else
			{
				add(i,n+j,1,0);
				add(n+j,i,0,0);
			}
		}
		
		for(int i=0;i<n;i++) 
		{
			add(start,i,1,0);
			add(i,start,0,0);
			
			add(n+i,end,1,0);
			add(end,n+i,1,0);
		}
		
		cost=0;
		minflow=0;
		while(spfa())
		{
			int temp=0x3f3f3f3f;
			for(int i=end;i!=start;i=pre[i]) {
				temp=Math.min(temp, es[belong_e[i]].w);
			}
			cost+=dis[end]*temp;
			minflow+=temp;
			for(int i=end; i!=start; i=pre[i]) {
				es[belong_e[i]].w-=temp;
				es[belong_e[i] ^1 ].w+=temp;
			}
		}
		return -cost;
	}

	public static void main(String[] args)
	{
		System.out.println(wk(a,b));
	}
}

T3

题目大意:求 m a x ( m i n ( a [ i ] , a [ j ] ) ∗ ( j − i ) ) max( min(a[i],a[j])*(j-i) ) max(min(a[i],a[j])(ji)) ( j > i ) (j>i) (j>i)
数据范围未知

二重循环可过?

import java.io.*;
import java.util.*;

public class zbr01
{
	public static int mn[][]=new int[1005][1005];
	public static int mx[][]=new int[1005][1005];
	public static int a[]=new int[1000005];
	
	public static int wk(int[] x)
	{
		int n=x.length;
		for(int i=0;i<n;i++) a[i+1]=x[i];
		int ans=0;
		for(int i=1;i<=n;i++)
		{
			for(int j=i;j<=n;j++)
			{
				int tmp=Math.min(a[i],a[j]);
				tmp*=(j-i);
				ans=Math.max(ans,tmp);
			}
		}
		
		return ans;
	}
	
	public static void main(String args[])
	{
		Scanner S=new Scanner(System.in);
		int []a={1,8,6,2,5,4,8,3,7};
		System.out.println(wk(a));
	}
}

T2/T1

( 咕 咕 咕 ) (咕咕咕)

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值