Определить прямоугольный текст с помощью прямоугольного кода

19

При наличии строки печатного текста ASCII (включая символы новой строки и пробелы), который содержит хотя бы один символ, который не является ни новой строкой, ни пробелом, выведите истинное значение, если строка является прямоугольной, и значение Falsey в противном случае. Кроме того, исходный код вашего решения должен быть прямоугольным .

Строка является прямоугольной, если она удовлетворяет всем следующим условиям:

  1. Первая и последняя строки не содержат пробелов.
  2. Первый и последний символ каждой строки не является пробелом.
  3. Все строки имеют одинаковое количество символов.

Например, следующий текст является прямоугольным:

abcd
e fg
hijk

Этот текст, однако, не является прямоугольным (требование № 3):

1234
567
8900

Тестовые случаи

Truthy:

sdghajksfg
asdf
jkl;
qwerty
u i op
zxcvbn
1234
5  6
7890
abcd
e fg
hijk

Falsey:

a b c
123
456
7 9
12
345
qwerty
 uiop
zxcvnm
1234
567
8900

Это , поэтому выигрывает самое короткое решение в байтах.

Mego
источник
2
Относящиеся .
AdmBorkBork
9
Таким образом, однострочник без пробела является действительным представлением, верно?
Арно
1
Можем ли мы принять входные данные как массив строк, по одному на каждую строку? Или мы должны ввести одну длинную строку, которая включает разрывы строк?
BradC

Ответы:

12

C (gcc) , 127 125 124 118 байт

  • Сохранено два байта при игре r*=!e&(!t|t==c);в гольф r>>=e||t&&t-c;. (Этот гольф послужил источником вдохновения для моего недавнего C-ответа на вопрос об обратном обновлении флага .)
  • Спас байт, играя *(_-2)в гольф _[~1].
  • Сэкономили шесть байтов, играя *_++-10||(...)в гольф *_++<11?...:0и используя нулевой заполнитель ...:0(который конструктивно не используется) для c++увеличения приращения. Эти гольфы позволили некоторые дальнейшие перестановки петли.
  • Когда можно использовать несколько значений Falsey, возможно 114 байтов .
r,e,c,t;_(char*_){for(r=1,t=c=0;*_;*_++<11?r*=(t||(t=c,!e))&*_>32&_[~1]>32&t==c,c=e=0:c++)*_-32||(e=1);r>>=e||t&&t-c;}

Попробуйте онлайн!

Разметка источника, достигающая более высокого прямоугольника.

объяснение

Далее поясняется версия длиной 124 байта.

r,e,c,t;_(char*_){     // `r` is the boolean result flag, `e` a boolean flag if the current line contains
                       //  a space, `t` the first line's width, `c` the current line's current width
 for(r=1,t=c=0;*_;c++) // initialize, loop through entire string
  *_-32||              // if the current char is a space,
   (e=1),              //  the current line contains a space
  *_++-10||            // if the current char is a newline (char pointer `_` now incremented)
   (r*=(t||(t=c,!e))   // if t is not yet set, the current line is the first line; set it
                       //  to this line's length, check that no spaces where found
    &*_>32             // the next line's first char should not be a space
    &_[~1]>32          // this line's last char should not have been a space
    &t==c,c=~0,e=0);   // the line lengths should match, reset `c` and `e` to zero
                       //  (`~0 == -1`, countering the loop's increment of `c`)
 r>>=e||t&&t-c;}       // return boolean flag, check that the last line does not contain spaces,
                       //  there was either no newline or line lengths match
                       //  (here) equivalent to `r*=!e&(!t|t==c)`

Попробуйте онлайн!

Джонатан Фрех
источник
10
+1 заr,e,c,t
Магическая урна с осьминогом
4

Java 10, 214 176 169 152 144 139 байт

s->{String[]a=s.split("\n")
;int r=1,i=0,R=a.length;for
(;i<R;i++)if(i<1|i>R-2?a[i]
.contains(" "):a[i].trim( )
!=a[i])r=0;return-r<0;}////

-5 байт благодаря @Neil .

Использует String[]aвместо var a; return-r<0;вместо return r>0;; и добавил комментарий// в самом конце, чтобы не было пробелов в первой и последней строках.

