String -> byte[] -> String - Sieht jemand was ich nicht sehe?

White_Fox

Top Contributor
Guten Morgen allerseits

Folgendes: Ich habe eine Klasse "MemberfieldDeskriptor", diese Klasse enthält Informationen über eine Membervariable eines Objekts. Die Klasse soll außerdem in der Lage sein, die Informationen in ein byte[] zu verpacken, und diese Informationen auch aus einem byte[] wieder herauszulesen.

Jetzt habe ich folgendes Problem: Mein Unittest kommt mit diesem Fehler zurück:
JUnit hat gesagt.:
Failed: expected:<canonical class name[]> but was:<canonical class name[]>

Wenn ich mit dem Debugger an besagter Stelle anhalte, dann liefert ein result.equals(expResult) sogar tatsächlich false zurück.

Die Konvertierung von String nach byte[] und umgekehrt erfolgt so:
Java:
//String -> byte[]
byte[] classNameBytes;

classNameBytes = StandardCharsets.UTF_8.encode(className).array();

//byte[] -> String
this.className = new String(classNameBytes, StandardCharsets.UTF_8);

Hat jemand eine Idee, was da schiefläuft?
 

JCODA

Top Contributor
Kannst du mal angeben, bei welchem String das genau fehlschlägt?
Du hast also einen String className, bei welchem folgender Test false liefert?

Java:
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
class Main {
  public static void main(String[] args) {
    String className = "abz";
    byte[] classNameBytes= StandardCharsets.UTF_8.encode(className).array();
    System.out.println(Arrays.toString(classNameBytes));
    String className2 = new String(classNameBytes, StandardCharsets.UTF_8);
    System.out.println(Arrays.toString(className2.getBytes()));
    System.out.println(className.equals(className2));

  }
}
 

White_Fox

Top Contributor
Der Test würde in deiner letzten System.out.println-Zeile fehlschlagen. Dort teste ich mit assertEquals() auf Gleichheit.

Im Ganzen sieht der Unittest wie folgt aus:
Java:
@Test
public void testFitUp() {
    MemberfieldDescriptor descriptor, instance;
    String name, className;
    String result, expResult;
    byte[] bytes, nextBytes;
    ByteBuffer buffer;
    MemberType resultType, expType;

    System.out.println("fitUp");

    name = "fieldname";
    className = "canonical class name";
    descriptor = new DescriptorImplementation(name, className);

    bytes = descriptor.stripDownToBytes();

    buffer = ByteBuffer.wrap(bytes);
    buffer.position(4);            //Der buffer enthält jetzt noch ein int vorne, das erst ausgeschnitten werden muß
    nextBytes = new byte[buffer.remaining()];
    buffer.get(nextBytes);
    instance = new DescriptorImplementation(nextBytes);

    expResult = descriptor.className();
    result = instance.className();
    assertEquals(expResult, result);    //Test schlägt hier mit der genannten Meldung fehl

    expResult = descriptor.name();
    result = instance.name();
    assertEquals(expResult, result);

    expType = descriptor.type();
    resultType = instance.type();
    assertEquals(expType, resultType);
}

Das byte[] wird mit der Methode stripDowntoBytes erzeugt, und mit der fitUp-Methode wieder in seinen ursprünglichen Zustand zurückversetzt:

Java:
byte[] stripDownToBytes(){
    ByteBuffer buffer;
    byte[] nameBytes;
    byte[] classNameBytes;
    byte[] valueBytes;

    nameBytes = StandardCharsets.UTF_8.encode(name).array();
    classNameBytes = StandardCharsets.UTF_8.encode(className).array();
    valueBytes = this.stripValueDownToBytes();

    buffer = ByteBuffer.allocate(
    4 +                            //mainlength
    4 +                            //MemberType
    4 + nameBytes.length +
    4 + classNameBytes.length +
    4 + valueBytes.length);

    buffer.putInt(buffer.limit());
    buffer.putInt(type().order());

    buffer.putInt(nameBytes.length);
    buffer.put(nameBytes);

    buffer.putInt(classNameBytes.length);
    buffer.put(classNameBytes);

    buffer.putInt(valueBytes.length);
    buffer.put(valueBytes);

    return buffer.array();
}

