苹果手机运行python_iPhone是卖的最好的手机?用Python照样把他玩弄鼓掌之间!

关于 iOS 的技术解读有很多,但是却鲜有设备可视化同步的介绍文章。本文一起了解下这个酷炫的 iOS 黑科技。

152818793145723fe0f22b0

1528187940452324f09ac81

我们的任务很简单——如上图所示,实时获取设备的当前方向。

UIDevice.current.orientation

首先,需要调用

beginGeneratingDeviceOrientationNotifications()

但仅仅这样还不行。因为如果设备上的旋转被锁定了,那么就不会产生以上通知。我的相机应用程序从头到尾都需要知道方向——所以我意识到我需要直接根据设备的加速度计算方向。

1528187972893fb4f6222b3

好了,现在有了这些值,我们该做些什么呢?这是一个较难的部分。如果将所有内容都输出到控制台,那么我们很快就会被大量数据淹没。我认为还是在屏幕上显示这些值比较好。

但是,等等,如果将数值显示在图表上,会怎么样?别想图表了,我们可以来用开源的 Blender 试试,它可以实现这些值的可视化,并且很容易扩展。

152818799860213f5ec97b2

1528188021765d184aff75e

1528188036071f4f149fddc

152818804774424fcebb8d4

然而 Blender 并不是很好的代码编辑器,所以我们还是使用钟爱的外部编辑器吧。为了调用外部文件,我们可以将 print("hi") 替换成以下代码:

import bpy

import os

filename = os.path.join(os.path.dirname(bpy.data.filepath), "server.py")

exec(compile(open(filename).read(), filename, ‘exec‘))

下一步,我们需要在与 .blend 文件相同的文件夹中创建新的 server.py 文件,我们真正的代码就要保存在这里。现在我们可以用任何编辑器打开它,你可以选择 Atom、Sublime,甚至 Word 2007 都行。

1528188077922bdda259c32

1528188110125e5bb3fc403

找到该 Cube 对象,点击右键并选择重命名,重命名为 iPhone。现在让我们再来看一看 server.py。

import socket

import select

import json

import threading

import traceback

class ServerThread(threading.Thread):

def __init__(self):

threading.Thread.__init__(self)

self.running = True

def stopServer(self):

self.running = False

self.server.running = False

def run(self):

try:

self.server = Server()

while self.running:

self.server.receive()

except:

pass

class Server:

def __init__(self):

self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

self.socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)

self.socket.setblocking(False)

self.socket.bind((str(socket.INADDR_ANY), 9845))

self.socket.listen(2)

self.running = True

def __exit__(self, exc_type, exc_value, traceback):

self.socket.close()

def receive(self):

pairs = []

timeout = 1

while self.running:

sockets = list(map(lambda x: x[0], pairs))

if len(pairs) > 0:

read_sockets, write_sockets, error_sockets = select.select(sockets, [], [], timeout)

for sock in read_sockets:

data = sock.recv(4096)

if not data :

print(‘Client disconnected‘)

pairs = []

else :

self.connectionReceivedData(connection, data.decode())

try:

try:

connection,address = self.socket.accept()

print("new connection: ", connection)

pairs.append((connection, address))

except:

pass

except:

pass

for pair in pairs:

(connection, address) = pair

connection.close()

def connectionReceivedData(self, connection, data):

try:

motionData = json.loads(data)

except json.decoder.JSONDecodeError:

print("Invalid JSON: ", data)

return None

receivedMotionData(motionData)

# This is a global so when we run the script again, we can keep the server alive

# but change how it works

import bpy

def receivedMotionData(motionData):

phone = bpy.context.scene.objects["iPhone"]

phone.rotation_quaternion.x = float(motionData[‘x‘])

phone.rotation_quaternion.y = 0 - float(motionData[‘z‘])

phone.rotation_quaternion.z = float(motionData[‘y‘])

phone.rotation_quaternion.w = float(motionData[‘w‘])

pass

try:

if serverThread.running == False:

serverThread = ServerThread()

serverThread.start()

print("Starting server")

else:

print("Server already running, using new motion handler.")

except:

serverThread = ServerThread()

serverThread.start()

print("Starting server")

1528188134330d92d6baef4

检查 Blender,你应该看到 iPhone 根本没有改变。这是因为上面的脚本使用四元组设置了 iPhone 的旋转角度,并且它使用了欧拉角进行旋转。需要做一些修改。将 Python 控制台面切换到 “Properties”,然后单击该面板顶部的橙色立方体图标。中部 Transform 的下面,点击 XYZ Euler 并选择 Quaternion。现在尝试再次运行 client.py。

你应该看到 iPhone 立即翻转过来了。不要惊慌,这就是我们想要的。现在,我们需要让这个模型跟着实际的 iPhone 旋转。

15281881747273b38e450e0

我们需要将运动数据从 iPhone 发送到运行 Blender 的计算机。感谢上苍我们不需要深入到 Swift 中的原始 C 套接字级别,因为 Foundation 具有抽象。

