# 事件

**URL:** https://heroiclabs.com/docs/zh/nakama/concepts/events/
**Summary:** 事件能够将数据发送到游戏服务器，并创建需要在服务器后台处理的数据。您可以创建和接收可转发给第三方服务（如分析、广告、应用内购买等）的数据。

---


# 事件

事件是将数据发送到游戏服务器并创建要在服务器后台处理的数据的强大方式。您可以创建和接收可转发给第三方服务（如分析、广告、应用内购买等）的数据。

这对游戏工作室和开发人员来说有很多好处：

- 它减少了客户端SDK的数量，从而缩减了游戏客户端；这对于提高FTUE和下载游戏的玩家数量非常重要；
- 游戏客户端使用的各种第三方服务的网络流量更少；
- 它大幅减少了集成时间和开发团队所花费的主动维护成本。

通过使用客户端或服务器端函数发出的事件来实现游戏分析或LiveOps是一个很好的用例。这些事件将在Nakama服务器后台处理。玩家的游戏体验受到最小的干扰，开发人员可以根据分析反馈优化和改进游戏。

在内部，该功能通过以下方式实现：高性能的循环缓冲器，用于存储所收到的事件，以及消费者（事件处理程序）工作池，用于确保事件处理程序无法跟上的情况下，所收到的大量事件不会使服务器超载。

## 生成事件

事件可以由游戏客户端发送至服务器，也可以在服务器上创建。一个事件有一个名称和一个属性映射，属性映射用更多信息装饰事件。

### 发送事件

使用事件API发送到服务器。

{{< code type="client" >}}
```bash
curl -vvv -H "Authorization: Bearer $SESSION"  http://127.0.0.1:7350/v2/event -d '{"name": "my_event", "properties": {"my_key": "my_value"}}'
```
{{< / code >}}

{{< code type="client" framework="godot3" >}}
```gdscript
yield(client.event_async(session, "my_event", {"my_key": "my_value"}), "completed")
```
{{< / code >}}

{{< code type="client" lang="lua" framework="defold" >}}
```lua
local external = true
local name = "my_event"
local properties = { my_key = "my_value" }
local result = nakama.event(client, external, name, properties)
```
{{< / code >}}

{{< missing type="client" lang="csharp" />}}
{{< missing type="client" lang="cpp" />}}
{{< missing type="client" lang="javascript" />}}
{{< missing type="client" lang="java" />}}
{{< missing type="client" lang="shell" />}}


### 创建事件

使用服务器端模块在您的Go代码中生成事件。

{{< code type="server" >}}
```go
// import "github.com/heroiclabs/nakama-common/api"
// import "github.com/heroiclabs/nakama-common/runtime"

// ctx context.Context, nk runtime.NakamaModule
evt := &api.Event{
    Name:       "event_name"
    Properties: map[string]string{
       "my_key": "my_value",
    },
    External:   true,
}
if err := nk.Event(ctx, evt); err != nil {
    // Handle error.
}
```
{{< / code >}}

{{< code type="server" >}}
```lua
local nk = require("nakama")

local properties = {
  my_key = "my_value"
}
local timestamp = nk.time
local external = false

nk.event("event_name", properties, timestamp, external)
```
{{< / code >}}

{{< code type="server" >}}
```typescript
let properties = {
  my_key: 'my_value'
};

nk.event('event_name', properties);
```
{{< / code >}}

{{< missing type="server" lang="typescript" />}}
{{< missing type="server" lang="lua" />}}

