首页 > 其他分享 >单词(Play On Words)

单词(Play On Words)

时间:2022-11-28 20:03:32浏览次数:48  
标签:Play vd int 26 单词 ++ Words new isVisited


单词(Play On Words)_欧拉道路

【分析】

       首先需对欧拉道路有所了解。

       存在欧拉道路的充分条件:      

对于无向图而言,如果一个无向图是连通的,且最多只有两个奇点(顶点的度数为奇数),则一定存在欧拉道路。如果有两个奇点,则必须从其中的一个奇点出发,另一个奇点终止;如果奇点不存在,则可以从任意点出发,最终一定会回到该点(称为欧拉回路)。

对于有向图而言,最多只能有两个点的入度不等于出度,而且必须是其中一个点的出度恰好比入度大1(把它作为起点),另一个的入度比出度大1(把它作为终点)。当然,还有一个前提条件:在忽略边的方向后,图必须是连通的。

       对于该题而言,把字母看作结点,单词看成有向边,则问题有解,当且仅当图中有欧拉路径。有向图存在欧拉道路的条件有两个:底图(忽略边方向后得到的无向图)连通,且度数满足上面讨论过的条件。这里判断图连通的方法采用DFS。

用java语言编写程序,代码如下:


import java.io.BufferedInputStream;
import java.util.ArrayList;
import java.util.Scanner;

public class Main {
public static void main(String[] args) {
Scanner input = new Scanner(new BufferedInputStream(System.in));
int t = input.nextInt();
for(int i = 0; i < t; i++) {
int n = input.nextInt();
int[][] udg = new int[26][26];//无向图(用于判断图是否连通)
int[][] dg = new int[26][26];//有向图(判断顶点的度数是否符合欧拉道路的条件)
boolean[] isVisited = new boolean[26];//标记顶点是否已被访问过
//这里先赋值为true的原因是为了知道哪些顶点是图中所具有的。录入图的信息中,再将其顶点对应的isVisited值赋值为false
for(int j = 0; j < 26; j++)
isVisited[j] = true;

for(int j = 0; j < n; j++) {
String word = input.next();//边
int first = word.charAt(0) - 'a';//边的起点
int last = word.charAt(word.length() - 1) - 'a';//边的终点
udg[first][last] = udg[last][first] = 1;//无向边
dg[first][last]++;//有向边(统计边的条数,用于计算各顶点的入度与出度)
isVisited[first] = isVisited[last] = false;
}

/*for(int j = 0; j < 26; j++)
System.out.println((char)('a' + j) + " : " + isVisited[j]);*/

boolean b1 = isConnected(udg, isVisited);//判断图是否具有连通性
boolean b2 = rightDegree(dg);//判断图中的顶点度数是否符合欧拉道路的充分条件
//System.out.println(b1 + "...." + b2);

if(b1 && b2)
System.out.println("Ordering is possible.");
else
System.out.println("The door cannot be opened.");
}
}

//采用DFS判断图是否具有连通性,如果图具有连通性的话,那么DFS只进行一次,否则DFS会进行多次
public static boolean isConnected(int[][] udg, boolean[] isVisited) {
int times = 0;
for(int i = 0; i < 26; i++) {
if(!isVisited[i]) {
//System.out.println((char)('a' + i));
times++;
dfs(i, udg, isVisited);
}
}

if(times > 1)
return false;
return true;
}

public static void dfs(int u, int[][] udg, boolean[] isVisited) {
if(u < 0 || u >= 26 || isVisited[u])
return;

isVisited[u] = true;
for(int v = 0; v < 26; v++)
if(!isVisited[v] && udg[u][v] > 0)
dfs(v, udg, isVisited);
}

public static boolean rightDegree(int[][] dg) {
Vertex[] vd = new Vertex[26];
//统计各顶点的入度与出度
for(int u = 0; u < 26; u++) {
for(int v = 0; v < 26; v++) {
if(dg[u][v] > 0) {
if(vd[u] == null)
vd[u] = new Vertex();
if(vd[v] == null)
vd[v] = new Vertex();

vd[u].outDegree += dg[u][v];
vd[v].inDegree += dg[u][v];
}
}
}

/*for(int i = 0; i < 26; i++)
if(vd[i] != null)
System.out.println((char)('a' + i) + " : "
+ vd[i].outDegree + " " + vd[i].inDegree);*/
//将出度与入度不同的顶点加入到链表中
ArrayList<Vertex> vdList = new ArrayList<Vertex>();
for(int i = 0; i < 26; i++) {
if(vd[i] != null && (vd[i].inDegree != vd[i].outDegree))
vdList.add(vd[i]);
}

if(vdList.size() < 2)
return true;
else if(vdList.size() == 2) {
Vertex v1 = vdList.get(0);
Vertex v2 = vdList.get(1);
if(((v1.inDegree - v1.outDegree == 1) && (v2.outDegree - v2.inDegree == 1))
|| ((v1.outDegree - v1.inDegree == 1) && (v2.inDegree - v2.outDegree == 1)))
return true;
}
return false;
}

static class Vertex {
int v;
int outDegree;
int inDegree;
}
}


