SchultzGames
Para tirar dúvidas, crie um tópico especifico na área "Dúvidas em Geral" e aguarde ser respondido. Evite comentar em tópicos não relacionados a sua dúvida :D
TÓPICOS RECENTES:
Tópico:
Postado em:
Postado por:
Podcast de desenvolvedores de jogos
ANIMAÇÃO DE ARMS
SOU MODELADOR DE MODELOS LOW POLY MEDIUM PRECISANDO E SO CHAMAR
Component Cloth unity
Pulo em Projeto 3d com Character Controller
pausar musica e sons
Selecionar itens com as teclas do teclado
Pausar jogo
Limitar movimento da camera com imput android!!
Spawnar objeto a 1 metro de distancia de mim
Cabeça do personagem virar ao mecher a câmera!
NÃO CONSIGO COLOCAR TEXTURAS NAS PAREDES DAS CASAS!!!
Simulador controlado a partir de. Um app
Google Play Services
Meu personagem customizado não quer morrer no RealisticFPSPrefab
Unity colisao nao funciona
FillAmount da imagem sincronizado com um timer
Como Fazer Camera do Euro Truck 2 (para Android)
Erro na transição de cena
vc code não reconhece minha função, ajuda!?
Player não segue a trajetória quando pula correndo
[RESOLVIDO] AS ÁRVORES DO MEU UNITY ESTÃO ROSAS, O QUE FAZER!!???
Batalha por turnos
jogo de corrida
Melhor Configuiração Para Rodar Unity e Programas de Modelagem !
Tetris: Ghost House
Limpeza de animação
ANIMAÇÃO DE MORTE EM LOOPING
Random Maze
Como fazer para os braços subirem e descerem?
Pedido de Script de Crosbow
Camera some com partes do corpo
Pack De Modelos Para Baixar
modificar armas sem tirar animações
Batalha por turnos
Converter UASSET em FBX
ajuda para adaptação em um script.
[TUTORIAL] Salvando e carregando dados com serialização (Parte 3)
Duvida sobre animação
[TUTORIAL] Salvando e carregando dados com serialização (Parte 2)
[TUTORIAL] Sistema de PORTAS e CHAVES
Copiar prefab de um projeto e enviar para outro!!
Unity + Photon 2
[TUTORIAL] Acessar o objeto pai de algum hit.
[TUTORIAL] Salvando e carregando dados com serialização (Parte 1)
ERRO: Falha ao inicializar mecanismo de armazenamento de persistência
[TUTORIAL] Script de Spawn Aleatório
Tamanho aleatorio sprite
Dúvida sobre PlayerPrefs
Como criar um offset fluido na Unity?
Cor e transparencia de um objeto
Problema na rotação da Bicicleta
expotaçao blender para unity
Sistema de ataque com cooldown e classe Time da unity.
Angulo unity
[EXEMPLOS DE CODIGO] Uso ou Para Aprendizagem
[RESOLVIDO] - Blender - Deixar a Face do Cubo pra cima
Interrogação beta, jogo de puzzle 3D
[TUTORIAL] Movimentação SQM (2D)
Duvida C#
sistema de dialogo mostra o html sendo escrito (Unity3D)
não consigo andar, pular ou mover a camera no meu jogo
Otimização
Alguem sabe resolver esse erro Gradle build failed?
Touch Drag
estou tentando rotacionar o personagem porem ele não segue a direção certa
[RESOLVIDO] alguém sabe resolver o erro Unable to Load Firebase?
Unity remote 5
[RESOLVIDO] Como retirar esses T que está do local onde tem textos?
[RESOLVIDO] Detectar qual lado da tela estou arrastando o dedo no touch?
Existe algo parecido com Shaders Graph ?
[Duvida] como implementar limite vector2
[RESOLVIDO] Bicicleta no Unity
Problema com um script de som
[RESOLVIDO] Efeito de vento Unity 2D
Como criar uma capa
Help em exibir informações do personagem na UI do game
[TUTORIAL] Fazer objeto se mover em pontos diferentes (Serve para 2D e 3D).
Texto Sumir e tela apagar Gradativamente
O Botão do Canvas não mostra os Métodos do Script que está no Objeto
[TUTORIAL] Desbloqueio e Escolha de Personagens.
erro IndexOutOfRangeException: Index was outside the bounds of the array.
Movimentação de x para Y
Unity 2d Aspect ratio
Sistema de tiro na diagonal em jogo 2D
Precisa-se Programador Unity - Trabalho remunerado
Estatisticas Gerais dentro do Jogo?
Bug(???) no AudioSource
Sniper Modelo 3D
Teaser da Gameplay de mais um jogo. Bio Weapons Z
FBX não gera sombra
Procuro programador para JOGO DE TERROR
Qual servidor com melhor custo-benefício?
Como pegar 2 Touchs e posicoes na tela
Doações na Google Play
Exercícios - iniciantes
Jogo 2D Personagem atravessa o chão quando Pula muito alto
(Dúvida) Criação de gráficos em um jogo na Unity
Meu personagem ignora as colisoes laterais dos tilemap 2D
Ontem à(s) 10:45 pm
Ontem à(s) 9:38 pm
Ontem à(s) 6:00 pm
Ontem à(s) 11:32 am
Sex Out 22, 2021 6:59 pm
Sex Out 22, 2021 3:39 pm
Sex Out 22, 2021 2:39 pm
Qui Out 21, 2021 10:34 am
Qua Out 20, 2021 3:42 pm
Qua Out 20, 2021 11:49 am
Ter Out 19, 2021 10:48 pm
Ter Out 19, 2021 4:19 pm
Ter Out 19, 2021 11:20 am
Ter Out 19, 2021 12:01 am
Seg Out 18, 2021 6:39 pm
Seg Out 18, 2021 5:51 pm
Seg Out 18, 2021 5:45 pm
Dom Out 17, 2021 11:56 pm
Dom Out 17, 2021 11:29 pm
Sab Out 16, 2021 5:55 pm
Sab Out 16, 2021 3:38 pm
Sex Out 15, 2021 9:57 pm
Sex Out 15, 2021 6:35 pm
Sex Out 15, 2021 2:31 pm
Sex Out 15, 2021 10:52 am
Qui Out 14, 2021 10:50 pm
Qui Out 14, 2021 2:50 am
Qua Out 13, 2021 12:07 pm
Qua Out 13, 2021 3:54 am
Ter Out 12, 2021 10:48 pm
Ter Out 12, 2021 10:46 pm
Seg Out 11, 2021 7:57 pm
Seg Out 11, 2021 5:44 pm
Seg Out 11, 2021 5:42 pm
Seg Out 11, 2021 10:14 am
Dom Out 10, 2021 8:22 pm
Dom Out 10, 2021 9:56 am
Sab Out 09, 2021 6:55 pm
Sab Out 09, 2021 6:11 pm
Sab Out 09, 2021 2:00 pm
Sab Out 09, 2021 12:33 am
Sab Out 09, 2021 12:24 am
Sex Out 08, 2021 8:19 pm
Sex Out 08, 2021 8:08 pm
Sex Out 08, 2021 2:00 pm
Qui Out 07, 2021 8:03 pm
Qui Out 07, 2021 1:47 pm
Qua Out 06, 2021 3:26 pm
Ter Out 05, 2021 9:01 pm
Ter Out 05, 2021 2:58 pm
Ter Out 05, 2021 2:45 pm
Ter Out 05, 2021 2:22 pm
Seg Out 04, 2021 7:59 pm
Seg Out 04, 2021 6:21 pm
Seg Out 04, 2021 6:20 pm
Dom Out 03, 2021 10:03 am
Sab Out 02, 2021 9:28 pm
Sab Out 02, 2021 7:47 pm
Sab Out 02, 2021 8:57 am
Sex Out 01, 2021 10:14 pm
Sex Out 01, 2021 4:10 pm
Sex Out 01, 2021 12:39 pm
Qui Set 30, 2021 9:20 pm
Qui Set 30, 2021 8:31 pm
Qui Set 30, 2021 3:31 pm
Qui Set 30, 2021 3:09 pm
Qui Set 30, 2021 12:41 pm
Qua Set 29, 2021 7:54 pm
Qua Set 29, 2021 6:30 pm
Qua Set 29, 2021 12:41 pm
Qua Set 29, 2021 8:28 am
Ter Set 28, 2021 8:34 pm
Ter Set 28, 2021 8:16 pm
Ter Set 28, 2021 7:34 pm
Ter Set 28, 2021 6:42 pm
Ter Set 28, 2021 10:34 am
Ter Set 28, 2021 9:36 am
Ter Set 28, 2021 8:09 am
Seg Set 27, 2021 11:32 am
Seg Set 27, 2021 5:53 am
Dom Set 26, 2021 7:48 pm
Dom Set 26, 2021 5:19 pm
Dom Set 26, 2021 1:14 pm
Sab Set 25, 2021 5:50 pm
Sab Set 25, 2021 10:04 am
Sab Set 25, 2021 8:44 am
Sex Set 24, 2021 9:43 pm
Sex Set 24, 2021 5:11 pm
Sex Set 24, 2021 5:09 pm
Sex Set 24, 2021 5:07 pm
Sex Set 24, 2021 4:30 pm
Sex Set 24, 2021 4:30 pm
Sex Set 24, 2021 4:23 pm
Sex Set 24, 2021 4:12 pm
Sex Set 24, 2021 4:10 pm
Sex Set 24, 2021 1:33 pm
Sex Set 24, 2021 8:00 am
Qui Set 23, 2021 5:58 pm
Qui Set 23, 2021 3:51 pm
Rangel Oblivion
edu63
claudiano2020
capim22
eadev99
joaozinpedrin
Prandini
joaozinpedrin
capim22
Fagner
jhonih00010
Crash Psycho
Joana
nando07
leozinhobx
gustakegamer@gmail.com
kalielson
Kayke_27
fernando.OVS
DuduBertol
WesFarias
Fagner
igorobm
Crash Psycho
Claus13
classicandsimple
AGAMENOM
ViniciusTKH
igorobm
stratengine
crazylagg
igorobm
edu63
edu63
edsonvinicius76
Xavier
lauderson
igorobm
igorobm
Fagner
marcussant
FelipeSouza11
igorobm
igorobm
MayLeone
kessisdiones
BBFree
MayLeone
MayLeone
MayLeone
MayLeone
Fagner
Pkneves
stratengine
stratengine
Rangel Oblivion
WLCS22
Rangel Oblivion
edsonvinicius76
Pokedlg
alnmg
Charlesoff
Xavier
kessisdiones
joaozinpedrin
Fagner
kessisdiones
joaozinpedrin
kessisdiones
Atho
GustavoPeky2
g
leleo
pdsimulador
Prandini
Xavier
tasf
edsonvinicius76
TIKO
TIKO
TIKO
TIKO
edsonvinicius76
nando07
sidney oliveira
Crash Psycho
Kayke_27
TIKO
TIKO
Patrick
Fagner
bertarele
Fagner
Fagner
Fagner
Rangel Oblivion
ronasser
poyser
Play3rAleatori0

