How do I build an array of const?

前端 未结 3 1591
庸人自扰
庸人自扰 2020-12-18 01:43

I am implementing an interpreter, and one of the functions my interpreter will support is like Delphi\'s Format. In fact, I\'m implementing my function using

相关标签:
3条回答
  • 2020-12-18 01:59

    Found this code at https://groups.google.com/forum/#!topic/borland.public.delphi.objectpascal/-xb6O0qX2zc

    procedure test(numArgs:integer; MyFormattingString:string);    
    var
      v:array of tvarrec;
      i:integer;
    begin
      setlength(v, numArgs);
      for i:=1 to numArgs do
      begin
        v[i-1].vtype:=vtpchar;
        v[i-1].vtpchar:=strnew(pchar(myDataSet.FieldByName(inttostr(i)).asstring));
      end;
      memo1.lines.add(Format(MyFormattingString,v);
      for i:=1 to numArgs do strdispose(v[i-1].vtpchar);
    end;
    

    Doesn't answer everything I have to deal with, but I think I know how to construct the array of TVarRec now.

    0 讨论(0)
  • 2020-12-18 02:17

    As you know, array of const is the same as array of TVarRec. To construct one, begin by declaring such as array, and then set the values of each of the elements, just as you would any other array.

    TVarRec is a variant record, which means it can hold many different types of values. It has a field, VType, to indicate the type of the value it holds. Of the other fields, only one has a valid value at a time. Set the VType field, and then set the corresponding value field, such as VInteger or VString.

    Beware that some of the fields are really pointers, like VVariant and VInt64. When you assign those pointer values, you'll need to make sure that whatever they point at remains accessible and valid for as long as Format needs it.

    Other fields are typeless versions of their real value types. These include VAnsiStringand VInterface. When you assign to those fields, beware that they don't maintain the usual reference count that an ordinary AnsiString or IUnknown variable would, so again, watch those variable lifetimes.

    The compiler is usually the only thing that generate such arrays, so there's little reference code available to see how they're built. Instead, you can look at other code that consumes arrays of const. For example, I implemented a Unicode-aware Format function for the JCL a number of years ago. It parses the format string one character at a time using a finite-state machine. Each time it finishes parsing an argument string, it fetches the corresponding argument from the input array and formats it according to the string and the argument type.

    It used a minimum of assembler, and only for some minor efficiency, not because it was ever truly necessary. For reference, all the assembler is accompanied by the equivalent Delphi code in comments.

    0 讨论(0)
  • 2020-12-18 02:25

    The array of const gives you the freedom to add strings, integers, floats and so on and having these formatted into a string. And there is no limit to how many items you can add.

    The way Delphi deals with this issue is that the array of const really is a array of TVarRec's.

    A TVarRec is a record of the following type:

    TVarRec = record
      case Byte of
        vtInteger:    (VInteger: Integer; VType: Byte);
        vtBoolean:    (VBoolean: Boolean);
        vtChar:       (VChar: Char);
        vtExtended:   (VExtended: PExtended);
        vtString:     (VString: PShortString);
        vtPointer:    (VPointer: Pointer);
        vtPChar:      (VPChar: PChar);
        vtObject:     (VObject: TObject);
        vtClass:      (VClass: TClass);
        vtWideChar:   (VWideChar: WideChar);
        vtPWideChar:  (VPWideChar: PWideChar);
        vtAnsiString: (VAnsiString: Pointer);
        vtCurrency:   (VCurrency: PCurrency);
        vtVariant:    (VVariant: PVariant);
    

    The type of the value inside the TVarRec is determined by the VType value.

    This gives you the flexibility to add either type you wish to the array of const, like in the Format() function:

    Format( '%s is a string, %d is an integer', ['string',10] );

    Using the array of const in your own procedure is no big deal. Take a look at this example:

     procedure VarArraySample( AVarArray : array of const );
      var
        i : integer;
      begin
        for i := 0 to High(AVarArray) do
          do_something;
      end;
    

    The function High() returns the last index of the array.

    You can also convert the contents of the TVarRec. This example is taken from the Delphi on-line help and revamped a bit. The function converts a TVarRec to a string:

    function VarRecToStr( AVarRec : TVarRec ) : string;
      const
        Bool : array[Boolean] of string = ('False', 'True');
      begin
        case AVarRec.VType of
          vtInteger:    Result := IntToStr(AVarRec.VInteger);
          vtBoolean:    Result := Bool[AVarRec.VBoolean];
          vtChar:       Result := AVarRec.VChar;
          vtExtended:   Result := FloatToStr(AVarRec.VExtended^);
          vtString:     Result := AVarRec.VString^;
          vtPChar:      Result := AVarRec.VPChar;
          vtObject:     Result := AVarRec.VObject.ClassName;
          vtClass:      Result := AVarRec.VClass.ClassName;
          vtAnsiString: Result := string(AVarRec.VAnsiString);
          vtCurrency:   Result := CurrToStr(AVarRec.VCurrency^);
          vtVariant:    Result := string(AVarRec.VVariant^);
        else
          result := '';
        end;
      end;
    

    You can combine the two functions above to one function that converts all elements in the array of const into one string:

    function VarArrayToStr( AVarArray : array of const ) : string;
      var
        i : integer;
      begin
        result := '';
        for i := 0 to High(AVarArray) do
          result := result + VarRecToStr( AVarArray[i] );
      end;
    

    you will now be able to create your own Format() function. The Format() function scans for %'s and replaces the %something with the value in the array of const, depending on the format specifiers and precision specifiers.

    0 讨论(0)
提交回复
热议问题