Skip to content
Snippets Groups Projects

Generalize samplers to arbitrary function of machine

Merged Vicentini Filippo requested to merge improve_sampler into master

Created by: gcarleo

This PR allows to sample from other functions of the machine. In general it is now possible to sample from F(machine(x)), for arbitrary real-valued F. By default, F(X)=|X|^2. e.g. Sampling from |machine(x)| is useful when dealing with density matrices for example.

Merge request reports

Merged by avatar (Apr 3, 2025 2:59am UTC)

Loading

Activity

Filter activity
  • Approvals
  • Assignees & reviewers
  • Comments (from bots)
  • Comments (from users)
  • Commits & branches
  • Edits
  • Labels
  • Lock status
  • Mentions
  • Merge request status
  • Tracking
  • requested review from @filippo.vicentini

  • requested review from @filippo.vicentini

  • Created by: twesterhout

    How about generalising the approach a bit further? Any function F for which std::is_invocable_r<double, F, std::complex<double>>::value == true can be used as MachineNorm. So why not just make std::function<double(std::complex<double>)> a parameter rather than an int denoting order?

  • Created by: gcarleo

    How about generalising the approach a bit further? Any function F for which std::is_invocable_r<double, F, std::complex<double>>::value == true can be used as MachineNorm. So why not just make std::function<double(std::complex<double>)> a parameter rather than an int denoting order?

    I originally wanted to pass just a std::function actually, but the problem is that at some point I needed to compare two functions, to make sure that the initialization of the sampler was correctly done. (see what happens in Sampler/ExactSampler.hpp) and this is not too safe, as far as I understand.... In practical cases however one never needs to sample from crazy functions of the machine, so I settled to just the norm int.

  • Created by: twesterhout

    Well, if a sampler already has order as a member variable, why pass it to Sweep as function argument as well?

    As for weird functions... you can never predict user needs in advance. Nice thing about std::function is that pybind11 can wrap Python callables in std::functions which means you can experiment with different norms without ever touching C++!

  • Created by: gcarleo

    Well, if a sampler already has order as a member variable, why pass it to Sweep as function argument as well?

    That's the thing, I don't want to pass order in the constructor, but only in Sweep, because that's more flexible. However there might be some samplers, like ExactSampler, that can speed up the sampling if they know in advance order, so that's why it is stored as a private variable in there. Notice that this variable is just used to check if the current order requested by Sweep is equal to previous one, if it is not, it recomputes some quantities from scratch, otherwise it can do it faster.

    If I do this with a generic function, I'd need to check if the currently stored function is the same I am asking for when I call Sweep...

  • Created by: twesterhout

    That's the thing, I don't want to pass order in the constructor, but only in Sweep, because that's more flexible.

    Could you, please, give an example when using a member variable is not flexible enough?

    The design I have in mind is having a function as a member variable + a member function allowing you to change it. This functionality can even be implemented in some base class, no need to redo it in every sampler. Then, 1) if you want to change the norm in the middle of a Monte Carlo simulation, just call the setter; and 2) ExactSampler can cache values since it knows the norm beforehand. It also knows when to clear the cache -- whenever someone calls the setter.

  • Created by: gcarleo

    You are right, that's a better solution. I am doing that

  • Vicentini Filippo changed title from Generalize samplers to arbitrary machine norm to Generalize samplers to arbitrary function of machine

    changed title from Generalize samplers to arbitrary machine norm to Generalize samplers to arbitrary function of machine

  • Created by: gcarleo

    @twesterhout this is closer to a final PR, however I have some weird run-time error when exposing the std::function to pybind. Maybe for the moment it's easier not to bind that functionality, and keep it only for C++...

  • requested review from @filippo.vicentini

  • Created by: gcarleo

    @twesterhout , unless there are further objections I am going to merge this tomorrow

  • Vicentini Filippo
  • Vicentini Filippo
  • Vicentini Filippo
  • Vicentini Filippo
  • Vicentini Filippo
  • Vicentini Filippo
  • 86 88 &AbstractSampler::SetVisible,
    87 89 R"EOF(
    88 90 numpy.array: The quantum numbers being sampled,
    89 and distributed according to $$|\Psi(v)|^2$$ )EOF")
    91 and distributed according to $$F(\Psi(v))$$ )EOF")
    90 92 .def_property_readonly("acceptance", &AbstractSampler::Acceptance, R"EOF(
    91 93 numpy.array: The measured acceptance rate for the sampling.
    92 94 In the case of rejection-free sampling this is always equal to 1. )EOF")
    93 95 .def_property_readonly("hilbert", &AbstractSampler::GetHilbert, R"EOF(
    94 96 netket.hilbert: The Hilbert space used for the sampling. )EOF")
    95 97 .def_property_readonly("machine", &AbstractSampler::GetMachine, R"EOF(
    96 netket.machine: The machine used for the sampling. )EOF");
    98 netket.machine: The machine used for the sampling. )EOF")
    99 .def_property("machine_func", &AbstractSampler::GetMachineFunc,
  • Created by: twesterhout

    Review: Changes requested

    @gcarleo thanks! Looks good. Sorry it took so long to review. The code needs just a few small touches.

  • Loading
  • Loading
  • Loading
  • Loading
  • Loading
  • Loading
  • Loading
  • Loading
  • Loading
  • Loading
  • Please register or sign in to reply
    Loading