28
\$\begingroup\$

This is taken from this question (with permission ofcourse). I'll quote:

Create a function which takes a string, and it should return true or false based on whether the input consists of only a repeated character sequence. The length of given string is always greater than 1 and the character sequence must have at least one repetition.

Some examples:

'aa' //true
'aaa' //true
'abcabcabc' //true
'aba' //false
'ababa' //false
'weqweqweqweqweqw' // false

Specifically, the check for a string strictly composed of repeating substrings (Update) can output any true or false representation, but no error output please. Strictly alphhanumeric strings. Otherwise standard code golf rules. This is Code Golf, so shortest answer in bytes for each language wins.

asked Apr 24, 2019 at 14:55
\$\endgroup\$
2
  • 4
    \$\begingroup\$ Hm, I was going to close this challenge as a dupe of that one, but I noticed that the other one scores on character count. So maybe we should close the other one (it also has an accepted answer) as a dupe of this one instead. \$\endgroup\$ Commented Apr 24, 2019 at 15:14
  • \$\begingroup\$ Let us continue this discussion in chat. \$\endgroup\$ Commented Apr 24, 2019 at 15:36

43 Answers 43

1
2
20
\$\begingroup\$

JavaScript (ES6), 22 bytes

Returns a Boolean value.

s=>/^(.*)1円+$/.test(s)

Try it online!


Without a regular expression, (削除) 33 (削除ここまで) 29 bytes

Returns either null (falsy) or an object (truthy).

s=>(s+s).slice(1,-1).match(s)

Try it online!

NB: Technically, \$s\$ is converted to a regular expression for match(), so the above title is a lie.

answered Apr 24, 2019 at 15:01
\$\endgroup\$
13
\$\begingroup\$

Brachylog,(削除) 4 (削除ここまで)3 bytes

ġ=Ṁ

Try it online!

Explanation

ġ=Ṁ Implicit input, say "abcabc"
ġ Split into chunks of equal lengths (except maybe the last one): ["abc","abc"]
 = Apply the constraint that all of the chunks are equal,
 Ṁ and that there are multiple of them.

The program prints true. if the constraints can be satisfied, and false. if not.

Unrelated String
23.9k3 gold badges37 silver badges61 bronze badges
answered Apr 24, 2019 at 19:28
\$\endgroup\$
9
  • \$\begingroup\$ I was just struggling through trying to get something like ~j↙ or =Ṁc working before I noticed you posted this an hour ago \$\endgroup\$ Commented Apr 24, 2019 at 21:27
  • 4
    \$\begingroup\$ Oh, yeah, this could be one byte shorter: ġ=Ṁ \$\endgroup\$ Commented Apr 24, 2019 at 21:36
  • \$\begingroup\$ ( is a variable constrained to be a list of two or more elements) \$\endgroup\$ Commented Apr 24, 2019 at 21:39
  • 1
    \$\begingroup\$ @UnrelatedString Great, thanks! I didn't think to check the variables wiki page. \$\endgroup\$ Commented Apr 25, 2019 at 9:12
  • 1
    \$\begingroup\$ A lot of great answers, and the LUA answer has a special place in my heart. Arnauld's answer is particularly sweet since the original question that I based this on (not the dupe) is actually tagged Javascript. Mainly selecting this one just because it does appear to be the overall shortest for all languages and, as this is my first question, I get a badge. \$\endgroup\$ Commented May 1, 2019 at 6:40
10
\$\begingroup\$

grep, 19

grep -qxE '(.+)1円+'

Test

while read; do 
 <<<"$REPLY" grep -qxE '(.+)1円+' && t="true" || t="false"
 echo "$REPLY: $t"
done < infile 

Output:

aa: true
aaa: true
abcabcabc: true
aba: false
ababa: false
weqweqweqweqweqw: false
answered Apr 24, 2019 at 15:56
\$\endgroup\$
9
\$\begingroup\$

Japt, 6 bytes

2é ¤øU

Saved one byte thanks to @Shaggy

Try it online!

 Implicit input, stored in variable 'U'
2 U+U, "abcabc" -> "abcabcabcabc"
 é Rotate 1 char to the right "abcabcabcabc" -> "cabcabcabcab"
 ¤ Remove first two chars, "cabcabcabcab" -> "bcabcabcab"
 øU Check if U is in the above
answered Apr 24, 2019 at 16:14
\$\endgroup\$
1
  • \$\begingroup\$ Nice one :) You can replace the p<space> with ² to save a byte. \$\endgroup\$ Commented Apr 24, 2019 at 16:25