Обратите внимание, что этот прямоугольник короче однострочного ввода, потому что его int r=1,...;следует заменить на int[]v{1,...};, и тогда будет использоваться все целые числа v[n](где n - индекс переменной в массиве v).

Попробуйте онлайн.

Объяснение:

s->{                        // Method with String parameter and boolean return-type
  String[]a=s.split("\n");  //  Input split by new-lines
  int r=1,                  //  Result-integer, starting at 1
      i=0,                  //  Index `i`, starting at 0
      R=a.length;           //  Amount of rows `R`
  for(;i<R;i++)             //  Loop `i` over the rows
    if(i<1                  //   If it's the first row,
       |i>R-2?              //   or the last row:
        a[i].contains(" ")  //   And the current row contains a space
       :a[i].trim()!=a[i])  //   Or either column of the current row contains a space
      r=0;                  //    Set the result `r` to 0
   return-r<0;}             //  Return whether `r` is still 1
////                        // Comment to comply to the rules of the challenge

Вот та же базовая программа с пробелами ( 128 126 байт ):

s->{var a=s.split("\n");int r=1,i=0,R=a.length;for(;i<R;i++)if(i<1|i>R-2?a[i].contains(" "):a[i].trim()!=a[i])r=0;return r>0;}

-2 байта благодаря @Neil .

Попробуйте онлайн.

Кевин Круйссен
источник
1
tio.run/…
Нейл
3

T-SQL, 237 207 байт

SELECT(SELECT(IIF(max(len(v))=min(len(v)),1,0)*IIF(SUM(len(v+'x')-len
(trim(v))-1)=0,1,0))FROM t)*(SELECT(IIF(SUM(charindex(' ',v))=0,1,0))
FROM[t]WHERE[i]IN(SELECT(min(i))FROM[t]UNION(SELECT(max(i))FROM[t])))

Выходы 1 для прямоугольного, 0 в противном случае. Мне пришлось использовать тонны лишних скобок и скобок для устранения пробелов, я уверен, что есть много возможностей для улучшения.

Пояснение :

В соответствии с нашими разрешенными вариантами ввода / вывода и пояснениями в комментариях к вопросу, ввод принимается как отдельные строки в существующей таблице t . Поскольку данные в SQL по своей природе неупорядочены, эта таблица содержит поле идентификатора «номер строки» i :

CREATE TABLE t (i INT IDENTITY(1,1), v VARCHAR(999))

В основном мой SQL выполняет 3 подзапроса, каждый из которых возвращает 0или 1основан на 3 критериях «прямоугольного» кода. Эти 3 значения умножаются вместе, возвращаясь только 1для кода, который удовлетворяет всем 3.

РЕДАКТИРОВАТЬ : объединить критерии 2 и 3 в один и тот же SELECT для экономии места

SELECT(
SELECT(IIF(max(len(v))=min(len(v)),1,0)                  --All rows same length
      *IIF(SUM(len(v+'x')-len(trim(v))-1)=0,1,0))FROM t) --no leading or trailing spaces
*(SELECT(IIF(SUM(charindex(' ',v))=0,1,0))               --No spaces at all in
FROM[t]WHERE[i]IN(SELECT(min(i))FROM[t]                  --   first row or
            UNION(SELECT(max(i))FROM[t])))               --   last row

TRIM(v)Функция поддерживается только SQL 2017 и выше. Более ранние версии потребовали бы LTRIM(RTRIM(v)), что потребовало бы перебалансировки строк.

Одно случайное замечание: LEN()функция в SQL игнорирует завершающие пробелы, поэтому LEN('foo ') = 3. Чтобы получить «истинную» длину, вы должны прикрепить символ до конца, а затем вычесть один: P

BradC
источник
3

С ++, 199 183 181 175 байтов

Эта шаблонная функция принимает строки как набор строк (которые могут быть широкими строками), переданные как пара итераторов.

#include<algorithm>//
template<class I>bool
f(I a,I b){return!~+(
*a+b[-1]).find(' ')&&
std::all_of(a,b,[&a](
auto&s){return' '+-s.
back()&&s[0]-' '&&a->
size()==s.size();});}

