• shell练习--PAT题目1003:我要通过!(成功案例)


    “答案正确”是自动判题系统给出的最令人欢喜的回复。本题属于 PAT 的“答案正确”大派送 —— 只要读入的字符串满足下列条件,系统就输出“答案正确”,否则输出“答案错误”。

    得到“答案正确”的条件是:

    1. 字符串中必须仅有 P、 A、 T这三种字符,不可以包含其它字符;
    2. 任意形如 xPATx 的字符串都可以获得“答案正确”,其中 x 或者是空字符串,或者是仅由字母 A 组成的字符串;
    3. 如果 aPbTc 是正确的,那么 aPbATca 也是正确的,其中 a、 b、 c 均或者是空字符串,或者是仅由字母 A 组成的字符串。

    现在就请你为 PAT 写一个自动裁判程序,判定哪些字符串是可以获得“答案正确”的。

    输入格式:

    每个测试输入包含 1 个测试用例。第 1 行给出一个正整数 n (<),是需要检测的字符串个数。接下来每个字符串占一行,字符串长度不超过 100,且不包含空格。

    输出格式:

    每个字符串的检测结果占一行,如果该字符串可以获得“答案正确”,则输出 YES,否则输出 NO

    具体代码如下:

    #/bin/bash
    
    func_pipei(){
    	p_index=$(expr index "$line" "P")  #返回首次匹配到P
    	t_index=$(expr index "$line" "T")  #返回首次匹配到T
    	#${string:position:length}      在$string中, 从位置$position开始提取长度为$length的子串
    	strA=${line:0:$(expr $p_index - 1)}
    	strB=${line:p_index:$(expr $t_index - $p_index - 1)}
    	strC=${line:$t_index}
    	if [[ $(expr ${#strA} * ${#strB} ) -eq ${#strC} ]] #此处判断 a*b=c
    	then
    	        echo "YES"
    	else
    	        echo "NO"
    	fi
    }
    
    read -p"请输入字符串内容,多行字符串以空格分隔:" n
    for line in $(echo $n)
    do
            length=$(echo $line |awk '{print length($0)}')
            length_p=$(echo ${line//P/} |awk '{print length($0)}')
            length_t=$(echo ${line//T/} |awk '{print length($0)}')
            #判断是否存在A P T 三个字符,排除替换语句报错
            if [[ $(expr index $line "APT") -ne 0 ]]  
            then
                    #条件1,把A P T字符替换为空,判断是否为空
                    #条件2,3 判断P T 字符的个数是否为1
                    #条件4 判断A的字符个数是否不为0
                    #条件5 判断P的位置在T的前面
                    if [[ -z ${line//[APT]/} ]] && [[ $(expr $length - $length_p) -eq 1 ]] && [[ $(expr $length - $length_t) -eq 1 ]] && [[ $(expr index $line "A" ) != 0 ]] || [[ $length_t -gt $length_p ]]
                    then
                            func_pipei
                    else
                            echo "NO"
                    fi
            else 
                    echo "NO"
            fi
    done

    em...虽然示例数据获得了结果,但是没有通过,不过也算练习了。

    这里,我忽略了一个问题,就是题目中,首行是数字,是对下方字符串的统计,这里没有进行处理。所以获取不到正确的结果。也就是说for循环需要跳过第一个字符。

    wyf349@ubuntu:~/user/study_shell$ bash  1003_rightanswer.err1.sh   
    请输入字符串内容,多行字符串以空格分隔:PAT PAAT AAPATAA AAPAATAAAA xPATx PT Whatever APAAATAA
    YES
    YES
    YES
    YES
    NO
    NO
    NO
    NO
    

      问题:

        1.主要是字符串的相关操作,获取字符串的位置 expr index命令,字符串长度  ${#var} ,字符串替换${string//substring/replacement};

        2.关于shell的多行脚本读写输入问题,目前只想到了通过分隔符来读取多个字符串,但是类似示例中的,按行区分,暂时没想到解决办法。

    #!/bin/bash
    read -d  / -p "分隔行读取:" n  #-d参数可以直接指定结束符
    for i in $(echo $n)
    do
            echo "--------------"
            echo "$i"
    done
    解答:关于分行读取的问题,可以将文本看作输入一行处理一行,没必要看做输入多行,再按照多行处理

      

     -------------------------------

    以下是正确答案:

    #/bin/bash
    
    func_pipei(){
        p_index=$(expr index "$line" "P")  #返回首次匹配到P
        t_index=$(expr index "$line" "T")  #返回首次匹配到T
        strA=${line:0:$(expr $p_index - 1)}
        strB=${line:p_index:$(expr $t_index - $p_index - 1)}
        strC=${line:$t_index}
        if [[ $(expr ${#strA} * ${#strB} ) -eq ${#strC} ]] #此处判断 a*b=c
        then
                echo "YES"
        else
                echo "NO"
        fi
    }
    
    read num
    for ((i >=0;i<$num;i++))
    do
        read  line
        length=$(echo $line |awk '{print length($0)}')
        length_p=$(echo ${line//P/} |awk '{print length($0)}')
        length_t=$(echo ${line//T/} |awk '{print length($0)}')
            #判断是否存在A P T 三个字符,排除替换语句报错
        if [[ $(expr index $line "T") -gt $(expr index $line "P") ]] 
        then
                if [[ -z ${line//[APT]/} ]] && [[ $(expr $length - $length_p) -eq 1 ]] && [[ $(expr $length - $length_t) -eq 1 ]] && [[ $(expr index $line "A" ) != 0 ]]
                then
                        func_pipei
                else
                        echo "NO"
                fi
        else
                echo "NO"
        fi
        
    done
    

      此题可分析出几个测试点:

        1.正确的输入输出问题;

        2.字符‘P A T’是否满足表达式 :以PT作为分割线,第1段A个数 *  第2段个数 = 第3段的个数问题

        3.满足条件 位置P 再位置T的左边;

        4.满足条件,P和T的个数为1个;

        5.满足条件,A的个数要大于0个。

     补充调整为python的版本:

    #!/usr/bin/python3
    #-*- coding:utf-8 -*-
    
    def check_PAT(line):  #判断位置与公式
        position_P=line.find('P')
        position_T=line.find('T')
        #根据位置关系计算字符数的计算公式是否成立
        strA=line.split('P')[0]
        strB=line[position_P + 1:position_T]
        strC=line.split('T')[-1]       
    
        if (len(strA) * len(strB) == len(strC) ) and (position_T > position_P):
            return True
        else:
            return False
    
    def main():
        num=int(input())
        for i in range(num):
            line=str(input())
            #判断是否为PAT三个字母构成
            line_c=line.replace('P','').replace('T','').replace('A','')
            if len(line_c)==0:
            #判断P T的个数
                if  line.count('P')==1 and line.count('T')==1 and line.count('A') > 0 and check_PAT(line):
                    print('YES')
                else:
                    print('NO')
            else:
                print('NO')
    
    if __name__ == '__main__':
        main()
    

      

  • 相关阅读:
    Access, SQL Server, and Oracle数据类型的对应关系
    [转]SQL Server 2005 从差异备份还原数据库
    疲惫
    关于在cmd命令里的路径包含空格的问题
    导Excel时的科学计数法问题
    [转]SQL SERVER 2005 备份与恢复简介
    [转]用C#创建Windows Service
    [转] vb.net option
    [转]sql server profiler only TrueType fonts are supported. this is not a truetype font
    进程、线程、协程之概念理解[转帖]
  • 原文地址:https://www.cnblogs.com/wyf-349/p/11233308.html
Copyright © 2020-2023  润新知