[TUTORIAL] Salvando e carregando dados com serialização (Parte 3)

3 participantes

Ir para baixo

DÚVIDA [TUTORIAL] Salvando e carregando dados com serialização (Parte 3)

Mensagem por MayLeone Sex Out 08, 2021 5:59 pm

The Last one:

Bem vindos(as) a última parte do nosso tutorial de como criar uma arquitetura simples e eficaz para persistir dados em seu jogo.
É de suma importância que você já esteja por dentro de assuntos já mencionados anteriormente, como serialização e playerprefs, para que possa dar seguimento a esta leitura de maneira mais segura. Caso contrário, recomendo dar uma rápida lida nos tópicos linkados como referência.

As quatro entidades:

Parece nome de filme de terror, mas esse assunto do tópico é de extrema importância para a concepção da arquitetura do nosso sistema de save/load para que ele fique -como nossos amigos gringos gostam de chamar- pretty much simple, ou seja, ele vai ficar tão simples de utilizar e vai se encaixar perfeitamente com qualquer situação e tipo de dado, que você vai ficar muito feliz por ter encontrado esse tópico aqui.

Para estruturar nosso sistema, vamos precisar pensar em quatro entidades responsáveis por gerenciar todo nosso ecossistema: a classe concreta que mantém o modelo de dados a serem carregados, a classe abstrata que assegura o tipo do dado e mantém um contrato de implementação, a classe responsável por realizar as ações em torno do save/load do sistema, e a classe MonoBehaviour necessária para consumir essas informações.

