Strings
Strings are used to store sequences of characters.
Strings are actually objects, so can be used where ever an object is expected. Strings provide the following methods:
| Method | Description | 
|---|---|
| Find:Int( subString:String,startIndex=0 ) | Finds first occurance of a sub string. Returns -1 if subString not found. | 
| FindLast:Int( subString:String,startIndex=0 ) | Finds last occurance of a sub string. Returns -1 if subString not found. | 
| Hash:ULong() | Returns the calculated hash for the string. | 
| Trim:String() | Removes leading and trailing non-printable characters from a string. | 
| Replace:String( subString:String,withString:String ) | Replaces all occurances of subString with withString. | 
| StartsWith:Int( subString:String ) | Returns true if string starts with subString. | 
| EndsWith:Int( subString:String ) | Returns true if string ends with subString. | 
| Contains:Int( subString:String ) | Returns true if string contains subString. | 
| Join:String( bits:String[] ) | Joins bits together by inserting this string between each bit. | 
| Split:String[]( separator$[] ) | Splits this string into bits delimited by separator. | 
| ToLower:String() | Converts string to lowercase. | 
| ToUpper:String() | Converts string to uppercase. | 
| ToInt:Int() | Converts string to an integer. | 
| ToUInt:UInt() | Converts string to an unsigned integer. | 
| ToLong:Long() | Converts string to a long. | 
| ToULong:ULong() | Converts string to an unsigned long. | 
| ToSizeT:Size_T() | Converts string to a size_t. | 
| ToFloat:Float() | Converts string to a float. | 
| ToDouble:Double() | Converts string to a double. | 
| ToCString:Byte Ptr() | Converts string to a null terminated sequence of 8 bit bytes. The returned memory must eventually be freed with a call to MemFree. | 
| ToWString:Short Ptr() | Converts string to a null terminated sequence of 16 bit shorts. The returned memory must eventually be freed with a call to MemFree. | 
| ToUTF8String:Byte Ptr() | Converts string to a null terminated sequence of UTF-8 characters. The returned memory must eventually be freed with a call to MemFree. | 
| ToUTF8StringBuffer:Byte Ptr(buf:Byte Ptr, length:Size_T Var) | Populates a pre-allocated buffer of length bytes with the string converted to a null terminated sequence of UTF-8 characters. On return, length contains the number of bytes copied to the buffer. | 
Strings also provide a read-only length field.
Here is an example of using string methods:
Local t:String="***** HELLO *****"
Print t.length          'return length of string: 17
Print t.Find( "HE" )        'returns index of substring "HE": 6
Print t.Replace( "*","!" )  'replaces all "*" with "!"
Print t.ToLower()       'coverts string to lowercase
Print ":".Join( ["Hello","World"] ) 'returns "Hello:World"
String also provide the following functions to help with the creation of strings:
| Function | Description | 
|---|---|
| FromInt:String( value:Int ) | Creates a string from an integer. | 
| FromUInt:String( value:UInt ) | Creates a string from an unsigned integer. | 
| FromLong:String( value:Long ) | Creates a string from a long. | 
| FromULong:String( value:ULong ) | Creates a string from an unsigned long. | 
| FromSizeT:String( value:Size_T ) | Creates a string from a size_t. | 
| FromFloat:String( value:Float ) | Creates a string from a float. | 
| FromDouble:String( value:Double) | Creates a string from a double. | 
| FromCString:String( cString:Byte Ptr ) | Creates a string from a null terminated sequence of 8 bit bytes. | 
| FromWString:String( wString:Short Ptr ) | Creates a string from a null terminated sequence of 16 bit shorts. | 
| FromUTF8String:String( utf8String:Byte Ptr ) | Creates a string from a null terminated sequence of UTF-8 characters. | 
| FromBytes:String( bytes:Byte Ptr,length ) | Creates a string from a sequence of length 8 bit bytes. | 
| FromUTF8Bytes:String( utf8String:Byte Ptr, length:Int ) | Creates a string from a sequence of length UTF-8 bytes. | 
| FromShorts:String( shorts:Short Ptr,length ) | Creates a string from a sequence of length 16 bit shorts. | 
To call a string function, use: String.Function - for example:
Print String.FromInt( 10 )
String Hashes
A String hash can be used for faster string equality tests. If two strings are equal, the Hash() method returns identical values for both strings.
However, because there is not a unique hash for each unique string value, it is possible for different strings to return the same hash.
By default, all static/const strings will be compiled with their hash value pre-calculated.
For dynamic strings, the hash is calculated on the first call to Hash(). Subsequent calls will return the cached hash value.
In string equality tests, if both strings have a calculated hash, the hash value is used. Otherwise, the standard process of comparing the contents of each string is performed, which involves comparing each character in turn until a difference is found.
In the following example, because the two static/const strings have hashes, the comparison will use the hash values rather than compare the String characters.
Local hello:String = "Hello"
Local help:String = "Help"
If hello <> help Then
    ...
End If
Multiline String Blocks
A multiline string block allows you to easily create strings spanning several lines - for example:
Local txt:String = """
  {
    "name" : "Bob"
    "distance" : 150
  }
  """
Which is the equivalent to the following string creation:
Local oldtxt:String = "{~n" + ..
    "  ~qname~q : ~qBob~q~n" + ..
    "  ~distance~q : 150~n" + ..
    "}"
As you can see, a multiline string block is more natural and less clumsy, and is more representative of the final string.
A multiline string block starts with three double-quote characters (""") followed by a new line.
It ends with a new line, followed by optional whitespace, and three more double-quote characters (""").
The following prints a multiline string of zero length:
Print """
"""
You cannot put a multiline string block on a single line, nor can the contents of the string block follow the three opening double-quotes without a new line.
The following are examples of invalid multiline strings, and produce compilation errors:
Local txt:String = """ """
Local txt2:String = """"""
Local txt3:String = """
hello"""
A multiline string can be used anywhere a normal string would typically be used, although in some places it may make the surrounding code intentions less clear:
For Local s:String = EachIn """
        A
        B
        C
        D
        E
        F
        G
        """.Split("~n")
    Print s
Next
Whitespace
Leading and trailing whitespace is removed from each line of a multiline string.
The whitespace defined on the line with the closing quotes denotes the common leading whitespace that will be removed from the rest of the string.
For example:
Print """
    <?xml version="1.0"/>
    <root>
        <line>
        </line>
    </root>
    """ ' <- whitespace from the start of this line defines the common indentation
will print:
<?xml version="1.0"/>
<root>
    <line>
    </line>
</root>
Line Endings
Line endings are normalized, meaning that all line endings are converted to the "line feed" (~n / Chr(10)) character.
This step is performed before escaped (~) characters are analyzed, allowing the use of other line ending characters within the final string.
If the line continuation character (\) appears at the end of a line, the usual line ending is skipped, and the following line is appended instead.
For example:
Print """
One
Two \
and a half
Three
"""
will print:
One
Two and a half
Three
