Description
There are three types of edits that can be performed on strings: insert a character, remove a character, or replace a character. Given two strings, write a function to check if they are one edit (or zero edits) away.
Example 1:
Input:
first = "pale"
second = "ple"
Output: True
Example 2:
Input:
first = "pales"
second = "pal"
Output: False
Solutions
Solution 1: Case Discussion + Two Pointers
We denote the lengths of strings $first$ and $second$ as $m$ and $n$, respectively, where $m \geq n$.
Next, we discuss different cases:
When $m - n > 1$, $first$ and $second$ cannot be obtained through a single edit, so we return false
.
When $m = n$, $first$ and $second$ can only be obtained through a single edit if and only if exactly one character is different.
When $m - n = 1$, $first$ and $second$ can only be obtained through a single edit if and only if $second$ is obtained by deleting one character from $first$. We can use two pointers to implement this.
The time complexity is $O(n)$, where $n$ is the length of the string. The space complexity is $O(1)$.
Python3 Java C++ Go TypeScript Rust Swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17 class Solution :
def oneEditAway ( self , first : str , second : str ) -> bool :
m , n = len ( first ), len ( second )
if m < n :
return self . oneEditAway ( second , first )
if m - n > 1 :
return False
if m == n :
return sum ( a != b for a , b in zip ( first , second )) < 2
i = j = cnt = 0
while i < m :
if j == n or ( j < n and first [ i ] != second [ j ]):
cnt += 1
else :
j += 1
i += 1
return cnt < 2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30 class Solution {
public boolean oneEditAway ( String first , String second ) {
int m = first . length (), n = second . length ();
if ( m < n ) {
return oneEditAway ( second , first );
}
if ( m - n > 1 ) {
return false ;
}
int cnt = 0 ;
if ( m == n ) {
for ( int i = 0 ; i < n ; ++ i ) {
if ( first . charAt ( i ) != second . charAt ( i )) {
if ( ++ cnt > 1 ) {
return false ;
}
}
}
return true ;
}
for ( int i = 0 , j = 0 ; i < m ; ++ i ) {
if ( j == n || ( j < n && first . charAt ( i ) != second . charAt ( j ))) {
++ cnt ;
} else {
++ j ;
}
}
return cnt < 2 ;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31 class Solution {
public :
bool oneEditAway ( std :: string first , std :: string second ) {
int m = first . length (), n = second . length ();
if ( m < n ) {
return oneEditAway ( second , first );
}
if ( m - n > 1 ) {
return false ;
}
int cnt = 0 ;
if ( m == n ) {
for ( int i = 0 ; i < n ; ++ i ) {
if ( first [ i ] != second [ i ]) {
if ( ++ cnt > 1 ) {
return false ;
}
}
}
return true ;
}
for ( int i = 0 , j = 0 ; i < m ; ++ i ) {
if ( j == n || ( j < n && first [ i ] != second [ j ])) {
++ cnt ;
} else {
++ j ;
}
}
return cnt < 2 ;
}
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28 func oneEditAway ( first string , second string ) bool {
m , n := len ( first ), len ( second )
if m < n {
return oneEditAway ( second , first )
}
if m - n > 1 {
return false
}
cnt := 0
if m == n {
for i := 0 ; i < n ; i ++ {
if first [ i ] != second [ i ] {
if cnt ++ ; cnt > 1 {
return false
}
}
}
return true
}
for i , j := 0 , 0 ; i < m ; i ++ {
if j == n || ( j < n && first [ i ] != second [ j ]) {
cnt ++
} else {
j ++
}
}
return cnt < 2
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31 function oneEditAway ( first : string , second : string ) : boolean {
let m : number = first . length ;
let n : number = second . length ;
if ( m < n ) {
return oneEditAway ( second , first );
}
if ( m - n > 1 ) {
return false ;
}
let cnt : number = 0 ;
if ( m === n ) {
for ( let i : number = 0 ; i < n ; ++ i ) {
if ( first [ i ] !== second [ i ]) {
if ( ++ cnt > 1 ) {
return false ;
}
}
}
return true ;
}
for ( let i : number = 0 , j : number = 0 ; i < m ; ++ i ) {
if ( j === n || ( j < n && first [ i ] !== second [ j ])) {
++ cnt ;
} else {
++ j ;
}
}
return cnt < 2 ;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26 impl Solution {
pub fn one_edit_away ( first : String , second : String ) -> bool {
let ( f_len , s_len ) = ( first . len (), second . len ());
let ( first , second ) = ( first . as_bytes (), second . as_bytes ());
let ( mut i , mut j ) = ( 0 , 0 );
let mut count = 0 ;
while i < f_len && j < s_len {
if first [ i ] != second [ j ] {
if count > 0 {
return false ;
}
count += 1 ;
if i + 1 < f_len && first [ i + 1 ] == second [ j ] {
i += 1 ;
} else if j + 1 < s_len && first [ i ] == second [ j + 1 ] {
j += 1 ;
}
}
i += 1 ;
j += 1 ;
}
count += f_len - i + s_len - j ;
count <= 1
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39 class Solution {
func oneEditAway ( _ first : String , _ second : String ) -> Bool {
let m = first . count , n = second . count
if m < n {
return oneEditAway ( second , first )
}
if m - n > 1 {
return false
}
var cnt = 0
var firstIndex = first . startIndex
var secondIndex = second . startIndex
if m == n {
while secondIndex != second . endIndex {
if first [ firstIndex ] != second [ secondIndex ] {
cnt += 1
if cnt > 1 {
return false
}
}
firstIndex = first . index ( after : firstIndex )
secondIndex = second . index ( after : secondIndex )
}
return true
} else {
while firstIndex != first . endIndex {
if secondIndex == second . endIndex || ( secondIndex != second . endIndex && first [ firstIndex ] != second [ secondIndex ]) {
cnt += 1
} else {
secondIndex = second . index ( after : secondIndex )
}
firstIndex = first . index ( after : firstIndex )
}
}
return cnt < 2
}
}