9
\$\begingroup\$

Java, (削除) 25 (削除ここまで) 24 bytes

-1 byte thanks to Olivier Grégoire!
Boring regex answer

s->s.matches("(.+)\1円+")

Try it online!

(削除) It's just 1 byte longer than the python answer aaaaa (削除ここまで) I'm tied now :)

answered Apr 24, 2019 at 19:49
\$\endgroup\$
2
  • 3
    \$\begingroup\$ You can remove the final $ as the matches method is an exact match, not a substring match by default. \$\endgroup\$ Commented Apr 24, 2019 at 23:18
  • \$\begingroup\$ I forgot matches adds its own $ to the regex. Thanks! \$\endgroup\$ Commented Apr 24, 2019 at 23:34
7
\$\begingroup\$

Excel, 26 bytes

=FIND(A1,A1&A1,2)<=LEN(A1)

Inputs from A1, outputs to whatever cell you put this formula.

answered Apr 24, 2019 at 16:51
\$\endgroup\$
4
  • \$\begingroup\$ You could save 4 bytes if you defined a single-letter range name (e.g. A) and set that as your input. \$\endgroup\$ Commented Apr 24, 2019 at 18:54
  • \$\begingroup\$ @i_saw_drones - I think that is disallowed by standard I/O rules: here's a link to the meta answer that would apply to that method; it's currently at -36 votes. \$\endgroup\$ Commented Apr 24, 2019 at 20:17
  • \$\begingroup\$ Apologies I hadn't seen that post, although thinking about it, isn't A1 also a "variable" since it contains the input value? :) \$\endgroup\$ Commented Apr 24, 2019 at 23:05
  • 1
    \$\begingroup\$ I would feel that way if I were doing anything special with the fact that it's A1 specifically, like if I relied somehow on its ROW(_) being 1. As is, though, it's just the most natural way of providing an Excel function with an arbitrary input. \$\endgroup\$ Commented Apr 25, 2019 at 17:03
7
\$\begingroup\$

R, 28 bytes

grepl("(.+)\1円+$",scan(,''))

Try it online!

Simple Regex version. R is (sometimes) very similar to Python, so this is similar to TFeld's Python 2 regex answer, albeit shorter!

Question (if anyone knows the answer)

I am still confused why this works, as the substring can be any length and will always work, and still works when I add a letter to the front of a valid string, like "cABABABABAB". If I personally read the regex, I see (.+), which captures any group of any length. And then \1円+$ which repeats the captured group any number of times until the end.

So why doesn't it capture just "AB" and find that it is repeated until the end of the string, especially since there is no restriction specified as to where the substring can start?

answered Apr 24, 2019 at 19:46
\$\endgroup\$
2
  • 1
    \$\begingroup\$ Interesting, this seems to be a bug in R's regex engine. Adding the option perl=TRUE makes it match cABABAB, as you'd expect. Running grep -E '(.*)1円+$' in bash also matches cABABAB, even though grep -E uses ERE, the same regex flavor R is supposed to support. \$\endgroup\$ Commented Apr 25, 2019 at 15:43
  • 2
    \$\begingroup\$ My guess is that this is an incorrectly applied optimization. Changing .+ at the start of a pattern to ^.+ is an important optimization, but if the .+ is inside capturing parens it stops being valid. \$\endgroup\$ Commented Apr 25, 2019 at 15:50
6
\$\begingroup\$

Retina 0.8.2, 9 bytes

^(.+)1円+$

Try it online! Link includes test cases.

answered Apr 24, 2019 at 15:13
\$\endgroup\$
6
\$\begingroup\$

Jelly, (削除) 5 (削除ここまで) 4 bytes

I see now that the optimal way is to follow xnor's method!

Ḋ;Ṗw

A monadic Link that accepts a list of characters and outputs an integer - the shortest possible length of a repeating slice or zero if none exists. Note that zero is falsey while non-zero numbers are truthy in Jelly.

Try it online!

How?

Ḋ;Ṗw - Link: list of characters, S e.g. "abcabcabc" or "abababa"
Ḋ - dequeue S "bcabcabc" "bababa"
 Ṗ - pop from S "abcabcab" "ababab"
 ; - concatenate "bcabcabcabcabcab" "bababaababab"
 w - first index of sublist 3 ^---here! 0 (not found)
answered Apr 24, 2019 at 16:18
\$\endgroup\$
0
4
\$\begingroup\$

Perl 5 -p, 14 bytes

$_=/^(.*)1円+$/