Classe concreta:

Vamos começar pela classe concreta que vai possuir as variáveis necessárias para serem salvas e carregadas.
Essa classe em nosso exemplo, vai se assemelhar muito à struct que tínhamos nos tutoriais anteriores (ainda mais porque vamos utilizar ainda a ideia de "PlayerData").
Então para tal, vamos manter as mesmas variáveis, com a diferença de que estamos lidando com uma classe e não uma struct:

Código:
namespace Data{
 [System.Serializable]
    public class PlayerData
    {
        public string name;
        public float coins;
        public float health;
        public int score;
}
}


Aqui estamos criando um namespace, para ser nosso container de classes e métodos referentes à manipulação e gerenciamento de dados.
Você pode criar um novo script na Unity com o nome "Data", remover todas as linhas existentes nele, e escrever o código acima para implementar nossa primeira entidade do sistema. Essa entidade será responsável por conter o modelo de dados que iremos querer salvar e carregar no jogo.

Agora vamos supor que você deseje não apenas persistir dados do player, mas também de outros elementos do game, como por exemplo, o inventário.
Poderíamos ter outra classe que modela nossos dados, da seguinte forma:

Código:
namespace Data {
(...)PlayerData

[System.Serializable]
public class InventoryData{
public string equippedItemName;
public Items[] allItems;
public int itemsCount;
}
}

