>백엔드 개발 >Golang >golang 유형 어설션을 이해하고 학습합니다.

golang 유형 어설션을 이해하고 학습합니다.

藏色散人
藏色散人앞으로
2021-05-27 15:27:412557검색

다음은 golang튜토리얼 칼럼의 golang 유형 어설션에 대한 소개입니다. 도움이 필요한 친구들에게 도움이 되길 바랍니다!

PHP에는 저장 및 전송을 위해 배열을 문자열로 직렬화할 수 있는 serialize() 함수가 있습니다.

이런 종류의 문자열을 역직렬화하려면 PHP에서 간단한 unserialize() 함수만 사용하면 됩니다. 하지만 golang에서는 그렇게 쉽지 않습니다. 많은 노력을 기울여야 하고 많은 코드를 작성해야 합니다.

이때엔 그냥 한숨이 나오네요. PHP는 정말 세상에서 가장 좋은 언어입니다!

오늘 개발에서 이런 문제가 발생했습니다. PHP 직렬화된 문자열을 구문 분석하려면 golang을 사용해야 합니다. github에서 구문 분석 패키지를 찾았는데 구문 분석 후 결과가 인터페이스{} 유형이라는 것을 알았습니다.

갑자기 어디서부터 시작해야 할지 몰랐습니다. 항상 데이터가 실제로는 맵이라고 생각했는데, 파싱하고 나니 인터페이스가 생겼습니다.{} 이것을 어떻게 사용해야 할까요? 주장이 있는데 어떻게 사용해야 할지 몰라서 커뮤니티에 가서 물어보면 큰 분들이 답변해 주셨으면 좋겠습니다.

실제로는 정말 잘 작동합니다.

아래 코드를 게시하세요:

package main

import (
	"github.com/yvasiyarov/php_session_decoder/php_serialize"
	"fmt"
	"log"
)


func main() {
        // 序列化后的字符串
	str := `a:3:{s:4:"name";s:3:"tom";s:3:"age";s:2:"23";s:7:"friends";a:2:{i:0;a:1:{s:4:"name";s:5:"jerry";}i:1;a:1:{s:4:"name";s:4:"jack";}}}`
	
        // 反序列化
        decoder := php_serialize.NewUnSerializer(str)
	if result, err := decoder.Decode(); err != nil {
		panic(err)
	} else {
                // 此处进行类型断言
		decodeData, ok := result.(php_serialize.PhpArray)
		if !ok {
			log.Println(err)
		}

                // 断言后,即可获取内容
		name := decodeData["name"]
		age := decodeData["age"]
		fmt.Println(name, age)

                // 内层数据仍需再次断言
		friends, ok := decodeData["friends"].(php_serialize.PhpArray)
		if !ok {
			log.Println(err)
		}

                // 断言成功后即可获取内部数据
		for _,v := range friends {
			fmt.Printf("type:%T, value:%+v\n", v,v )
			friend, ok := v.(php_serialize.PhpArray)
			if !ok {
				log.Println(err)
			}
			friendName := friend["name"]
			fmt.Println(friendName)
		}
	}
}

대략 다음과 같이 이해할 수 있습니다: 변수의 유형이 무엇인지, 어떤 유형의 어설션이 이루어지고, 어설션 후에 결과를 얻을 수 있습니다

변수의 유형을 결정하는 방법 변하기 쉬운?

인쇄하세요: fmt.Printf("%T", verb)

%T는 변수의 유형을 표시할 수 있습니다.

아래 예가 있습니다:

package main

import (
	"github.com/yvasiyarov/php_session_decoder/php_serialize"
	"fmt"
	"log"
)

func main() {
	// 序列化后的字符串
	str := `a:3:{s:4:"name";s:3:"tom";s:3:"age";s:2:"23";s:7:"friends";a:2:{i:0;a:1:{s:4:"name";s:5:"jerry";}i:1;a:1:{s:4:"name";s:4:"jack";}}}`
	
	// 反序列化
	decoder := php_serialize.NewUnSerializer(str)
	result, err := decoder.Decode()
	if err != nil {
		panic(err)
	}
	// 类型断言
	t := result.(php_serialize.PhpArray)
	strVal := php_serialize.PhpValueString(t["name"])
	fmt.Println(strVal)
	
	switch t := result.(type) {
	default:
		fmt.Printf("unexpected type %T\n", t)
	case php_serialize.PhpArray:
		fmt.Println(t)
		fmt.Println(t["name"])
		fmt.Println(t["age"])

		switch f := t["friends"].(type) {
		default:
			fmt.Printf("unexpected type %T\n", t)
		case php_serialize.PhpArray:
			fmt.Println(f)
			fmt.Println(f[0])
			fmt.Println(f[1])
		}
	}
}

위의 두 데모 모두 효과를 얻을 수 있습니다. 그냥 다르게 썼어요.

나중에 효과를 얻을 수 있는 다른 패키지를 소개받았습니다.

package main

import (
	"fmt"
	"log"
	"github.com/wulijun/go-php-serialize/phpserialize"
)

func main() {
	str := `a:3:{s:4:"name";s:3:"tom";s:3:"age";s:2:"23";s:7:"friends";a:2:{i:0;a:1:{s:4:"name";s:5:"jerry";}i:1;a:1:{s:4:"name";s:4:"jack";}}}`
	decodedRes, err := phpserialize.Decode(str)
	if err != nil {
		panic(err)
	}
	//fmt.Printf("%T\n", decodedRes) //type is map[interface{}]interface{}

	//type assert
	decodedData, ok := decodedRes.(map[interface{}]interface{})
	if !ok {
		fmt.Printf("unexpected type %T\n", decodedRes)
	}
	fmt.Println(decodedData["name"])
	fmt.Println(decodedData["age"])

	//fmt.Printf("%T\n", decodedData["friends"]) // type is map[interface{}]interface{}
	// type assert
	friendsRes, ok := decodedData["friends"].(map[interface{}]interface{})
	if !ok {
		fmt.Printf("unexpected type %T\n", decodedData["friends"])
	}
	for _,v := range friendsRes {
		//fmt.Printf("type: %T, val: %#v\n", v,v) // type is map[interface{}]interface{}
		friend, ok := v.(map[interface{}]interface{})
		if !ok {
			fmt.Printf("unexpected type %T\n", decodedData["friends"])
		}
		//fmt.Printf("type: %T, val: %#v\n", friend,friend) // type is map[interface{}]interface{}
		fmt.Println(friend["name"])
	}
}

이 패키지로 구문 분석된 모든 결과의 유형은 map[interface{}]interface{}이므로 유형 어설션을 만들 때 다음과 같이 할 수 있습니다. 더 간단하고 조잡합니다.

위 내용은 golang 유형 어설션을 이해하고 학습합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 cnblogs.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제