假设我想每4小时发送一堆电子邮件或重新创建站点地图或其他东西,我在凤凰城或只有Elixir如何做到这一点?


当前回答

通常我们使用Oban来实现这一点,但这取决于任务的优先级。 如果您只想运行一个应该在特定时间段后运行的作业。那么你也可以使用Genserver。

Genservers在应用程序启动时启动。你可以使用周期进程Process.send_after(self(),:work, time)和添加handle_info来处理你想做的工作。当我需要向我的项目添加长轮询时,我使用了这个。

其他回答

我用的是量子库Quantum- Elixir。 遵循以下指示。

#your_app/mix.exs
defp deps do
  [{:quantum, ">= 1.9.1"},  
  #rest code
end



#your_app/mix.exs
def application do
  [mod: {AppName, []},
   applications: [:quantum,
   #rest code         
 ]]
end

#your_app/config/dev.exs
config :quantum, :your_app, cron: [
  # Every minute
  "* * * * *": fn -> IO.puts("Hello QUANTUM!") end
]

所有的设置。运行以下命令启动服务器。

iex -S mix phoenix.server 

通常我们使用Oban来实现这一点,但这取决于任务的优先级。 如果您只想运行一个应该在特定时间段后运行的作业。那么你也可以使用Genserver。

Genservers在应用程序启动时启动。你可以使用周期进程Process.send_after(self(),:work, time)和添加handle_info来处理你想做的工作。当我需要向我的项目添加长轮询时,我使用了这个。

我发现:计时器。send_interval/2与GenServer一起使用比Process更符合人体工程学。Send_after /4(用于接受的答案)。

而不是必须重新安排你的通知每次你处理它,:timer。Send_interval /2设置了一个接收消息的时间间隔——不需要像接受的答案那样不断调用schedule_work()。

defmodule CountingServer do
  use GenServer

  def init(_) do
    :timer.send_interval(1000, :update)
    {:ok, 1}
  end

  def handle_info(:update, count) do
    IO.puts(count)
    {:noreply, count + 1}
  end
end

每1000毫秒(即每秒一次),IntervalServer。handle_info/2将被调用,打印当前的计数,并更新GenServer的状态(计数+ 1),给你输出如下:

1
2
3
4
[etc.]

Quantum很棒,我们在工作中使用它作为phoenix前端的cron替代品,我们还可以实时添加作业,这非常简洁。

有一个简单的替代方案,不需要任何外部依赖:

defmodule MyApp.Periodically do
  use GenServer

  def start_link(_opts) do
    GenServer.start_link(__MODULE__, %{})
  end

  def init(state) do
    schedule_work() # Schedule work to be performed at some point
    {:ok, state}
  end

  def handle_info(:work, state) do
    # Do the work you desire here
    schedule_work() # Reschedule once more
    {:noreply, state}
  end

  defp schedule_work() do
    Process.send_after(self(), :work, 2 * 60 * 60 * 1000) # In 2 hours
  end
end

现在在你们的监督树中:

children = [
  MyApp.Periodically
]

Supervisor.start_link(children, strategy: :one_for_one)