Aqui temos uma situação onde queremos também guardar informações sobre nosso inventário, como item equipado, lista de itens nesse inventário, a quantidade desses itens e etc.

Perceba que agora temos duas classes referente à nossa entidade de modelo de dados, ou seja, temos classes que possuem propósito em comum, mas que não se conectam de forma alguma em nossa estrutura arquitetural. 

Para fazer com que todas as classes em nosso jogo, que forem modelos de dados, sejam um tipo em comum, vamos buscar pela nossa segunda entidade: A classe abstrata.

Classe Abstrata:

A nossa classe abstrata aqui, entra com o papel de fornecer um tipo em comum para todas as nossas classes de modelagem de dados (PlayerData e InventoryData, por exemplo).
Não só isso, como também estabelece um contrato de comportamento para a implementação da ID do nosso objeto.
Esse ID é muito importante para que possamos diferenciar cada instância da classe para que sejam únicos, onde esse ID pode posteriormente ser utilizado como chave do PlayerPrefs, também.

Nós vamos implementar nossa classe abstrata (que chamarei de "PersistentData") com um campo privado, porém, que pode ser serializado (Utilizando o atributo UnityEngine.SerializableField), e que vai ser acessado externamente através de uma propriedade somente leitura. Como mencionado, esse campo será referente ao ID do objeto.

Vamos também criar um método que seta o valor do ID internamente. Esse ID será do tipo "string".
A implementação da nossa segunda entidade ficará assim:

Código:
namespace Data
{
    public abstract class PersistentData
    {
        [UnityEngine.SerializeField]
        private string id;

        public string ID
        {
            get { return id; }
            private set { id = value; }
        }

        protected void SetID(string id) => this.id = id;
    }

(...) PlayerData
(...) InventoryData
}

Perceba que o modificador "private" foi setado ao nosso acessor "set", pois esse valor só poderá ser atribuído pela classe abstrata, internamente e uma única vez.

Após ter feito isto, podemos fazer com que nossas classes concretas herdem de "PersistentData":

Código:
public class PlayerData : PersistentData {...}
public class InventoryData : PersistentData {...}

Nosso último passo é definir o construtor de nossas classes, pois é através do construtor que iremos ter acesso ao valor da chave de nosso objeto instanciado. Esse construtor será responsável por chamar o método "SetID" implementado na classe abstrata, onde é responsável por setar o valor do id da nossa instância.

Código:
public PlayerData(string id) => SetID(id);
*Implementação na classe "PlayerData", mas você deve fazer o mesmo para todas as outras classes de modelagem!

Neste ponto estamos preparados para implementar nossa terceira entidade: o gerenciador de dados.

Gerenciador de dados:

Nossa terceira entidade será responsável por realizar de fato as ações de save/load e também de limpeza de dados no nosso jogo.
Para criar essa entidade, crie um novo script no projeto, chame-o de "DataManager", e exclua todas as linhas nesse script.