Спасибо пользователю Erroneous за напоминание о back()члене std::stringи за указание на то, чтоnpos+1 это ноль.

Ungolfed эквивалент

Единственная настоящая игра в гольф состоит в объединении первой и последней линий, чтобы мы могли выполнить сингл findдля пробелов в них.

#include <algorithm>
template<class It>
bool f(It a, It b)
{
    return (*a+b[-1]).find(' ') == a->npos
        && std::all_of(a, b,
                       [=](auto s) {
                           return s.back() != ' '
                               && s.front() != ' '
                               && s.size() == a->size(); });
}

Тестовая программа

#include <iostream>
#include <string>
#include <vector>
int expect(const std::vector<std::string>& v, bool expected)
{
    bool actual = f(v.begin(), v.end());
    if (actual == expected) return 0;
    std::cerr << "FAILED " << (expected ? "truthy" : "falsey") << " test\n";
    for (auto const& e: v)
        std::cerr << "  |" << e << "|\n";
    return 1;
}
int expect_true(const std::vector<std::string>& v) { return expect(v, true); }
int expect_false(const std::vector<std::string>& v) { return expect(v, false); }
int main()
{
    return
        // tests from the question
        + expect_true({"sdghajksfg"})
        + expect_true({"asdf", "jkl;",})
        + expect_true({"qwerty", "u i op", "zxcvbn",})
        + expect_true({"1234", "5  6", "7890",})
        + expect_true({"abcd", "e fg", "hijk",})
        + expect_false({"a b c",})
        + expect_false({"123", "456", "7 9",})
        + expect_false({"12", "345",})
        + expect_false({"qwerty", " uiop", "zxcvnm",})
        + expect_false({"1234", "567", "8900",})
        // extra tests for leading and trailing space
        + expect_false({"123", " 56", "789"})
        + expect_false({"123", "45 ", "789"})
        // the function source
        + expect_true({"#include<algorithm>//",
                       "template<class I>bool",
                       "f(I a,I b){return!~+(",
                       "*a+b[-1]).find(' ')&&",
                       "std::all_of(a,b,[&a](",
                       "auto&s){return' '+-s.",
                       "back()&&s[0]-' '&&a->",
                       "size()==s.size();});}",})
        ;
}
Тоби Спейт
источник
Это может быть добавлено до 183 байтов с шириной строки 22, используя .find(' ')+1==0и s.back()вместо *s.rbegin().
Ошибочный
2

Haskell , 79 байтов

g(x:r)=all((==(0<$x)).(0<$))r&&all(>='!')(x++last(x:r)++(head<$>r)++(last<$>r))

Попробуйте онлайн! Принимает ввод в виде списка строк.

Шаблон g(x:r)= ...связывает первую строку с xи (возможно, пустой) список оставшихся строк с r. Затем all((==(0<$x)).(0<$))rпроверяет, все ли строки rимеют одинаковую длину с x(Используя этот совет ).

Если нет, то соединение &&замыкается и возвращается False, в противном случае оценивается правая сторона. Там есть строка build, которая состоит из xпервой строки, last(x:r)последней строки r(или первой строки снова, если rона пуста), а (head<$>r)также первого и (last<$>r)последнего символа каждой строки. Для этой строки all(>='!')проверяется, что она не содержит пробелов (мы не можем использовать (>' ')из-за ограничения исходного кода).

Laikoni
источник
Ошибки на "\ n \ n"
Angs
@Angs Хороший улов. К счастью, ОП уточнил, что ввод contains at least one character that is neither a newline nor a space, который также позволяет отбрасывать пустой список дел.
Лайкони
О, хорошо, не заметил, что добавили
Angs
2

MATL , 13 байт

ctgF6Lt&()32>

Ввод представляет собой массив строк в формате {'abc' 'de'}.

Выходные данные - это массив, содержащий только единицы, которые являются правдивыми , или массив, содержащий по крайней мере ноль, который является ложным .

Попробуйте онлайн! Или проверьте все контрольные примеры , включая проверку на достоверность / ложность.

объяснение

c       % Implicit input. Convert to char. This concatenates the
        % strings of the input cell array as rows of a rectangular
        % char array, right-padding with spaces as needed
tg      % Duplicate, convert to logical. Gives a logical array with
        % the same size containing true in all its entries
