Skip to content

Commit

Permalink
AK: Made Strings reversible
Browse files Browse the repository at this point in the history
`AK::String` can now be reversed via AK::String::reverse(). This makes
life a lot easier for functions like `itoa()`, where the output
ends up being backwards. Very much not like the normal STL
(which requires an `std::reverse` object) way of doing things.

A call to reverse returns a new `AK::String` so as to not upset any
of the possible references to the same `StringImpl` shared between
Strings.
  • Loading branch information
Quaker762 authored and awesomekling committed Sep 13, 2019
1 parent 093961d commit 26e81ad
Show file tree
Hide file tree
Showing 4 changed files with 28 additions and 4 deletions.
10 changes: 8 additions & 2 deletions AK/String.h
100755 → 100644
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,13 @@ class String {
return m_impl->to_uppercase();
}

String reversed() const
{
if (!m_impl)
return String();
return m_impl->reversed();
}

Vector<String> split_limit(char separator, int limit) const;
Vector<String> split(char separator) const;
String substring(int start, int length) const;
Expand Down Expand Up @@ -227,7 +234,6 @@ struct Traits<String> : public GenericTraits<String> {
struct CaseInsensitiveStringTraits : public AK::Traits<String> {
static unsigned hash(const String& s) { return s.impl() ? s.to_lowercase().impl()->hash() : 0; }
static bool equals(const String& a, const String& b) { return a.to_lowercase() == b.to_lowercase(); }

};

inline bool operator<(const char* characters, const String& string)
Expand Down Expand Up @@ -264,5 +270,5 @@ inline bool operator<=(const char* characters, const String& string)

}

using AK::String;
using AK::CaseInsensitiveStringTraits;
using AK::String;
17 changes: 16 additions & 1 deletion AK/StringImpl.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
#include "kmalloc.h"

#ifndef __serenity__
#include <new>
# include <new>
#endif

//#define DEBUG_STRINGIMPL
Expand Down Expand Up @@ -172,4 +172,19 @@ void StringImpl::compute_hash() const
m_has_hash = true;
}

NonnullRefPtr<StringImpl> StringImpl::reversed() const
{
if (m_length == 0)
return the_empty_stringimpl();

char* buffer;
const char* pos = &m_inline_buffer[m_length - 1];
auto new_impl = create_uninitialized(m_length, buffer);

for (int i = 0; i < m_length; i++)
buffer[i] = *pos--;

return new_impl;
}

}
4 changes: 3 additions & 1 deletion AK/StringImpl.h
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#pragma once

#include <AK/RefPtr.h>
#include <AK/RefCounted.h>
#include <AK/RefPtr.h>
#include <AK/Types.h>
#include <AK/kmalloc.h>

Expand Down Expand Up @@ -44,6 +44,8 @@ class StringImpl : public RefCounted<StringImpl> {
return m_hash;
}

NonnullRefPtr<StringImpl> reversed() const;

private:
enum ConstructTheEmptyStringImplTag {
ConstructTheEmptyStringImpl
Expand Down
1 change: 1 addition & 0 deletions AK/Tests/TestString.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ TEST_CASE(compare)
EXPECT(!("a" >= String("b")));
EXPECT("a" <= String("a"));
EXPECT(!("b" <= String("a")));
EXPECT(!strcmp(test_string.reversed().characters(), "FEDCBA"));
}

TEST_CASE(index_access)
Expand Down

0 comments on commit 26e81ad

Please sign in to comment.