मेरे बारे में
नमस्कार, हेब्र! मेरा नाम पावेल है, मैं IoT उपकरणों के उत्पादन में लगी एक कंपनी में तकनीकी निदेशक के रूप में काम करता हूं। हम अपने पेटेंट सेंसर नेटवर्क प्रोटोकॉल पर स्मार्ट होम कंट्रोलर से स्मार्ट मीटरिंग उपकरणों तक - बहुत सी चीजों का उत्पादन करते हैं।
वे आईटी कंपनी के सामान्य निदेशक के रूप में भी कार्य करते हैं। अतीत में, विश्व कप प्रोग्रामिंग के एसीएम ICPC सेमीफाइनलिस्ट।
प्रेरणा
मैं यह लेख इसलिए लिखता हूं क्योंकि हमारी टीम ने आपकी परियोजनाओं में आपके लिए समय बचाने के लिए, डेटाबेस में मान्यता प्राप्त व्यक्तियों को संग्रहीत करने और कुशलता से खोजने के लिए एक समाधान (परीक्षण को लागू करने और लिखने के लिए एक और दो सप्ताह) खोजने के लिए एक महीने के बारे में मार डाला। Spoiler: उन्हें मौजूदा DBMS के लिए कूल प्लग-इन की तरह तैयार कुछ भी नहीं मिला, लेकिन समय-सीमा समाप्त हो रही थी, इसलिए हमने इस कार्य के लिए अपना स्वयं का DBMS लिखा (बड़ी मात्रा में फेस एम्बेडिंग संग्रहीत)। मेरा लेख किसी भी तरह से एक संपूर्ण मार्गदर्शक होने का दावा नहीं करता है, लेकिन मुझे आशा है कि यह हमारे विचारों के आगे के अध्ययन और विकास के लिए एक प्रारंभिक बिंदु प्रदान करेगा।
एंबेडिंग एक पूर्वनिर्धारित आयाम के साथ एक निरंतर वेक्टर में वर्गीकृत विशेषताओं के असतत वेक्टर से एक मानचित्रण है।
तो यह सब कहां से शुरू हुआ
- , , , / , . , , , , . "" 87%, , , . , 3 . . 2-3 . — … , , - "" ( ) 6 . .
, , : ( ) , , , , , , . 10 200 , . , 50 . 10.
, +- , .
, , , dlib, . C++, BLAS, Python CPU. .
, dlib , 0.6, . 128.
, . , , , , , . , , k , k , , - -. , -, -.
.
. dlib .
def get_img_vector(img):
    dets = detector(img, 1)
    for k, d in enumerate(dets):
        shape = sp(img, d)
        return facerec.compute_face_descriptor(img, shape)
    return None
def prepare_database():
    database = {}
    for file in glob.glob("_images/*"):
        identity = os.path.splitext(os.path.basename(file))[0]
        img = cv2.imread(file, 1)
        database[identity] = get_img_vector(img)
    return database
def who_is_it(img, shape, database):
    face_descriptor1 = facerec.compute_face_descriptor(img, shape)
    min_dist = 100
    identity = None
    for (name, db_enc) in database.items():
        dist = distance.euclidean(db_enc, face_descriptor1)
        if dist < min_dist:
            min_dist = dist
            identity = name
    print(min_dist)
    if min_dist > 0.57:
        return None
    else:
        return str(identity)
if __name__ == "__main__":
    global sp
    sp = dlib.shape_predictor('weights/shape_predictor_face_landmarks.dat')
    global facerec
    facerec = dlib.face_recognition_model_v1('weights/dlib_face_recognition_resnet_model_v1.dat')
    global detector
    detector = dlib.get_frontal_face_detector()
    database = prepare_database()
    webcam_face_recognizer(database)
webcam_face_recognizer ( cv2- ) . who_is_it, . , , , , !
, 1 . (N*k), N — , k — . , , . , , - . .
, ?
— , , . — . , , , L2 .
scores = np.linalg.norm(face_descriptor1 - np.asarray(database.items()), axis=1)
min_el_ind = scores.argmax()
, , , .
, , nmslib. HNSW k . , . :
import nmslib
index = nmslib.init(method='hnsw', space='l2', data_type=nmslib.DataType.DENSE_VECTOR)
for idx, emb in enumerate(database.items()):
    index.addDataPoint(idx, emb)
index_params = {
    'indexThreadQty': 5,
    'skip_optimized_index': 0,
    'post': 3,
    'delaunay_type': 2,
    'M': 100,
    'efConstruction': 2000
}
index.createIndex(index_params, print_progress=True)
index.saveIndex('./db/database.bin')
HNSW .
"" . ?
, 4 . dlib , .

, . , , . .
postgresql
- , , (, . ) .
:
import postgresql
def setup_db():
    db = postgresql.open('pq://user:pass@localhost:5434/db')
    db.execute("create extension if not exists cube;")
    db.execute("drop table if exists vectors")
    db.execute("create table vectors (id serial, file varchar, vec_low cube, vec_high cube);")
    db.execute("create index vectors_vec_idx on vectors (vec_low, vec_high);")
:
query = "INSERT INTO vectors (file, vec_low, vec_high) VALUES ('{}', CUBE(array[{}]), CUBE(array[{}]))".format(
            file_name,
            ','.join(str(s) for s in encodings[0][0:64]),
            ','.join(str(s) for s in encodings[0][64:128]),
        )
db.execute(query)
:
import time
import postgresql
import random
db = postgresql.open('pq://user:pass@localhost:5434/db')
for i in range(100):
    t = time.time()
    encodings = [random.random() for i in range(128)]
    threshold = 0.6
    query = "SELECT file FROM vectors WHERE sqrt(power(CUBE(array[{}]) <-> vec_low, 2) + power(CUBE(array[{}]) <-> vec_high, 2)) <= {} ".format(
        ','.join(str(s) for s in encodings[0:64]),
        ','.join(str(s) for s in encodings[64:128]),
        threshold,
    ) + \
            "ORDER BY sqrt(power(CUBE(array[{}]) <-> vec_low, 2) + power(CUBE(array[{}]) <-> vec_high, 2)) ASC LIMIT 1".format(
                ','.join(str(s) for s in encodings[0:64]),
                ','.join(str(s) for s in encodings[64:128]),
            )
    print(db.query(query))
    print('inset time', time.time() - t, 'ind', i)
10^5 (4- i5, 2,33 GHz) 0.034 .
? +- ( ). , , … , .
K-d
- , , " ", . , .
, , , .
K-d — k- . , ( , . .), .
, :
.
(N) , , , (N). , !
"" , NDA.
. , .
15 . , . . .
— . , , :)
k-tree , ( ), , ( -), 4-6 . .
, . — 1,5 , , , , , , . , , .
, , , . , mail cloud. , .
यदि इस समस्या को हल करने के लिए अन्य विकल्प हैं, तो मैं ख़ुशी से टिप्पणियों में उनके बारे में पढ़ूंगा।
और इस कल्पित का नैतिक यह है - भीड़ भी गिर गई शेरखुले स्रोत के समाधान समय के अनुसार परीक्षण किए गए। एल्गोरिदम, पृष्ठन सीखें :)