Nós vamos declarar uma classe static que vai possuir os seguintes métodos:


  • Save -> Para salvar alguma estrutura de dados, passada como argumento;
  • Load -> Para carregar um objeto anteriormente salvo, passando a chave (id) desse objeto como argumento;
  • DeleteKey -> Método para deletar/resetar algum dado salvo, passando a chave do objeto a ser apagado, como argumento;
  • DeleteAll -> Método responsável por deletar/resetar todos os objetos salvos em nosso jogo;
  • HasKey -> Método simples que vai validar se uma chave existe em nosso PlayerPrefs, passando essa chave como argumento.


Vamos primeiramente implementar nosso método "Save", utilizando os conceitos que vimos nos tópicos anteriores, sobre serialização:

Código:
using UnityEngine;

namespace Data
{
    public static class DataManager
    {
        public static void Save(PersistentData persistentData)
        {
            string keyToSave = persistentData.ID;

            if (keyToSave == null)
                return;

            string objectToSave = JsonUtility.ToJson(persistentData);

            PlayerPrefs.SetString(keyToSave, objectToSave);
            PlayerPrefs.Save();
        }
}


Aqui temos um método bem parecido com o que tínhamos no tópico anterior, onde serializamos um simples texto; com a diferença de que passamos pro método "ToJson" um objeto do tipo "PersistentData" para ser serializado, ou seja, pode ser tanto um objeto "PlayerData" quanto "InventoryData" para ser salvo, ou qualquer outra estrutura de dado que você precise salvar, desde que seja subclasse de "PersistentData".

Agora vamos implementar nosso processo de desserialização através do método Load:

Código:
public static T Load<T>(string dataKey) where T : PersistentData
        {
            string loadedData = PlayerPrefs.GetString(dataKey);

            return JsonUtility.FromJson<T>(loadedData);
        }

Nosso método de Load é muito semelhante ao que tínhamos para nosso processo de carregamento de dados no tópico anterior, ele utiliza o PlayerPrefs com a chave para resgatar a string serializada, e usa esse valor para desserializar essa classe.

Porém, note que temos um retorno do tipo "T" aqui. Ou seja, nós estamos dizendo que esse método pode retornar qualquer tipo, no caso, o tipo que vai ser declarado em sua chamada. Fizemos isso através dos tipos genéricos do C#.
Para não virar bagunça, nós também estipulamos que esse tipo genérico deverá obrigatoriamente ser subclasse de "PersistentData", através de uma restrição de tipo, ou seja, esse tipo pode ser qualquer tipo, desde que seja uma classe filha de "PersistentData", em nosso caso, podendo ser aqui um objeto do tipo "PlayerData" ou "InventoryData".

Por fim, nosso método retorna o objeto desserializado.

O método de deletar um dado salvo é bem simples de implementar, vide:

Código:
public static void DeleteData(string key)
        {
            if(HasKey(key))
            PlayerPrefs.DeleteKey(key);
        }

Aqui, solicitamos uma chave como parâmetro, validamos se essa chave é existente no PlayerPrefs, e por fim, através do método "DeleteKey" do PlayerPrefs, deletamos essa chave, com seu valor salvo.

Para implementar o método "DeleteAll" é mais simples ainda:

Código:
public static void DeleteAllData() => PlayerPrefs.DeleteAll();

Por fim, nosso último método dessa entidade, o método de validar a chave existente:

Código:
public static bool HasKey(string key)
        {
            return !string.IsNullOrEmpty(PlayerPrefs.GetString(key));
        }

Depois de tudo isso, estamos prontos para implementar nossa última entidade: o controlador de dados.

Controlador de dados:

Aqui teremos uma classe que será MonoBehaviour e que vai estar num gameobject da cena, fazendo o papel de controlador de dados.
Essa classe vai de fato chamar as ações de carregamento e salvamento de dados e consumir essas informações para algum propósito no jogo.

Vou utilizar como exemplo a questão de gerenciar os dados do player com o PlayerData.

Crie um novo script com o nome "PlayerManager" e anexe ele a um object da cena.

Para consumir os dados salvos/carregados, vamos criar uma variável do tipo "PlayerData" e também uma chave para esse objeto:

Código:
using UnityEngine;
using Data;

public class PlayerManager : MonoBehaviour{
public PlayerData _playerData {get; private set;}
private const string playerDataKey = "PLAYER_DATA_";
}

Podemos criar um método para atualizar os dados do nosso objeto:

Código:
private void UpdateData(){
_playerData.name = "May";
_playerData.health += 5;
_playerData.score += 100;
_playerData.coins += 200;

DataManager.Save(_playerData);

ShowDataOnConsole();
}

Aqui estamos setando valores arbitrários para nosso objeto, note também que ao final chamamos o método "Save" do DataManager, passando nosso objeto como argumento, para que possamos salvar esses valores atualizados. 
*É importante que sempre que você modifique os valores de uma estrutura persistente, você a salve logo em seguida!

Também chamamos um método chamado "ShowDataOnConsole" que eu vou utilizar para mostrar as informações salvas, no console:

Código:
private void ShowDataOnConsole()
    {
            Debug.Log($"Nome do jogador: {_playerData.name}");
            Debug.Log($"Pontos do jogador: {_playerData.score}");
            Debug.Log($"Vida do jogador: {_playerData.health}");
            Debug.Log($"Moedas do jogador: {_playerData.coins}");

            Debug.Log("================");

    }

Para carregar os dados do nosso player, podemos realizar essa ação toda vez que nossa classe é ativada!
Vamos chamar o método "Load" de "DataManager" e passar a nossa chave, fazendo com que o resultado retornado seja armazenado em nosso objeto "_playerData":

Código:
private void OnEnable(){
_playerData = new PlayerData(playerDataKey);

if(DataManager.HasKey(playerDataKey))
_playerData = DataManager.Load<PlayerData>(playerDataKey);
}

Então primeiramente nós instanciamos esse objeto, passando como construtor a chave do PlayerPrefs, e depois fazemos uma validação se essa chave existe dentro do PlayerPrefs, se sim, chamamos o "Load" passando qual o tipo que queremos retornar, no caso aqui, o PlayerData, e passando como argumento a chave. O valor retornado vai ser colocado dentro da variável "_playerData", restaurando para si, os valores armazenados.

Agora podemos testar tudo isso, fazendo dentro do Update com que quando apertamos o botão "Space" no teclado, os valores sejam atualizados:

Código:
private void Update(){
if(Input.GetKeyDown(KeyCode.Space))
UpdateData();
}


Agora podemos dar play no jogo e testar tudo isso.
Pressione "space" no teclado, e veja no console os valores sendo incrementados.
Agora pare o teste; defina esse método aqui:

Código:
private void Awake() => ShowDataOnConsole();

Salve o script e assim que você entrar no jogo, os valores salvos serão mostrados corretamente no console.

Salvando múltiplos objetos:

Caso tenha a necessidade de salvar mais de um objeto em seu jogo, como por exemplo, os dados de dois jogadores num multiplayer local, você pode utilizar um array de dados para isso, e a manipulação das chaves pode ser um pouco diferente, segue:

Código:
using UnityEngine;
using Data;

public class PlayerManager : MonoBehaviour
{
    public PlayerData[] playersData = new PlayerData[2];

