Python的随机性

xen*_*oid 70 python random

我正在使用Python使用虚线为点画生成图像。破折号的周期是恒定的,改变的是破折号/空间比。这将产生如下内容:

在此处输入图片说明

但是,在该图像中,虚线具有统一的原点,这会产生难看的垂直装订线。因此,我尝试将原点随机化以去除排水沟。这种工作,但有一个明显的模式:

在此处输入图片说明

我想知道这是从哪里来的,所以我制作了一个非常简单的用虚线堆叠的测试用例:

  • 冲刺率:50%
  • 破折号20px
  • 使用random.uniform(-10.,+10.)(*)(初始值后,原点从-10px变为+ 10pxrandom.seed()

在此处输入图片说明

并增加了随机性:

在此处输入图片说明

因此仍然存在模式。我不明白的是,要获得可见的装订线,您需要使6个或7个连续值落入同一范围内(例如,总范围的一半),这应该是1/64的概率,但是似乎发生了很多通常会在200行中生成。

我误会了吗?仅仅是我们的人类大脑正在看到没有模式的模式吗?是否有更好的方法来生成更“视觉上随机的”东西(python 2.7,最好不安装任何东西)?

(*)在这种情况下部分像素有效

附件:我使用的代码(这是一个Gimp脚本):

#!/usr/bin/env python
# -*- coding: iso-8859-15 -*-

# Python script for Gimp (requires Gimp 2.10)
# Run on a 400x400 image to see something without having to wait too much
# Menu entry is in "Test" submenu of image menubar

import random,traceback
from gimpfu import *

def constant(minShift,maxShift):
    return 0

def triangle(minShift,maxShift):
    return random.triangular(minShift,maxShift)

def uniform(minShift,maxShift):
    return random.uniform(minShift,maxShift)

def gauss(minShift,maxShift):
    return random.gauss((minShift+maxShift)/2,(maxShift-minShift)/2)

variants=[('Constant',constant),('Triangle',triangle),('Uniform',uniform),('Gauss',gauss)]

def generate(image,name,generator):
    random.seed()
    layer=gimp.Layer(image, name, image.width, image.height, RGB_IMAGE,100, LAYER_MODE_NORMAL)
    image.add_layer(layer,0)
    layer.fill(FILL_WHITE)
    path=pdb.gimp_vectors_new(image,name)

    # Generate path, horizontal lines are 2px apart, 
    # Start on left has a random offset, end is on the right edge right edge
    for i in range(1,image.height, 2):
        shift=generator(-10.,10.)
        points=[shift,i]*3+[image.width,i]*3
        pdb.gimp_vectors_stroke_new_from_points(path,0, len(points),points,False)
    pdb.gimp_image_add_vectors(image, path, 0)

    # Stroke the path
    pdb.gimp_context_set_foreground(gimpcolor.RGB(0, 0, 0, 255))
    pdb.gimp_context_set_stroke_method(STROKE_LINE)
    pdb.gimp_context_set_line_cap_style(0)
    pdb.gimp_context_set_line_join_style(0)
    pdb.gimp_context_set_line_miter_limit(0.)
    pdb.gimp_context_set_line_width(2)
    pdb.gimp_context_set_line_dash_pattern(2,[5,5])
    pdb.gimp_drawable_edit_stroke_item(layer,path)

def randomTest(image):
    image.undo_group_start()
    gimp.context_push()

    try:
        for name,generator in variants:
            generate(image,name,generator)
    except Exception as e:
        print e.args[0]
        pdb.gimp_message(e.args[0])
        traceback.print_exc()

    gimp.context_pop()
    image.undo_group_end()
    return;

### Registration
desc="Python random test"

register(
    "randomize-test",desc,'','','','',desc,"*",
    [(PF_IMAGE, "image", "Input image", None),],[],
    randomTest,menu="<Image>/Test",
)

main()
Run Code Online (Sandbox Code Playgroud)

Joh*_*man 46

这样想:直到排水沟被阻塞(或几乎被阻塞),才可以察觉。仅当两条连续的线几乎完全异相时才发生这种情况(第一行中的黑色线段几乎位于下一行中的白色线段上方)。这样的极端情况每10排中只有大约1排发生,因此可见的水槽在被阻塞之前似乎延伸了10排。

换种方式看-如果您打印出图像,则确实存在长长的白色通道,您可以通过这些通道轻松地用笔画一条线。为什么您的大脑不应该感知它们?

为了获得更好的视觉随机性,请找到一种方法,使连续的行相互依存,而不是相互独立,这样几乎会出现异相行为。

  • 确实。即使那样,大脑也很擅长忽略完全异相的一条线。 (7认同)
  • 按照您的推理,最终最好的方法就是完全没有随机性。 (3认同)

Eri*_*nil 27

我们在“随机”图片中看到图案的原因至少有一个显而易见的原因:400x400像素与20x400像素重复20次相同。

在此处输入图片说明

因此,每个明显的动作都平行重复20次,这确实有助于大脑分析图像。

实际上,相同的10px宽的图案会重复40次,黑白交替:

在此处输入图片说明

您可以为每一行分别随机设置破折号周期(例如12和28之间):

在此处输入图片说明

这是相应的代码:

import numpy as np
import random

from matplotlib import pyplot as plt
%matplotlib inline
plt.rcParams['figure.figsize'] = [13, 13]

N = 400

def random_pixels(width, height):
    return np.random.rand(height, width) < 0.5

def display(table):
    plt.imshow(table, cmap='Greys', interpolation='none')
    plt.show()

display(random_pixels(N, N))

def stripes(width, height, stripe_width):
    table = np.zeros((height, width))
    cycles = width // (stripe_width * 2) + 1
    pattern = np.concatenate([np.zeros(stripe_width), np.ones(stripe_width)])
    for i in range(height):
        table[i] = np.tile(pattern, cycles)[:width]
    return table

display(stripes(N, N, 10))

def shifted_stripes(width, height, stripe_width):
    table = np.zeros((height, width))
    period = stripe_width * 2
    cycles = width // period + 1
    pattern = np.concatenate([np.zeros(stripe_width), np.ones(stripe_width)])
    for i in range(height):
        table[i] = np.roll(np.tile(pattern, cycles), random.randrange(0, period))[:width]
    return table

display(shifted_stripes(N, N, 10))

def flexible_stripes(width, height, average_width, delta):
    table = np.zeros((height, width))
    for i in range(height):
        stripe_width = random.randint(average_width - delta, average_width + delta)
        period = stripe_width * 2
        cycles = width // period + 1
        pattern = np.concatenate([np.zeros(stripe_width), np.ones(stripe_width)])
        table[i] = np.roll(np.tile(pattern, cycles), random.randrange(0, period))[:width]
    return table

display(flexible_stripes(N, N, 10, 4))
Run Code Online (Sandbox Code Playgroud)


xen*_*oid 8

将我的最终解决方案发布为答案,但请支持其他人。

约翰·科尔曼(John Coleman)说:

为了获得更好的视觉随机性,请找到一种方法,使连续的行相互依存,而不是相互独立,这样几乎会出现异相行为。

因此,最后,避免混乱的最佳方法是放弃随机性,并采用非常固定的移位方案,而效果很好的方案是4相0.25%,75%,50%周期:

在此处输入图片说明

好的,仍然有轻微的菱形图案,但是它比我尝试的随机方案引入的图案要少得多。


Mar*_*ley 7

这有点违反直觉,但是当您将随机元素加在一起时,随机性就会变小。如果我正确遵循,则每个元素的范围是10px-30px。因此,10个元素的总大小为100像素到300像素,但是分布甚至不在该范围内。极端情况不太可能出现,平均而言,它将接近200像素,因此将出现基本的20像素模式。您的随机分布需要避免这种情况。

编辑:我看到我有点误解了,所有的破折号都是20px具有随机偏移量。因此,我认为查看任意1个垂直破折号看起来都是随机的,但是在页面上重复了相同的随机破折号,形成了图案。

  • 它被称为“中心极限定理”。您可以在[此处](https://towardsdatascience.com/understanding-the-central-limit-theorem-642473c63ad8)上阅读或自行搜索。所以是的,随着您不断地对一组随机数求和,该和将变得越来越[正常](https://en.wikipedia.org/wiki/Normal_distribution)。 (3认同)
  • “随着将随机元素加在一起,随机性变少”:否,但是随着您*平均*越来越多的随机元素,均值的方差减小。总和的方差增加。 (3认同)