如果我们想要在UE4中读写Json文件,那么我们必须使UE4包含Json和JsonUtilities这两个模块,那么UE4如何添加模块呢?
UE4添加预定义模块的方法很简单,我只需打开工程的.Biuld.cs文件,在其中的PublicDependencyModuleNames.AddRange()
函数中追加两个模块即可,如:
using UnrealBuildTool;
public class DATA_sys : ModuleRules
{
public DATA_sys(ReadOnlyTargetRules Target) : base(Target)
{
PCHUsage = PCHUsageMode.UseExplicitOrSharedPCHs;
PublicDependencyModuleNames.AddRange(new string[] { "Core", "CoreUObject", "Engine", "InputCore" ,"Json","JsonUtilities"});
PrivateDependencyModuleNames.AddRange(new string[] { });
// Uncomment if you are using Slate UI
// PrivateDependencyModuleNames.AddRange(new string[] { "Slate", "SlateCore" });
// Uncomment if you are using online features
// PrivateDependencyModuleNames.Add("OnlineSubsystem");
// To include OnlineSubsystemSteam, add it to the plugins section in your uproject file with the Enabled attribute set to true
}
}
如果我们要添加自定义模块就有点麻烦了,当然这里就不赘述了。
包含这两个模块之后,我们还需要用到 Json.h、JsonObject.h、JsonSerializer.h三个头文件,其中JsonSerializer.h是用于Json序列化与反序列化用的。
至此我们就可以使用UE4自带的工具进行Json文件的读写工作了。
void AMysqlJsonCpp::CreatJson()
{
FString filePath = FPaths::GameContentDir() + TEXT("MysqlConfig/Connect.json");
FString jsonStr;
TSharedRef> jsonWriter = TJsonWriterFactory<>::Create(&jsonStr);
jsonWriter->WriteObjectStart();
jsonWriter->WriteValue(TEXT("server"), TEXT("127.0.0.1"));
jsonWriter->WriteObjectEnd();
jsonWriter->Close();
FFileHelper::SaveStringToFile(jsonStr, *filePath);
}
FPaths::GameContentDir()
返回当前工程目录的Content文件夹的路径,FPaths为UE4的路径读写工具;
FString jsonStr
的作用是用于关联UE4的Json写工具TJsonWriter<>,作为输入流载体,且JsonStr必须是FString类型;
TShareRef
是UE4自身的共享指针类型;
TJsonWriter<>
是UE4专门用于写Json的模板类,其中类型参数通常为TCHAR,其中有多个方法供开发者使用:
函数 | 作用 |
---|---|
Close() | 关闭写工具 |
WriterArrayStart() | 开始一个Json数组 |
WriterArrayEnd() | 结束一个Json数组 |
WriterNull(FString) | 为一个键写一个空值 |
WriterObjectStart() | 开始一个Json对象 |
WriterObjectEnd() | 结束一个Json对象 |
WriterValue(FString,FString/int32/float/bool) | 向Json文件写入键值对 |
TJsonWriterFactory<>::Create(FString)
是UE4用来生成Json写工具TJsonWriter<>的类,TJsonFactory<>只有一个方法就是Create;
FFileHelper
UE4的文件读写工具,方法SaveStringToFile(TJsonWriter*,FString*)
函数的作用就是将Json写工具中的Json数据写到FString字符串中的路径文件中。
使用非序列化方式写入Json时,写入方式需要严格按照Json的语法格式来做,如最开始需要使用WriterObjectStart()创建一个根前括号,即Json语法中最外面一层的{
,所有写入结束后需要使用WriterObjectEnd()声明根对象结束,即Json语法中的最外面一层的}
,同理数组也需要按对象一样的方法进行处理。如此才能写入一个结构完整的Json文本。
void AMyActor::Test()
{
TSharedPtr rootObj = MakeShareable(new FJsonObject());
rootObj->SetStringField("root", "1");
TArray> arrValue;
TSharedPtr tmp = MakeShareable(new FJsonValueString("array"));
arrValue.Add(tmp);
rootObj->SetArrayField("array", arrValue);
FString filePath = FPaths::GameContentDir() + TEXT("MysqlConfig/text.json");
FString jsonStr;
TSharedRef> jsonWriter = TJsonWriterFactory::Create(&jsonStr);
FJsonSerializer::Serialize(rootObj.ToSharedRef(), jsonWriter);
FFileHelper::SaveStringToFile(jsonStr, *filePath);
UE_LOG(LogTemp, Error, TEXT("%s"),*filePath);
}
序列化的写入方式则无需考虑按Json的语法结构进行写入,序列化的写入方式是通过一个FJsonObject对象进行Json文本的写入。
SetArrayField(FString,TArray>)
、SetBoolFiled(FString,bool)
、SetNumberField(FString,Number)
、SetStringField(FString,FString)
、SetObjectField(FString,TSharePtr)
、SetField(FString,TSharePtr)
等函数向FJsonObject对象中写分别入数组、bool值、数字、字符串、对象和Json键值对。其中数组的写入较为麻烦,我们需要先向创建Json键值对类型共享指针的TArray数组TArray>
。并向数组中添加指向FJsonValue对象的共享指针后然后才可以使用SetArrayField进行数组的Json文本写入。FJsonSerializer::Serialize(TSharePtr.ToShareRef(),TSharePtr)
;其中TSharePtr
和非序列化写入一样需要绑定一个FString作为输入流载体。TArray AMysqlJsonCpp::ReadMysqlConnectConfig()
{
FString filePath = FPaths::GameContentDir() + TEXT("MysqlConfig/Connect.json");
if (FPaths::FileExists(filePath))
{
FString server;
FString dbName;
FString userId;
FString passwd;
TArray connectConfig;
FString fileStr;
FFileHelper::LoadFileToString(fileStr, *filePath);
TSharedPtr rootObject = MakeShareable(new FJsonObject());
TSharedRef> jsonReader = TJsonReaderFactory<>::Create(fileStr);
if (FJsonSerializer::Deserialize(jsonReader, rootObject))
{
server = rootObject->GetStringField("server");
dbName = rootObject->GetStringField("dbName");
userId = rootObject->GetStringField("userId");
passwd = rootObject->GetStringField("passwd");
}
connectConfig.Add(FName(*server));
connectConfig.Add(FName(*dbName));
connectConfig.Add(FName(*userId));
connectConfig.Add(FName(*passwd));
}
return TArray();
}
FJsonSerializer::Deserialize(TSharePtr>,TSahrePtr)
将输入流载体的Json数据反序列化到FJsonObject对象中;GetArrayField(FString)
、GetBoolFiled(FString)
、GetNumberField(FString)
、GetStringField(FString)
、GetObjectField(FString)
、GetField(FString)
等方法从Json对象中读取指定键的值了。