    private void Awake()
    {
        InitializePlayers();

        ShowDataOnConsole();
    }

private void Update(){
if(Input.GetKeyDown(KeyCode.Space))
UpdateData();
}

    private void InitializePlayers()
    {
        for (int i = 0; i < playersData.Length; i++)
        {
            string playerKey = playerDataKey + i.ToString();

            playersData[i] = new PlayerData(playerKey);

            if (DataManager.HasKey(playerKey))
            {
                playersData[i] = DataManager.Load<PlayerData>(playerKey);
            }
        }

    }

    public void UpdateData()
    {
        playersData[0].name = "May";
        playersData[0].health += 5;
        playersData[0].score += 200;
        playersData[0].coins += 100;

        playersData[1].name = "May2";
        playersData[1].health += 10;
        playersData[1].score += 400;
        playersData[1].coins += 200;

        DataManager.Save(playersData[0]);
        DataManager.Save(playersData[1]);

        ShowDataOnConsole();
    }

    private void ShowDataOnConsole()
    {
        foreach (PlayerData _playerData in playersData)
        {
            Debug.Log($"Nome do jogador: {_playerData.name}");
            Debug.Log($"Pontos do jogador: {_playerData.score}");
            Debug.Log($"Vida do jogador: {_playerData.health}");
            Debug.Log($"Moedas do jogador: {_playerData.coins}");

            Debug.Log("================");
        }

    }
}

Nesse exemplo, a chave será o texto "PLAYER_DATA_" + o índice do array daquele objeto, ou seja, para o primeiro player teremos a chave: "PLAYER_DATA_0" e para o segundo: "PLAYER_DATA_1".
Perceba em seus testes, que ao carregar os dois objetos, cada um é individual um do outro, por possuírem valores distintos. 

Finalização:

E aqui chega ao fim nosso tutorial! Veja que essa arquitetura além de ser simples de ser implementada, lhe faz economizar código, pois com ela você centraliza as ações de salvar, carregar e deletar dados, tudo numa mesma classe, precisando apenas chamar esses comandos onde achar necessário.
Você pode também utilizar todo esse sistema para diversos tipos de dados diferentes que necessitar, basta criar sua classe de modelagem e fazer ela herdar de "PersistentData".
Esse sistema também é apto para lidar com múltiplas instâncias, igual vimos no exemplo de querer armazenar os valores de dois jogadores diferentes! 
Espero que tenham gostado dessa série de tutoriais sobre serialização e manipulação de dados em armazenamento, acredito que eu tenha ajudado em algo com todos esses conceitos aqui apresentados.

E digo mais: Se estes tópicos obtiverem um certo engajamento relevante, posso criar um vídeo sobre tudo isso lá no meu canal.
Então se gostou, deixe seu feedback aqui, nem que seja apenas com um "like" no tópico.  Cool


Última edição por MayLeone em Ter Out 12, 2021 5:22 pm, editado 1 vez(es)
MayLeone
MayLeone
Instrutor

Feminino PONTOS : 2562
REPUTAÇÃO : 541
Áreas de atuação : Unity & C#
Respeito as regras : [TUTORIAL] Salvando e carregando dados com serialização (Parte 3) Aad8pUi

http://compilemosfacil.blogspot.com.br

Ir para o topo Ir para baixo

DÚVIDA Re: [TUTORIAL] Salvando e carregando dados com serialização (Parte 3)

Mensagem por Fagner Sab Out 09, 2021 2:09 pm

Show de bola, eu apoio o vídeo kkk Cheers , posso dar uma sugestão? sei que não vai impedir quem queira trapacear alterando os dados salvos, mas ao menos já dificulta pro Joãozinho que aprendeu a usar Lucky patcher no YouTube, seria bom gerar uma hash Sha256 do dado serializado e guardar também, durante o carregamento poderia gerar novamente a hash do arquivo salvo e comparar com a armazenada, sei que não seria 100% efetivo, mas a grande maioria dos usuários nem saberiam que aquilo é uma hash, muito menos saber gerar uma pra alterar também, seria uma camada extra de dificuldade pra quem tenha más intenções.
Fagner
Fagner
Moderador

Masculino PONTOS : 3296
REPUTAÇÃO : 655
Áreas de atuação : Modelagem 3D, Programação.
Respeito as regras : [TUTORIAL] Salvando e carregando dados com serialização (Parte 3) Aad8pUi

Ir para o topo Ir para baixo

DÚVIDA Re: [TUTORIAL] Salvando e carregando dados com serialização (Parte 3)

Mensagem por igorobm Sab Out 09, 2021 6:55 pm

@Fagner escreveu:Show de bola, eu apoio o vídeo kkk Cheers , posso dar uma sugestão? sei que não vai impedir quem queira trapacear alterando os dados salvos, mas ao menos já dificulta pro Joãozinho que aprendeu a usar Lucky patcher no YouTube, seria bom gerar uma hash Sha256 do dado serializado e guardar também, durante o carregamento poderia gerar novamente a hash do arquivo salvo e comparar com a armazenada, sei que não seria 100% efetivo, mas a grande maioria dos usuários nem saberiam que aquilo é uma hash, muito menos saber gerar uma pra alterar também, seria uma camada extra de dificuldade pra quem tenha más intenções.
Não acho que vá ajudar, pois tanto o lucky Patcher e o cheat engine, usam os valores na memoria, vulgo endereço de memoria. Ler na leitura ou na escrita e o de menos.
igorobm
igorobm
MembroAvançado

Masculino PONTOS : 1687
REPUTAÇÃO : 37
Idade : 24
Áreas de atuação : Unity, programação C#.
Respeito as regras : [TUTORIAL] Salvando e carregando dados com serialização (Parte 3) WvDYdlf

Ir para o topo Ir para baixo

Ir para o topo


 
Permissões neste fórum
Você não pode responder aos tópicos