我们可以将以下代码放入新的 iOS 项目中,以替换默认的 ViewController。请确保使用计算机的本地 IP 地址替换 host 变量。

import UIKit

import CoreMotion

class CoreMotionViewController: UIViewController, StreamDelegate {

let motionManager = CMMotionManager()

let queue = OperationQueue()

let host = "192.168.1.2"

override func viewDidLoad() {

super.viewDidLoad()

setUpStreams(host: host)

motionManager.startDeviceMotionUpdates(to: queue) { (data: CMDeviceMotion?, error: Error?) in

guard let data = data else {

print("Error: \(error!)")

return

}

let attitude: CMAttitude = data.attitude

let quaternion = attitude.quaternion

var motionData = MotionData()

motionData.x = quaternion.x

motionData.y = quaternion.y

motionData.z = quaternion.z

motionData.w = quaternion.w

let encoder = JSONEncoder()

do {

let json = try encoder.encode(motionData)

self.send(data: json)

} catch let error {

print("Couldn‘t send data, error: \(error)")

}

}

}

// MARK: - Streams

var inputStream: InputStream?

var outputStream: OutputStream?

func setUpStreams(host: String) {

var readStream: Unmanaged?

var writeStream: Unmanaged?

CFStreamCreatePairWithSocketToHost(kCFAllocatorDefault,

host as CFString, 9845,

&readStream,

&writeStream)

inputStream = readStream!.takeRetainedValue()

outputStream = writeStream!.takeRetainedValue()

guard let inputStream = inputStream, let outputStream = outputStream else {

print("Failed to create streams")

return

}

inputStream.delegate = self

outputStream.delegate = self

inputStream.schedule(in: .current, forMode: .commonModes)

outputStream.schedule(in: .current, forMode: .commonModes)

inputStream.open()

outputStream.open()

}

func send(data: Data) {

guard let outputStream = outputStream else {

return

}

_ = data.withUnsafeBytes {

outputStream.write($0, maxLength: data.count)

}

}

func stream(_ aStream: Stream, handle eventCode: Stream.Event) {

if eventCode == .errorOccurred {

inputStream = nil

outputStream = nil

print("Error: Stream error")

} else if eventCode == .endEncountered {

inputStream = nil

outputStream = nil

print("Error: Encountered end of stream")

}

let maxReadLength = 4096

if eventCode == .hasBytesAvailable {

guard let inputStream = inputStream else {

return

}

while inputStream.hasBytesAvailable {

let buffer = UnsafeMutablePointer.allocate(capacity: maxReadLength)

inputStream.read(buffer, maxLength: maxReadLength)

buffer.deallocate()

}

}

}

}

// MARK: - Data Model

private struct MotionData: Codable {

var x: Double = 0

var y: Double = 0

var z: Double = 0

var w: Double = 0

}

15281882107940010a81870

15281882187904381609f92

那么最终我是如何从移动管理器获取方向信息的?

欢迎关注我的博客或者公众号:https://home.cnblogs.com/u/Python1234/ Python学习交流

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/305002.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

这几个动图告诉你科学的神奇,看完瞬间觉得智商都提高了

生活中简单平常的事物和现象背后,往往有着奇妙的原理,赶快跟着一起来看看涨点知识吧! 夹心雪糕的制作原理 ▼ 难怪雪糕大小,厚度都一模一样 原来都是从一个模子里出来的 ▼ 煎饼可以统一翻面 再也不用担心烤焦了 ▼ 冰淇淋蛋筒的制…

自定义 ocelot 中间件输出自定义错误信息

自定义 ocelot 中间件输出自定义错误信息Introocelot 中默认的 Response 中间件在出错的时候只会设置 StatusCode 没有具体的信息,想要展示自己定义的错误信息的时候就需要做一些自定义了,对 ocelot 中的 Response 中间件做了一些小改动,实现…

盘点小坏蛋的礼物

我们家小坏蛋一天天长大了,妈妈从觉得很辛苦过渡到习惯了很辛苦,苦中作乐的妈妈终于抽出空来写博客啦! 为什么叫他小坏蛋呢?因为他吃饭不乖。没满月的时候蛮乖的,每顿奶都吃的很香,咕咚咕咚的喝下去&#x…

为什么对gRPC做负载均衡会很棘手?

在过去的几年中,随着微服务的增长,gRPC在这些较小的服务之间的相互通信中获得了很大的普及,在后台,gRPC使用http/2在同一连接和双工流中复用许多请求。使用具有结构化数据的快速,轻便的二进制协议作为服务之间的通信介质确实很有吸…

给新手程序猿的16个必备小妙招

写在前面: 这个文章核心并不是程序优化的具体技巧,而是拿到一个问题如何思考和利用工具的通用方法。比如即使我们不知道 profiler 这个东西,通过搜索"代码 每一行 时间"也可以很快知道有这样的工具叫做 profiler,并且学…

python字符串设置字体_python怎么更改字符串后几位

python更改字符串后几位的方法:可以利用replace()函数来实现。replace()函数可以把字符串中的旧字符串替换成新字符串,并返回替换后的新字符串。具体使用方法如:【str.replace("is", "was")】。可以利用replace()函数来修…