Try it online!

answered Apr 24, 2019 at 16:32
\$\endgroup\$
4
\$\begingroup\$

Pyke, 4 bytes

+tO{

Try it here!

+ - input+input
 t - ^[1:]
 O - ^[:-1]
 { - input in ^
answered Apr 24, 2019 at 17:44
\$\endgroup\$
4
\$\begingroup\$

Python 2, 24 bytes

lambda s:s in(s*2)[1:-1]

Try it online!

Shamelessly stolen from xnor's answer to the original question.


More intuitive version:

Python 2, (削除) 59 (削除ここまで) (削除) 55 (削除ここまで) 53 bytes

lambda s:s in[len(s)/i*s[:i]for i in range(1,len(s))]

Try it online!


Boring regex version:

Python 2, 44 bytes

lambda s:re.match(r'(.+)1円+$',s)>0
import re

Try it online!

answered Apr 24, 2019 at 16:28
\$\endgroup\$
4
\$\begingroup\$

J, (削除) 26 25 15 (削除ここまで) 14 bytes

Using xnor method

+./@E.}:@}.@,~

Try it online!

original (two different approaches)

J, 25 bytes

1<1#.(#%#\)=<\+/@E.&:>"{]

Try it online!

J, 26 bytes

1<1#.-@#\([:(-:##{.)<\)"{]

Try it online!

answered Apr 24, 2019 at 17:13
\$\endgroup\$
4
\$\begingroup\$

Wolfram Language (Mathematica), (削除) 24 (削除ここまで) 23 bytes

StringMatchQ[x__..~~x_]

Try it online!

StringMatchQ[ (*a function that checks if its input (string) matches:*)
 x__.. (*a sequence of one or more characters, repeated one or more times*)
 ~~x_] (*and one more time*)
answered Apr 25, 2019 at 3:48
\$\endgroup\$
4
\$\begingroup\$

GNU Bash, 28 bytes

[[ ${1:1}${1::-1} == *1ドル* ]]

Save the above script into a file, and run bash file.sh "string to test".

Exit code 0 is truthy and non-zero is falsy. (as all Unix shells would interpret)

answered Apr 27, 2019 at 12:50
\$\endgroup\$
3
\$\begingroup\$

05AB1E, 5 bytes

xnor's method from the previous question appears to be optimal in 05AB1E as well.

«¦ ̈så

Try it online! or as a Test Suite

Explanation

« # append input to input
 ¦ ̈ # remove the first and last character of the resulting string
 så # check if the input is in this string
answered Apr 25, 2019 at 6:33
\$\endgroup\$
2
  • 1
    \$\begingroup\$ Of course.. I was about to make a 05AB1E answer when I saw none were there. Colleague asked me some questions and talked about his vacation. I look back at the screen: one new answer. Tada, beat again XD \$\endgroup\$ Commented Apr 25, 2019 at 6:55
  • \$\begingroup\$ @KevinCruijssen: That's typical. Has happened to me a bunch of times as well ;) \$\endgroup\$ Commented Apr 25, 2019 at 6:56
3
\$\begingroup\$

PowerShell, (削除) 23 (削除ここまで) 24 bytes

+1 byte to fully match rules

"$args"-match"^(.+)1円+$"

Try it online!

Pretty boring. Based on the other Regex answers. Luckily PowerShell doesn't use \ as an escape character!

answered Apr 24, 2019 at 20:13
\$\endgroup\$
2
  • \$\begingroup\$ it returns true for aabcabc \$\endgroup\$ Commented Apr 25, 2019 at 3:41
  • 1
    \$\begingroup\$ @mazzy just fixed! \$\endgroup\$ Commented Apr 25, 2019 at 11:38
3
\$\begingroup\$

C# (Visual C# Interactive Compiler), 70 bytes

xnor's shameless adaptation (46 bytes)

s=>(s+s).Substring(1,s.Length*2-2).Contains(s)

My non Regex Solution:

s=>s.Select((x,y)=>y).Count(z=>s.Replace(s.Substring(0,z+1),"")=="")>1

Explanation:

Replace every possible substring that starts at index 0 with an empty string. If the result is an empty string, the string is entirely made of that substring. Since this includes evaluating the entire string with itself, the amount of expected results must be greater than 1.

Example: abcabc

Possible substrings starting at index 0:

'a', 'ab', 'abc', 'abca', 'abcab', 'abcabc'

If we replace them with empty strings

Substring Result
'a' => 'bcbc'
'ab' => 'cc'
'abc' => ''
'abca' => 'bc'
'abcab' => 'c'
'abcabc' => ''