标签:Play,vd,int,26,单词,++,Words,new,isVisited
From: https://blog.51cto.com/u_15894233/5893611

相关文章

  • vue vue-video-player视频播放器
    vuevue-video-player视频播放器安装依赖npmivue-video-player@5.0.2-Dnpmivideojs-contrib-hls@5.15.0-D使用<template><div><video-playerclass="v......
  • Java: How To Count Words
    publicclassMain{publicstaticvoidmain(String[]args){Stringwords="OneTwoThreeFour";intcountWords=words.split("\\s").length;Sy......
  • 倒排单词
    输入:Iamastudent输出:studentaamI最简单的做法:#include<iostream>usingnamespacestd;intmain(){stringstr,res;while(cin>>str)res=str+''+res;c......
  • leetcode_D4_58最后一个单词的长度
    1.题目  2.解一2.1  2.2  2.3  主要思路:3种方法相差不大,思路基本一致,主要是细节处理上有所不同,都是首先去掉字符串最前面和最后面的空元素,然后从最后......
  • Patch Available for Cut or Copy displaying “insuf
    you'reexperiencingfrequent“insufficientmemory”errorswhendoingsmallcutorcopyoperationsinVisualStudio2010RTM,please​​downloadthispatch​......
  • 论文理解【RL - Exp Replay】—— 【LFIW】Experience Replay with Likelihood-free I
    摘要:经验回放(ExperienceReplay),即使用过去的经验来加速价值函数的时序差分(TD)学习,是深度强化学习的关键组成部分。对重要的经验进行优先排序或重新加权已经被证明可以提高TD......
  • 欧陆词典生词本导入不被单词生词本(Python)
    欧陆词典生词本导入不被单词生词本(Python)前言众所周知,不被单词作为背单词软件体验很不错,而且还可以用户自定义导入生词本(支持.txt格式)。但不背单词无法在其他场景提取默......
  • #yyds干货盘点# LeetCode 腾讯精选练习 50 题:反转字符串中的单词 III
    题目:给定一个字符串 s ,你需要反转字符串中每个单词的字符顺序,同时仍保留空格和单词的初始顺序。 示例1:输入:s="Let'stakeLeetCodecontest"输出:"s'teLekatedoCteeL......
  • 单词翻转
    将一句话中每个单词前后翻转#define_CRT_SECURE_NO_WARNINGS1#include<stdio.h>#include<stdlib.h>#include<string.h>intmain(){chararr[500]="0";gets(arr);i......
  • [LeetCode] 809. Expressive Words
    Sometimespeoplerepeatletterstorepresentextrafeeling.Forexample:"hello"->"heeellooo""hi"->"hiiii"Inthesestringslike "heeellooo",wehaveg......