Monday, July 30, 2018

quantum chemistry - How to compute 2-electron integral for Hartree-Fock code?


I'm working on writing my own code for HX2 in an STO-3G basis set using Hartree-Fock (HF), and I am currently stuck on how to construct the two-electron integral matrix. I know how to evaluate the primitive two-electron integrals, but I can't figure out how to transform that into the 4×4 matrix of possible values for this system.


For my previous integrals, I determined the sum of the primitives by constructing a 6×6 matrix of the primitive integrals, and did a transformation using a 6×2 matrix of the coefficient values from the fit to the Slater function, for example: D=[d110d120d1300d210d220d23]S=DSprimD


I tried doing a similar technique for the two-electron integrals but with a 12×12 matrix, but the values didn't come out right. It seems that I need to make a 4-dimensional array of all the primitive integrals, but I don't know how to evaluate that in order to determine the sum and hence the final

4×4 matrix.


I've been using Szabo and Ostlund's Modern Quantum Chemistry book as my main reference.



Answer



You can also write this down in a similar way as for the one-electron integral. You already had:


S=D2SprimD2


where S is (2×2), D2 is (6×2) and Sprim is (6×6).


For the 2-electron integral we now need


V=D4VprimD4


where V is (2×2×2×2), D4 is (6×2×6×2) and Vprim is (6×6×6×6).


You can get D4 from D2: D4=D2D2

where you do not sum over any index in order to get the 4-index tensor.



The equation for V is a bit more tricky, since you need to pay attention over which indices you sum (which is actually not specified by the notation). I add an example on how you can do this with Python/Numpy with Einstein summations.


# 1-electron integral
S = numpy.einsum('ba,bc,cd', D2, Sprim, D2)

# 2-electron integral Option 1: form (6x2x6x2) tensor
D4 = numpy.tensordot(D2, D2, 0)
V = numpy.einsum('badc,bedf,egfh->agch', D4, Vprim, D4)

# 2-electron integral Option 2: transform indices one-by-one
V = Vprim.copy()

V = numpy.einsum('ba,bcde->acde', D2, V) # i
V = numpy.einsum('abcd,be->aecd', V, D2) # j
V = numpy.einsum('da,bcde->bcae', D2, V) # k
V = numpy.einsum('abcd,de->abce', V, D2) # l

You will need to give D2, Sprim and Vprim as input.


(If anyone knows how to write this down in a formal notation, please edit.)


You were asking how to get V as a 2-index (4×4) matrix. For this we can simply reshape the above 4-index tensors. So Vprim becomes (36×36). Now this means D4 needs to be (36×4), and not (12×12). To these reshaped 2-index matrices we can apply again the same transformation equation from above. Here is the code for my example:


# 2-electron integral Option 3: reshape to 2-index matrices
Vprim = Vprim.reshape((Nprim**2, Nprim**2))

D4 = numpy.swapaxes(D4, 1, 2) # making sure to combine the correct axes
D4 = D4.reshape((Nprim**2, Ncntr**2))
V = numpy.einsum('ba,bc,cd', D4, Vprim, D4)

Now V will be: [(ϕ1ϕ1|ϕ1ϕ1)(ϕ1ϕ1|ϕ1ϕ2)(ϕ1ϕ1|ϕ2ϕ1)(ϕ1ϕ1|ϕ2ϕ2)(ϕ1ϕ2|ϕ1ϕ1)(ϕ1ϕ2|ϕ1ϕ2)(ϕ1ϕ2|ϕ2ϕ1)(ϕ1ϕ2|ϕ2ϕ2)(ϕ2ϕ1|ϕ1ϕ1)(ϕ2ϕ1|ϕ1ϕ2)(ϕ2ϕ1|ϕ2ϕ1)(ϕ2ϕ1|ϕ2ϕ2)(ϕ2ϕ2|ϕ1ϕ1)(ϕ2ϕ2|ϕ1ϕ2)(ϕ2ϕ2|ϕ2ϕ1)(ϕ2ϕ2|ϕ2ϕ2)]


And here is the structure of D4: D4=D2D2=[d11D20D2d12D20D2d13D20D20D2d21D20D2d22D20D2d23D2]


No comments:

Post a Comment

periodic trends - Comparing radii in lithium, beryllium, magnesium, aluminium and sodium ions

Apparently the of last four, MgX2+ is closest in radius to LiX+. Is this true, and if so, why would a whole larger shell ($\ce{...