private void fitUp(byte[] bytes){
    ByteBuffer buffer;
    MemberType type;
    int nameLength;
    int classNameLength;
    int valueLength;
    byte[] nameBytes;
    byte[] classNameBytes;
    byte[] valueBytes;

    buffer = ByteBuffer.wrap(bytes);

    type = MemberType.getMemberTypeFromOrder(buffer.getInt());
    fitTypeUp(type);

    nameLength = buffer.getInt();
    nameBytes = new byte[nameLength];
    buffer.get(nameBytes);
    this.name = new String(nameBytes, StandardCharsets.UTF_8);

    classNameLength = buffer.getInt();
    classNameBytes = new byte[classNameLength];
    buffer.get(classNameBytes);
    this.className = new String(classNameBytes, StandardCharsets.UTF_8);

    valueLength = buffer.getInt();
    valueBytes = new byte[valueLength];
    buffer.get(valueBytes);
    this.fitValueUp(valueBytes);
}
 

temi

Top Contributor
Es gibt da einen Unterschied:
Java:
    public static void main(String[] args) {
        String className = "canonical class name";
        
        byte[] classNameBytes1 = StandardCharsets.UTF_8.encode(className).array();
        byte[] classNameBytes2 = className.getBytes(StandardCharsets.UTF_8);

        System.out.println(Arrays.toString(classNameBytes1));
        System.out.println(Arrays.toString(classNameBytes2));

Code:
[99, 97, 110, 111, 110, 105, 99, 97, 108, 32, 99, 108, 97, 115, 115, 32, 110, 97, 109, 101, 0, 0]
[99, 97, 110, 111, 110, 105, 99, 97, 108, 32, 99, 108, 97, 115, 115, 32, 110, 97, 109, 101]
[99, 97, 110, 111, 110, 105, 99, 97, 108, 32, 99, 108, 97, 115, 115, 32, 110, 97, 109, 101, 0, 0]
 

White_Fox

Top Contributor
Hm...und wie krieg ich das gelöst?

Edit:
Warum ist da überhaupt ein Unterschied? Ich dachte, das ist doch beides UTF-8? Und da ist sowas wie Nullterminierung doch vorgesehen?
 

temi

Top Contributor
Hmm, hab mal mit dem String gespielt:

Bei String className = "class name"; ist das Resultat:
Code:
[99, 108, 97, 115, 115, 32, 110, 97, 109, 101, 0]
[99, 108, 97, 115, 115, 32, 110, 97, 109, 101]
[99, 108, 97, 115, 115, 32, 110, 97, 109, 101, 0]

und bei String className = "class nam";
Code:
[99, 108, 97, 115, 115, 32, 110, 97, 109]
[99, 108, 97, 115, 115, 32, 110, 97, 109]
[99, 108, 97, 115, 115, 32, 110, 97, 109]
 

temi

Top Contributor
Ich würde sagen, sobald der String länger ist als 9 Zeichen, wird eine 0 angehängt. Empirisch ermitteltes Ergebnis.

Und zwar jedesmal eine zusätzliche Null, wenn die 10 voll wird.
10 Zeichen => 1 Null
20 Zeichen => 2 Nullen
30 Zeichen => 3 Nullen

String className = "123456789012345678901234567890";
Code:
[49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 0, 0, 0]
[49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 48]
[49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 48, 0, 0, 0]

Verrückt, aber bestimmt gibt es einen guten Grund dafür. :cool:
 
Zuletzt bearbeitet:

White_Fox

Top Contributor
Hm...ich hab jetzt einfach mal die Methode, den String in bytes zu zerlegen, geändert. Jetzt funktioniert es (zumindest an dieser Stelle).

Aber dennoch: Ich finde das Verhalten merkwürdig, soll das so sein?
 

LimDul

Top Contributor
die array Methode von einem ByteBuffer (der von Charset.encode zurück kommt) aufzurufen macht vermutlich nicht das, was du denkst.
Returns the byte array that backs this buffer (optional operation).
Modifications to this buffer's content will cause the returned array's content to be modified, and vice versa.

Invoke the hasArray method before invoking this method in order to ensure that this buffer has an accessible backing array.
Und nirgendwo ist in der Spezifikation beschrieben, dass das Array des ByteBuffers nur so groß sein darf, wie nötig. Du könntest mal versuchen vorher compact aufzurufen.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
A Byte zu String Allgemeine Java-Themen 4
O Byte-Array zu String Allgemeine Java-Themen 7
P Datentypen String-Daten zu Byte-Zahlen konvertieren - Komme nicht weiter nach vielem versuchen :-/ Allgemeine Java-Themen 7
E Byte zu String & umgekehrt Allgemeine Java-Themen 3
S byte [] in string und zurück konvertieren Allgemeine Java-Themen 2
E Byte-Array to String: Zeichenkaputt Allgemeine Java-Themen 11
R byte[] to String Konvertieren Allgemeine Java-Themen 14
J Hex-String zu byte transformieren Allgemeine Java-Themen 7
G String in byte- Array Allgemeine Java-Themen 3
F byte in hex-String oder: Wer hat in Mathe aufgepasst Allgemeine Java-Themen 3
J NumberFormatException bei String->byte[] Allgemeine Java-Themen 12
I String -> byte[] -> String Allgemeine Java-Themen 2
E String -> byte[] Allgemeine Java-Themen 6
C Byte[] to String Allgemeine Java-Themen 7
J byte-Array als String übers http schicken Allgemeine Java-Themen 8
F List<String> zu byte[] Allgemeine Java-Themen 7
R byte - string? Allgemeine Java-Themen 10
X Byte To String Allgemeine Java-Themen 9
I Performance bei "String <-> Byte"-Umwandlung Allgemeine Java-Themen 4
Reeny byte[] als String speichern _ohne_ es zu dekodieren Allgemeine Java-Themen 6
OnDemand Java String in Hashmap als Key NULL Allgemeine Java-Themen 27
JAnruVA Datentypen Berechneten String-Wert in Double umwandeln um weiter zu rechnen Allgemeine Java-Themen 7
M String Allgemeine Java-Themen 10
M Suche nach String mit unbekannten characters Allgemeine Java-Themen 53
kodela String kann nicht zu Pfad konvertiert werden Allgemeine Java-Themen 16
melaniemueller Einzelne Zeile aus einer txt Datei in einem String speichern Allgemeine Java-Themen 12
E Objekte in einen String packen und wieder laden Allgemeine Java-Themen 5
M Map<String,String>funktioniert nicht richtig Allgemeine Java-Themen 4
O String in Long Hexerdezimal umwandel Allgemeine Java-Themen 14
N String vergleichen. Allgemeine Java-Themen 27
P String.replace() funktioniert nicht? Allgemeine Java-Themen 3
SaschaMeyer Arbeitet String.split falsch? Allgemeine Java-Themen 4
M Switches ohne String Allgemeine Java-Themen 18
AmsananKING String Iteration Allgemeine Java-Themen 5
S Shuffle String aus if-clause Allgemeine Java-Themen 11
Besset Variablen Ist String = "" + int inordnung? Allgemeine Java-Themen 6
M Map <Long, String> zu Map<String, Long> Allgemeine Java-Themen 9
S String Encoding Verständnisproblem Allgemeine Java-Themen 22
N Prüfen, ob ein String 2x das selbe Zeichen hat Allgemeine Java-Themen 10
SaftigMelo Bug Fixen von String-spliten Allgemeine Java-Themen 8
Monokuma String List nach Zahlen und Worten sortieren Allgemeine Java-Themen 9
Kingamadeus2000 Alle mehrfach vorkommenden Buchstaben rekursiv aus einem String entfernen. Allgemeine Java-Themen 6
YohnsonM String - Aufteilung und Nutzung einzelner Chars Allgemeine Java-Themen 7
O Formatierte String ausgabe bei vier Variablen in einer Zeile Allgemeine Java-Themen 1
S String umbenennen: wie? Allgemeine Java-Themen 4
x46 String Format Fehler Allgemeine Java-Themen 2
S ISO 8601 -> getter / setter String Allgemeine Java-Themen 3
L String zu repräsentativen Wert Allgemeine Java-Themen 0
H Array mit dem Datentype String[] initializieren Allgemeine Java-Themen 7
L ArrayList mit String Arrays in ein Array umwandeln Allgemeine Java-Themen 1
L regex ganzer string? Allgemeine Java-Themen 2
L Ist ein string ein erlaubter variabel name? Allgemeine Java-Themen 2
Z JNA Cpp-DLL String Verwendung Allgemeine Java-Themen 2
A String auf Zahlen überprüfen Allgemeine Java-Themen 5
N String Array Eingabe Allgemeine Java-Themen 6
MiMa Datum von String zu LocalDateTime Allgemeine Java-Themen 8
R char aus String entfernen Allgemeine Java-Themen 10
LimDul Mittels Streams aus Strings A B C den String A, B und C machen Allgemeine Java-Themen 12
M Programm erkennt String aus .txt Datei nicht Allgemeine Java-Themen 3
P einen public <Optinal String> in einer anderen Klasse mit einem Int vergleichen Allgemeine Java-Themen 2
S Ini Text aus String parsen Allgemeine Java-Themen 1
T String-Manipulation beim Ablauf in Eclipse und als JAR-File Allgemeine Java-Themen 8
M String lässt sich nicht Zusammenfügen Allgemeine Java-Themen 10
Drachenbauer Wie kann ich das Wort "concrete" in einem String durch ein anderes Wort ersetzen lassen? Allgemeine Java-Themen 5
R Schlüsselworte "Throw new exception" gibt nicht den String als Fehlermeldung aus Allgemeine Java-Themen 2
R Variablen String mit split-Funktion aufteilen Allgemeine Java-Themen 7
F Datei in String-Array einlesen Allgemeine Java-Themen 8
S Marker aus String ermitteln Allgemeine Java-Themen 5
T Objekt mit String und Int aus TxT Datei erstellen Allgemeine Java-Themen 23
M Bei String.format ein Komma statt einem Punkt ausgeben lassen Allgemeine Java-Themen 1
S MSSQL Exception & Connection String Allgemeine Java-Themen 19
B Bei Email: FW / AW... - Hilfe bei String suche Allgemeine Java-Themen 21
J String - Vergleiche Allgemeine Java-Themen 7
K Aus String zwei Jahreszahlen auslesen Allgemeine Java-Themen 18
Drachenbauer Wie kann eine vorgegebene Farbe über einen String erkannt werden? Allgemeine Java-Themen 11
G CSV in String Allgemeine Java-Themen 7
P String-Verschlüsselung - Frage zur Sicherheit Allgemeine Java-Themen 21
K Methodenaufruf mit String / String zu Objekt konvertieren Allgemeine Java-Themen 8
D Erste Schritte Fehler mit negativen und 0 Zahlen im String Allgemeine Java-Themen 6
Xge Replace x Zeichen aus String Allgemeine Java-Themen 2
coolian warum bekomme ich ein string index out of bounds exception Allgemeine Java-Themen 17
F In String 2 Buchstaben vertauschen Allgemeine Java-Themen 2
J Class Decompile als String (Procyon) Allgemeine Java-Themen 2
I Datentypen String in class sicher verwahren Allgemeine Java-Themen 17
J Falls der String ein "X" beinhaltet Allgemeine Java-Themen 2
T String mehrere Worte Allgemeine Java-Themen 2
D String Groß-/Kleinschreibung Allgemeine Java-Themen 2
D String und Klassenvariable Allgemeine Java-Themen 6
Aruetiise Funktion(y = mx+n) in String speichern und berechnen Allgemeine Java-Themen 9
C String in Objektnamen umwandeln Allgemeine Java-Themen 3
E Variablen Aus .txt ausgelesener string mit if() überprüfen? Allgemeine Java-Themen 2
L String-Schema-Aufspaltung Allgemeine Java-Themen 2
E String in Zahl umwandeln, ohne Befehl Integer.parseInt Allgemeine Java-Themen 3
L String splitten und multiplizeren Allgemeine Java-Themen 10
G String mit umbekannter länge splitten. Allgemeine Java-Themen 2
S Einzigartigen String in Datenbank finden und löschen Allgemeine Java-Themen 23
B Von String zu <Objekt> ||Speichern/Laden Allgemeine Java-Themen 17
T Komplexitätsoptimierung String vergleich Allgemeine Java-Themen 4
heinz ketchup String im JLabel ausgeben und erneuern Allgemeine Java-Themen 6
S Input/Output Beste Möglichkeit einen String in einen Datei zu Schreiben Allgemeine Java-Themen 2

Ähnliche Java Themen

Neue Themen


Oben