F       % Push false
6L      % Push the array [2, j-1], where j is the imaginary unit.
        % When used as an index, this is interpreted as 2:end-1
t       % Duplicate
&(      % Assignment indexing with 4 inputs: original array, new
        % value, two indexing arrays. This writes false at the inner
        % rectangle (2:end-1)×(2:end-1) of the logical array that
        % initially only contained true. This will be used as a
        % logical index (mask) into the rectangular char array
)       % Reference indexing. This selects the border of the char
        % array. The result is a column vector of chars
32>     % Is each entry greater than 32? (ASCII code for space)
        % Implicit display
Луис Мендо
источник
11 байт: cO6Lt&(32=~ попробуйте онлайн! Просто обнуляет не-границы частей, а затем проверяет, есть ли пробелы.
sundar - Восстановить Монику
@sundar Хорошая идея! Это достаточно по-другому, опубликовать его самостоятельно
Луис Мендо
1
Нет, слишком похоже на твой ответ, особенно если я напишу его как cF6Lt&(32=~. Не стесняйтесь редактировать это, или если нет, мы можем просто оставить это в комментариях.
sundar - Восстановить Монику
1

Холст , 17 15 байтов

4[↷K;}┐){SL]∑4≡

Попробуй это здесь!

Пояснение (ASCII-Fied для моно-пространства):

4[↷K;}┐){SL]∑4=  full program; pushes the input to the stack.
4[   }           repeat 4 times
  ↷                rotate ToS clockwise. This also pads the input with spaces
   K;              take off the last line and put it below the item
      ┐          pop the remaining of the input (the center)
       )         and wrap the rest (the sides) in an array
        {  ]     map over those
         S         split on spaces - should result to one item in the array
          L        and get the length
            ∑    sum those lengths together
             4=  check if equal 4
dzaima
источник
4
Я нахожу ироничным, что эти символы UTF8 в моно-подобном шрифте создают ощущение, что в источнике много пробелов. (По крайней мере, они делают в моем браузере.)
Арно
1
@Arnauld полномасштабные символы делают это. И именно поэтому я сделал шрифт для моего переводчика, чтобы он был красивее: p
dzaima
1

Красный , 216 191 байт

func[s][d:(length?(first(s:(split(s)"^/"))))sp:
func[a][none = find a" "]b: on foreach c s[b: b
and(d = length? c )and(c/1 <>" ")and(" "<> last
c)]res:(sp(first(s)))and(sp(last(s)))and(b)res]

Попробуйте онлайн!

Я положил много других необязательных скобок в первом и последнем рядах.

Гален Иванов
источник
0

Желе , 17 байт

Ỵµ.ịЀ;ịɗẎ⁶e<L€E$

Попробуйте онлайн!

Эрик Outgolfer
источник
@JonathanFrech А, исправлено. > _>
Эрик Outgolfer
@MagicOctopusUrn А? Можете ли вы дать ссылку на вход, где это ведет себя неправильно?
Эрик Outgolfer
О, нет, ты Does not seem to enforce equal line lengthтоже вызвал меня - это все, что я говорил.
Волшебная урна осьминога
Кажется, не работает для " \n " Попробуйте онлайн!
Angs
1
@Angs Попробуйте процитировать это. Это, очевидно, разбирается как ничто, если вы скажете это так.
Эрик Outgolfer
0

Желе , 15 байт

Использует метод, разработанный Mnemonic в (в настоящее время - из-за сбоя в пограничном случае) удаленном представлении Pyth. (если это сейчас исправлено, иди дай кредит !)

ỴµL€Eȧt€⁶ZUƊ4¡⁼

Монадическая ссылка, принимающая список символов, который возвращает 1 или 0.

Попробуйте онлайн!

Как?

ỴµL€Eȧt€⁶ZUƊ4¡⁼ - Link: list of characters
Ỵ               - split at newlines (making a list of lists - the rows)
 µ              - start a new monadic chain, call that ROWS
  L€            - length of €ach row in ROWS
    E           - all equal? (an integer: 1 if so, otherwise 0)
            4¡  - repeat four times:
           Ɗ    -   last three links as a monad:
      t€⁶       -     trim spaces (⁶) from €ach row in current ROWS
         Z      -     transpose that result
          U     -     upend (reverse each new row)
     ȧ          - logical AND (0 if L€E was 0 else the result of the repeated transform)
              ⁼ - equal to X? (the integer 0 is not equal to any listy of characters)