Since there is a substring other than 'abcabc' that returns an empty string, the string is entirely made of another substring ('abc')

Try it online!

answered Apr 25, 2019 at 9:09
\$\endgroup\$
3
\$\begingroup\$

Python 3, (削除) 62 (削除ここまで) (削除) 60 (削除ここまで) (削除) 56 (削除ここまで) 54 bytes

-4 bytes thanx to ArBo

lambda s:s in(len(s)//l*s[:l]for l in range(1,len(s)))
  1. Iterate over all possible prefixes in the string.
  2. Try to build the string out of the prefix.
  3. Return whether this succeeds with any prefix at all.

Try it online!

answered Apr 25, 2019 at 19:11
\$\endgroup\$
4
  • 1
    \$\begingroup\$ Nice answer! The f= can be dropped; anonymous functions are generally allowed. Also, by switching to Python 2 and checking membership of a list instead of the any construct, you can get to 55 bytes \$\endgroup\$ Commented Apr 26, 2019 at 8:09
  • 1
    \$\begingroup\$ Nice catch with the list membership, thanx! I won't switch to Python 2, as this is like switching the language, which is obviously not the point here ;) Also, is there a convenient way to test an anonymous function in TIO, keeping the byte-count? \$\endgroup\$ Commented Apr 26, 2019 at 14:13
  • 1
    \$\begingroup\$ @movatica In the header, put `f = ` (\ is the line continuation character in python) \$\endgroup\$ Commented Apr 26, 2019 at 15:33
  • \$\begingroup\$ Annoyingly, \ is also an escape character. Here, without code formatting, is what you should put in the header: f = \ \$\endgroup\$ Commented Apr 26, 2019 at 15:35
3
\$\begingroup\$

Ruby -n, (削除) 16 (削除ここまで) 13 bytes

-3 bytes thanks to south

For completeness, more or less the same as the other regex answers. Prints 0 for truthy, nil for falsey.

p~/^(.+)1円+$/

Attempt This Online!

answered Nov 14, 2022 at 1:15
\$\endgroup\$
2
  • 1
    \$\begingroup\$ 13 bytes p~/^(.+)1円+$/. ~ is equivalent to rxp =~ $_. \$\endgroup\$ Commented Nov 14, 2022 at 4:23
  • \$\begingroup\$ @south Oops! Good catch. \$\endgroup\$ Commented Nov 14, 2022 at 13:55
2
\$\begingroup\$

Japt, 10 bytes

Returns a positive number if truthy and 0 if falsey. If you want a bool output just add flag

å+ k@rXÃÊÉ

å+ k@rXÃÊÉ Full program. Implicit input U.
 e.g: U = "abcabcabc"
å+ Take all prefixes 
 U = ["a","ab","abc","abca","abcab","abcabc","abcabca","abcabcab","abcabcabc"]
 k@ Filter U by:
 rXÃ Values that return false (empty string)
 when replacing each prefix in U
 e.g: ["bcbcbc","ccc","","bcabc","cabc","abc","bc","c",""]
 take ↑ and ↑
 U = ["abc","abcabcabc"]
 ÊÉ Get U length and subtract 1. Then return the result

Try it online!

answered Apr 24, 2019 at 15:18
\$\endgroup\$
0
2
\$\begingroup\$

Husk, 6 bytes

Ṡ€ȯhtD

Try it online!

I feel like this is one byte more than optimal, but I couldn't find an arrangement that made the explicit composition ȯ unnecessary.

Explanation

Ṡ€ Find the argument in the result of applying the following function to the argument
 ȯhtD Duplicate the argument, then remove the first and last elements.
answered Apr 24, 2019 at 17:16
\$\endgroup\$
2
  • 2
    \$\begingroup\$ €htD¹ avoids the ȯ. \$\endgroup\$ Commented Apr 24, 2019 at 19:30
  • \$\begingroup\$ That's fantastic! I had thought about λ€htD¹ but I didn't realize that lambdas would be added implicitly \$\endgroup\$ Commented Apr 24, 2019 at 20:10
2
\$\begingroup\$

Mathematica 11.x, 74 bytes

{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&

where, throughout, # represents the input string, and

StringCases[#,<pattern>]

finds substrings of the input string matching the pattern

StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="") 

This pattern requires matches, x, must start at the start of the string and must satisfy the condition that (1) the match is not the whole input string and (2) if we replace occurrences of the match in the input string with the empty string we obtain the empty string. Finally, comparing the list of matches to the empty list,

{}!=

is True if the list of matches is nonempty and False if the list of matches is empty.

Test cases:

{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&["aa"]
(* True *)
{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&["aaa"]
(* True *)
{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&["abcabc"]
(* True *)

and

{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&["aba"]
(* False *)
{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&["ababa"]
(* False *)
{}!=StringCases[#,StartOfString~~x__/;(x!=#&&StringReplace[#,x->""]=="")]&["weqweqweqweqweqw"]
(* False *)
answered Apr 24, 2019 at 17:40
\$\endgroup\$
2
\$\begingroup\$

Python 3, 84 bytes

import textwrap
lambda s:any(len(set(textwrap.wrap(s,l)))<2 for l in range(1,len(s)))

Uses textwrap.wrap (thanks to this answer) to split the string into pieces of length n to test each possible length of repeating substring. The split pieces are then compared to each other by adding them to a set. If all of the pieces are equal, and the set is of length 1, then the string must be a repeating string. I used <2 instead of ==1 because it saves a byte, and the length of the input string was guaranteed to be greater than zero.

If there is no n for which repeating substrings of length n make up the entire string, then return false for the whole function.

answered Apr 24, 2019 at 20:20
\$\endgroup\$
2
\$\begingroup\$

Clean, 73 bytes

Doesn't use regex.

import StdEnv,Data.List
$s=or[isPrefixOf s(cycle t)\\t<-tl(tails s)|t>[]]

Try it online!

Defines $ :: [Char] -> Bool.
Checks if the given string is a prefix of the repetition of any sub-string taken from the end.

answered Apr 26, 2019 at 0:43
\$\endgroup\$
2
\$\begingroup\$

Zsh, 19 bytes

Outputs via error code:

[[ 1ドル1ドル = ?*1ドル*? ]]

Try it online!

However, the challenge did specify "output on stdout":

Zsh, 22 bytes

Outputs an empty string if true, and a non-empty string for false:

<<<${${:-1ドル1ドル}/?*1ドル*?}

If the output must be fixed:

Zsh, 25 bytes

Outputs 0 for true, 1 for false

[[ 1ドル1ドル = ?*1ドル*? ]]
<<<$?

Try it online!


The heart of these solutions is ?*1ドル*?, which is a glob which matches the string in question surrounded by at least one character on each side: * is a string of any length, and ? is a single character. This is an alternate version of xnor's solution, but using globs to match extra characters rather than removing leading and trailing characters.

answered Apr 26, 2019 at 11:56
\$\endgroup\$
2
\$\begingroup\$

C++ (gcc), 36 bytes

#define f(x)(x+x).find(x,1)<x.size()

Try it online!

Another port of xnor's solution. Uses a macro to expand the argument into the expression. The argument is assumed to be of type std::string.

answered Apr 25, 2019 at 23:42
\$\endgroup\$
2
\$\begingroup\$

K (ngn/k), 11 bytes

&/1_=':#'=:

Try it online!

Thanks to PyGamer0 for the "check if all the values of a list are equal" code snippet!

Return 1 for true and 0 for false

Explanations:

&/1_=':#'=: Main function. Takes implicit input with : on the right-most
 = Group the characters into a dictionary
 #' Get the length in the values of each of the keys
 (The values are the occurence positions of the characters in the string)
 : Apply everything on the left to the right
 =' Check if the values of the values in the dictionary are equal
 1_ Weed out the first one (as we know it will always be false)
&/ Get the minimum value
answered Nov 13, 2022 at 13:37
\$\endgroup\$
2
\$\begingroup\$

Vyxal r, (削除) 5 (削除ここまで) 4 bytes

dḢṪc

-1 byte by porting xnor's answer from the other question

Try it online!

Concatenates the string to itself, removes the first and last characters, then checks if the input string is in that.

answered Nov 12, 2022 at 22:42
\$\endgroup\$
1
\$\begingroup\$

QlikView Variable, 27 bytes

This should be defined as a variable, which then allows you to pass parameters, e.g. 1ドル as your input value.

It returns 0 or -1 (equivalent to QlikView's TRUE() function).

=substringcount(1ドル&1,ドル1ドル)>2
answered Apr 24, 2019 at 19:19
\$\endgroup\$
1
2

Your Answer

Draft saved
Draft discarded

Sign up or log in

Sign up using Google
Sign up using Email and Password

Post as a guest

Required, but never shown

Post as a guest

Required, but never shown

By clicking "Post Your Answer", you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.