程序猿的双十一最佳攻略

讲个恐怖故事 一年一度虐汪“光棍节”又来了 小天拍拍(不存在的)胸脯告诉大家 这个节日 我陪你们 买买买! 限时特惠专场 1 welcome 7天教你学会数学建模及Matlab编程 数学建模涉及的内容比较广泛,比如碎纸片问题中所涉及的图像识…

关于Word中审阅的一个问题!

前两天,在帮一个杂志撰稿的过程中,有一个关于Word审阅的问题。(以下内容以Word2007为例) 本身,审阅的应用不算困难。 我们可以为文章添加“批注”,或是你的上级对文章进行修改,他们可以开启“审…

ASP.NET Core - 在ActionFilter中使用依赖注入

上次ActionFilter引发的一个EF异常,本质上是对Core版本的ActionFilter的知识掌握不够牢固造成的,所以花了点时间仔细阅读了微软的官方文档。发现除了IActionFilter、IAsyncActionFilter的问题,还有一个就是依赖注入在ActionFilter上的使用也是需要注意的…

魏尔斯特拉斯函数与分形图形的动画演示

一般人会直觉上认为连续的函数必然是近乎可导的。即使不可导,所谓不可导的点也必然只占整体的一小部分。根据魏尔斯特拉斯在他的论文中所描述,早期的许多数学家,包括高斯,都曾经假定连续函数不可导的部分是有限或可数的。这可能是…

大脚战场插件怎么关闭_PM工具栏插件:HonmToolBar

视频演示:问:HonmToolBar是一款什么样的插件?答:HonmToolBar是一款高度自由化的插件,用户可以自己增加宏文件按钮或者宏命令按钮。该插件类似工具栏,有水平和垂直两个工具栏。插件悬浮在PM图形窗口左上角&a…

android 导航 美国,变美了 Android N或用全新虚拟导航按键

原标题:变美了 Android N或用全新虚拟导航按键变美了 Android N或用全新虚拟导航按键【IT168 资讯】虽然距离谷歌Android N系统的发布已经有一段时间了,不过该系统目前仍旧属于内测阶段,短时间内仍难以迅速推广。另外也正是由于Android N正式…

python特性和属性_Python面向对象-类的特性及公私有属性 | 【韩涛博客】

构造函数 __init__ self.name name #属性,成员变量,字段 def sayhi() #方法,动态属性 公有属性 在类中直接定义的属性,大家都可以用 私有属性 __两个下划线定义,外部不可以访问,内部可以访问 class Role(o…

[Abp 源码分析]自动审计记录

点击上方蓝字关注我们0.简介Abp 框架为我们自带了审计日志功能,审计日志可以方便地查看每次请求接口所耗的时间,能够帮助我们快速定位到某些性能有问题的接口。除此之外,审计日志信息还包含有每次调用接口时客户端请求的参数信息,…

我的老公是枚码农

前两天看到一篇写程序员的爆文,虽然略显夸张,但也着实有趣。忽然想到身边人也是一枚码农,浑身上下也是浓厚的码农气息,遂也胡乱写了几笔,博君一笑,为了方便起见,就称其为“码农哥”。 1 码农哥还…

harmonyos con,鸿蒙HarmonyOS系统中的JS开发框架

HarmonyOS开源至今已经一个多月,源码托管在国内知名开源平台码云上,https://gitee.com/openharmony 。我最感兴趣的就是JS 框架 ace_lite_jsfwk,从名字中可以看出来这是一个非常轻量级的框架,官方介绍说是“轻量级 JS 核心开发框架…

.NET工资低?那肯定是你打开的方式不正确

点击上方蓝字关注我们因为工作的关系,本人总是会接触到一些刚踏入社会没多久的.NET开发小伙伴。尤其是年关将近,这时候想要跳槽的人特别多,所以收到一些小伙伴的迷茫求解。今天就拿其中一个来说,我们暂且称他为A同学吧。A同学是一…

10分钟读懂人工智能、机器学习到底有什么关系

文末彩蛋,错过哭一年。。。。 人工智能的浪潮正在席卷全球,诸多词汇时刻萦绕在我们耳边:人工智能(Artificial Intelligence)、机器学习(Machine Learning)。不少人对这些高频词汇的含义及其背后…

苏泊尔搭载华为鸿蒙系统,华为鸿蒙打算在一年内跨过生死线,拿下16%的市场份额...

原标题:华为鸿蒙打算在一年内跨过生死线,拿下16%的市场份额华为鸿蒙操作系统发布已经有一段时间了,这个操作系统直到上个月月底才开启了公测,很多用户都已经使用上了华为的这个鸿蒙操作系统。根据不少用户的反馈情况来看&#xff…

python获取系统时间函数_简单记录python的时间函数操作

1. time和datetime模块 import datetime,time 2. 获得当前时间 time.time() #获得当前时间,返回float型 time.localtime([float time]) #获得本地当前时间,返回time.struct_time类型 说明:struct_time是一个只读的9元组,其中参数命…