Джонатан Аллан
источник
@Mnemonic - Jelly-fied :)
Джонатан Аллан
0

Japt , 22 байта

Неконкурентный ответ: в Japt есть известная ошибка , когда двумерные вращения массивов усекают результаты. Из-за этой ошибки приведенный ниже код работает только с квадратными входами. Если же ошибки не было, приведенный ниже код должен работать совершенно корректно.

e_ʶUÌÊéUeº4o)r_z)mx}U
e_                      // Check if every line in the input array
  ʶUÌÊ                 // has the same length as the last item.
       é               // Also,
               r_z)mx}U // check if rotating and trimming the input array
           º4o)         // four times
         Ue             // is equal to the input array.

Принимает ввод как массив строк. Использование скобок вместо пробелов делает требование к прямоугольному коду довольно простым.
Попробуй это здесь .

гнида
источник
0

Ruby 2.5+, 63 байта

->a{!a.uniq(&:size)[1]&&a.none?(/^\s|\s$/)&&!(a[0]+a[-1])[?\s]}

Принимает ввод как массив строк. Нет тестовой ссылки, так как версия на TIO (2.4) слишком старая для этой. Вместо этого вот немного более длинная (69 байт) версия для тестирования:

->a{!a.uniq(&:size)[1]&&a.none?{|l|l=~/^\s|\s$/}&&!(a[0]+a[-1])[?\s]}

Попробуйте онлайн!

Разница в том, что, поскольку 2.5 Ruby поддерживает прямую передачу шаблона Regex в all?, any?, none? методы, что экономит нам несколько байтов. Сам метод довольно понятен - мы тестируем:

  1. Если есть только 1 уникальный размер строки
  2. Если на линиях есть пробелы
  3. Если есть пробелы в первой и последней строках.
Кирилл Л.
источник
0

C (gcc) , 119 байт

Принимает ввод в виде списка (ов) из n строк.

f(s,n,m,r,p)char**s,*p;{for(r=m=n;m--;r*=strlen(*s)==strlen(s[m])&(!p||m&&m^n-1&&p!=s[m]&&p[1]))p=strchr(s[m],32);n=r;}

Попробуйте онлайн!

gastropner
источник
0

C # (.NET Core) , 145 167 байт

S[0].Length>1&&S[0].IndexOf
(" ") + S[ S.Count() - 1 ].
IndexOf(" ")<-1&Array.Find(
S,x=>x[0]==' '| x [x.Length
-1]  ==  ' '  | S[0].Length
!=x.Length)==null?11>0:0>1;

Попробуйте онлайн!

S[0].Length>1&                                    // And if the lenght of the first argument is more than 1 char
Array.Find(                                       // Find a string in an array
    S,                                            // The array which will be searched in
    x=>                                           // For x as the current string from the array
    x.Length!=S[0].Length|                        // If the string lenght match not the first argument lenght
    x[0]==' '|                                    // Or if the string begins with a spacer
    x[x.Length-1]==' '                            // Or if the string ends with a spacer
)==null&                                          // And if there was no string found which matched the conditions
S[0].IndexOf(" ")+S[S.Count()-1].IndexOf(" ")<-1  // And if the first and last string doesn't have a spacer
?                                                 // If all above is true do
1>0                                               // Return True
:                                                 // Else
0>1                                               // Return False
Хилле
источник
Без пробелов в первой строке.
FrownyFrog
@FrownyFrog S[0].IndexOf(" ")ищет пробел в первой строке и S[S.Count()-1].IndexOf(" ")ищет в последней строке. Если в первой и последней строке нет пробелов, то это -2, что соответствует истине -2 < -1.
Хилле
2
Я имею в виду проблему, ваш код имеет то же ограничение, поэтому у вас не должно быть пробелов в первой строке.
FrownyFrog
1
Ваш код должен возвращаться Trueпри передаче в вашу программу. Это дополнительное ограничение в этом вызове.
FrownyFrog