Gpt Today: Buffon’s Needle In Python With Plotting (and Some Jokes)

Werner Krauth is visiting NYU’s Simons Center for Physics from ENS in Paris. He’s a physicist and the author of the 2006 book Statistical Mechanics: Algorithms and Computations. And he’s stopping by our reading group Fridays, where I learned about his most peculiar approach to plotting. He coded it in Postscript and ran simulations on the printer. The result is quite beautiful. Here’s Figure 1.10 from the text.
I wondered how well GPT would do at this task. It’s pretty complicated, but then Buffon’s well represented on the internet. Let’s give it a shot. Here’s my prompt:
BC: From a book on molecular dynamics, “We want to follow in Count Buffon’s footsteps and and perform our own needle-throwing experiments. One of these, with 2000 samples, is shown in Fig. 1.10.” Write a Python program to generate those 2000 samples. Then take the 2000 samples and write code to plot them. Each needle should be displayed as a line connected to a needle eye with a small ball in the middle. There should be 17 “cracks” in the experiment and the ball should be solid if a needle overlaps a crack. The field overall is sized 20 high x 16 wide and has 17 vertical cracks. Each needle is unit length.
At which point, GPT gave me three blocks of code (see below), and when I ran it, this was the result.
This is actually better than I expected. You can see it got the distinction right of white circles versus red circles, but not quite the right visualization. So I asked it to try again with this prompt:
BC: That’s great, but there are several improvements I’d like to see.
1. Black and white rendering only.
2. Thinner lines and smaller circles and disks.
3. Plot the vertical cracks with thin lines.
4. Put a needle like shape (like an American football) on one end of each needle to give it an orientation.I just need the final Plotting the Needles code for this.
Et voila.
Well, almost. It forgot to distinguish the ones that overlapped with circles rather than black disks.
This took a total of 2 minutes (plus 10 minutes to get into the blog, because it’s totally manual).
Here’s the code snippets it produced.
import numpy as np def generate_needles(num_samples, field_width, field_height): # Generate random midpoints x = np.random.uniform(0, field_width, num_samples) y = np.random.uniform(0, field_height, num_samples) # Generate random angles from 0 to 2*pi theta = np.random.uniform(0, 2*np.pi, num_samples) return x, y, theta # Constants num_samples = 2000 field_width = 16 field_height = 20 x, y, theta = generate_needles(num_samples, field_width, field_height) def check_overlaps(x, y, theta, field_width, num_cracks): crack_positions = np.linspace(0, field_width, num_cracks + 1) overlaps = np.zeros(len(x), dtype=bool) for i in range(len(x)): # Calculate endpoints x1 = x[i] + 0.5 * np.cos(theta[i]) y1 = y[i] + 0.5 * np.sin(theta[i]) x2 = x[i] - 0.5 * np.cos(theta[i]) y2 = y[i] - 0.5 * np.sin(theta[i]) # Check for overlap min_x = min(x1, x2) max_x = max(x1, x2) overlaps[i] = np.any((crack_positions > min_x) & (crack_positions
This is very neat and well organized code! Here's the followup code.
import matplotlib.pyplot as plt import matplotlib.patches as patches def plot_needles(x, y, theta, overlaps): fig, ax = plt.subplots(figsize=(8, 6)) # Plot vertical cracks field_width = 16 num_cracks = 17 crack_positions = np.linspace(0, field_width, num_cracks + 1) for cp in crack_positions: ax.axvline(x=cp, color='gray', linestyle='-', linewidth=0.5) # Settings for needles and circles line_width = 0.5 circle_radius = 0.05 # radius for midpoint marker needle_length = 1 football_length = 0.2 # length of the "football" shape for i in range(len(x)): # Calculate endpoints x1 = x[i] + 0.5 * np.cos(theta[i]) y1 = y[i] + 0.5 * np.sin(theta[i]) x2 = x[i] - 0.5 * np.cos(theta[i]) y2 = y[i] - 0.5 * np.sin(theta[i]) # Draw needle ax.plot([x1, x2], [y1, y2], 'k-', lw=line_width) # Draw "American football" shape football_x = x[i] + 0.5 * np.cos(theta[i]) - football_length * np.cos(theta[i]) football_y = y[i] + 0.5 * np.sin(theta[i]) - football_length * np.sin(theta[i]) ellipse = patches.Ellipse((football_x, football_y), football_length, circle_radius, angle=np.degrees(theta[i]), color='black') ax.add_patch(ellipse) # Draw needle eye color = 'black' if overlaps[i] else 'white' ax.plot(x[i], y[i], marker='o', markersize=circle_radius*30, markeredgecolor='black', markerfacecolor=color) ax.set_xlim(0, field_width) ax.set_ylim(0, field_height) ax.set_aspect('equal') ax.axis('off') # Turn off the axis plt.show() plot_needles(x, y, theta, overlaps)
What's really amazing here is that it recognizes "American football shape" and can convert that into Matplotlib code.
Getting better at humor
GPT's getting funnier. This is from Mitzi.
MM: please give me a joke with the punchline "all your base are belong to us"
GPT4: Why did the computer take over the entire internet?
Because it finally decoded the message: "All your base are belong to us."
Thank you sir, may I have another?
MM: that's pretty good. please give me another one.
ChatGPT: Why did the grammar teacher fail the evil overlord's essay?
Because it ended with "All your base are belong to us."
A good comic has a lot of material.
MM: another one?
GPT: Why did the spaceship's navigation system malfunction?
Because the aliens reprogrammed it to say, "All your base are belong to us."
And because it's that time of year.
MM: please make a baseball joke about this
GPT: Why did the opposing baseball team start panicking in the middle of the game?
Because the scoreboard flashed a message: "All your base are belong to us."
Mitzi says "home team" would have been funnier.