如果需要从服务器的其他功能发送事件，还可以利用服务器运行中的[钩子后](../../server-framework/introduction/hooks/#after-hooks)。例如，当更新用户账户时，您可能需要发送要处理的事件。

{{< code type="server" >}}
```go
func afterUpdateAccount(ctx context.Context, logger runtime.Logger, db *sql.DB, nk runtime.NakamaModule, in *api.UpdateAccountRequest) error {
    evt := &api.Event{
        Name:       "account_updated",
        Properties: map[string]string{},
        External:   false, // used to denote if the event was generated from the client
    }
    return nk.Event(context.Background(), evt)
}

func InitModule(ctx context.Context, logger runtime.Logger, db *sql.DB, nk runtime.NakamaModule, initializer runtime.Initializer) error {
    if err := initializer.RegisterAfterUpdateAccount(afterUpdateAccount); err != nil {
        return err
    }
    return nil
}
```
{{< / code >}}

{{< code type="server" >}}
```lua
local nk = require("nakama")

local function after_update_account(ctx, payload)
  local properties = {
    my_key = "my_value"
  }
  local timestamp = nk.time
  local external = false  -- used to denote if the event was generated from the client
  return nk.event("event_name", properties, timestamp, external)
end

nk.register_req_after(after_update_account, "UpdateAccount")
```
{{< / code >}}

{{< code type="server" >}}
```typescript
function InitModule(ctx: nkruntime.Context, logger: nkruntime.Logger, nk: nkruntime.Nakama, initializer: nkruntime.Initializer) {
  let afterUpdateAccount: nkruntime.AfterHookFunction<void, nkruntime.UserUpdateAccount> = function(ctx: nkruntime.Context, logger: nkruntime.Logger, nk: nkruntime.Nakama, data: void, payload: nkruntime.UserUpdateAccount) {
    let properties = {
      my_key: 'my_value'
    };

    nk.event('event_name', properties);
  }
  initializer.registerAfterUpdateAccount(afterUpdateAccount);
}
```
{{< / code >}}

### 内置事件

服务器将产生内置事件，可以专门针对__SessionStart__和__SessionEnd__操作处理这些事件。当服务器启动新的套接字会话和结束会话时，会发生这些特殊事件。有关如何处理这些事件的信息见下文。

## 处理事件

可以使用在服务器启动时注册到运行时初始值设定项的函数来处理事件。事件的外部字段将被标记为`true`客户端生成的字段。

{{< note "important" >}}
目前尚不支持通过Lua/TypeScript运行时函数处理事件。
{{< / note >}}

{{< code type="server" >}}
```go
func processEvent(ctx context.Context, logger runtime.Logger, evt *api.Event) {
    switch evt.GetName() {
    case "account_updated":
        logger.Debug("process evt: %+v", evt)
        // Send event to an analytics service.
    default:
       logger.Error("unrecognised evt: %+v", evt)
    }
}

// initializer runtime.Initializer
if err := initializer.RegisterEvent(processEvent); err != nil {
    return err
}
```
{{< / code >}}

{{< missing type="server" lang="typescript" />}}
{{< missing type="server" lang="lua" />}}

### 内置事件

内部生成的事件具有自己的注册功能。在服务器上打开新的套接字连接时，会创建SessionStart。

{{< code type="server" >}}
```go
func eventSessionStart(ctx context.Context, logger runtime.Logger, evt *api.Event) {
    logger.Debug("process event session start: %+v", evt)
}

// initializer runtime.Initializer
if err := initializer.RegisterEventSessionStart(eventSessionStart); err != nil {
    return err
}
```
{{< / code >}}

{{< missing type="server" lang="typescript" />}}
{{< missing type="server" lang="lua" />}}

会话的套接字连接被关闭时，会创建SessionEnd事件。套接字可能因多种原因而关闭，但可以观察到它的反应。

{{< code type="server" >}}
```go
func eventSessionEnd(ctx context.Context, logger runtime.Logger, evt *api.Event) {
   logger.Debug("process event session end: %+v", evt)
}

// initializer runtime.Initializer
if err := initializer.RegisterEventSessionEnd(eventSessionEnd); err != nil {
   return err
}
```
{{< / code >}}

{{< missing type="server" lang="typescript" />}}
{{< missing type="server" lang="lua" />}}

## 高级设置

为保护服务器的性能，事件处理子系统的设计限制分配给处理事件的资源。您可以使用这些配置设置调整分配给此子系统的资源。

| 配置键 | 值 | 描述 |
| ----------------- | ----- | ----------- |
| runtime.event_queue_size | int | 事件队列缓冲的大小。默认值为65536。 |
| runtime.event_queue_workers | int | 用于并发处理事件的工作线程的数量。默认值为8。 |

要查看其他配置设置，请查看[这些](../../getting-started/configuration/)文档。

## 示例

这是完整的插件示例，它处理SessionStart、SessionEnd以及用户账户更新时产生的事件。

{{< code type="server" >}}
```go
package main

import (
    "context"
    "database/sql"
    "github.com/heroiclabs/nakama-common/api"
    "github.com/heroiclabs/nakama-common/runtime"
)

func afterUpdateAccount(ctx context.Context, logger runtime.Logger, db *sql.DB, nk runtime.NakamaModule, in *api.UpdateAccountRequest) error {
    evt := &api.Event{
        Name:       "account_updated",
        Properties: map[string]string{},
        External:   false,
    }
    return nk.Event(context.Background(), evt)
}

func processEvent(ctx context.Context, logger runtime.Logger, evt *api.Event) {
    switch evt.GetName() {
    case "account_updated":
        logger.Debug("process evt: %+v", evt)
        // Send event to an analytics service.
    default:
        logger.Error("unrecognised evt: %+v", evt)
    }
}

func eventSessionEnd(ctx context.Context, logger runtime.Logger, evt *api.Event) {
   logger.Debug("process event session end: %+v", evt)
}

func eventSessionStart(ctx context.Context, logger runtime.Logger, evt *api.Event) {
    logger.Debug("process event session start: %+v", evt)
}

//noinspection GoUnusedExportedFunction
func InitModule(ctx context.Context, logger runtime.Logger, db *sql.DB, nk runtime.NakamaModule, initializer runtime.Initializer) error {
    if err := initializer.RegisterAfterUpdateAccount(afterUpdateAccount); err != nil {
       return err
    }
    if err := initializer.RegisterEvent(processEvent); err != nil {
        return err
    }
    if err := initializer.RegisterEventSessionEnd(eventSessionEnd); err != nil {
       return err
    }
    if err := initializer.RegisterEventSessionStart(eventSessionStart); err != nil {
       return err
    }
    logger.Info("Server loaded.")
    return nil
}
```
{{< / code >}}

{{< missing type="server" lang="typescript" />}}
{{< missing type="server" lang="lua" />}}